From f6c82dcaba9248dd93a15346ae2b2489c4fbae6c Mon Sep 17 00:00:00 2001 From: Fadi Khadra Date: Sun, 1 Oct 2023 22:05:25 +0200 Subject: [PATCH 01/87] [v2 mac] add fullscreen option to preference (#2953) * [v2 mac] add fullscreen option to preference * update changelog * replace space by tabs --- v2/internal/frontend/desktop/darwin/WailsContext.h | 1 + v2/internal/frontend/desktop/darwin/WailsContext.m | 6 ++++++ v2/internal/frontend/desktop/darwin/window.go | 4 ++++ v2/pkg/options/mac/preferences.go | 3 +++ website/docs/reference/options.mdx | 3 +++ website/src/pages/changelog.mdx | 1 + 6 files changed, 18 insertions(+) diff --git a/v2/internal/frontend/desktop/darwin/WailsContext.h b/v2/internal/frontend/desktop/darwin/WailsContext.h index fe66f3549dd..99b6ce2def7 100644 --- a/v2/internal/frontend/desktop/darwin/WailsContext.h +++ b/v2/internal/frontend/desktop/darwin/WailsContext.h @@ -58,6 +58,7 @@ struct Preferences { bool *tabFocusesLinks; bool *textInteractionEnabled; + bool *fullscreenEnabled; }; - (void) CreateWindow:(int)width :(int)height :(bool)frameless :(bool)resizable :(bool)fullscreen :(bool)fullSizeContent :(bool)hideTitleBar :(bool)titlebarAppearsTransparent :(bool)hideTitle :(bool)useToolbar :(bool)hideToolbarSeparator :(bool)webviewIsTransparent :(bool)hideWindowOnClose :(NSString *)appearance :(bool)windowIsTranslucent :(int)minWidth :(int)minHeight :(int)maxWidth :(int)maxHeight :(bool)fraudulentWebsiteWarningEnabled :(struct Preferences)preferences; diff --git a/v2/internal/frontend/desktop/darwin/WailsContext.m b/v2/internal/frontend/desktop/darwin/WailsContext.m index 1c84a576101..bf191e4725f 100644 --- a/v2/internal/frontend/desktop/darwin/WailsContext.m +++ b/v2/internal/frontend/desktop/darwin/WailsContext.m @@ -224,6 +224,12 @@ - (void) CreateWindow:(int)width :(int)height :(bool)frameless :(bool)resizable config.preferences.textInteractionEnabled = *preferences.textInteractionEnabled; } } + + if (@available(macOS 12.3, *)) { + if (preferences.fullscreenEnabled != NULL) { + config.preferences.elementFullscreenEnabled = *preferences.fullscreenEnabled; + } + } // [config.preferences setValue:[NSNumber numberWithBool:true] forKey:@"developerExtrasEnabled"]; diff --git a/v2/internal/frontend/desktop/darwin/window.go b/v2/internal/frontend/desktop/darwin/window.go index 803d04c1ef8..c0f62741920 100644 --- a/v2/internal/frontend/desktop/darwin/window.go +++ b/v2/internal/frontend/desktop/darwin/window.go @@ -95,6 +95,10 @@ func NewWindow(frontendOptions *options.App, debug bool, devtools bool) *Window if mac.Preferences.TextInteractionEnabled.IsSet() { preferences.textInteractionEnabled = bool2CboolPtr(mac.Preferences.TextInteractionEnabled.Get()) } + + if mac.Preferences.FullscreenEnabled.IsSet() { + preferences.fullscreenEnabled = bool2CboolPtr(mac.Preferences.FullscreenEnabled.Get()) + } } windowIsTranslucent = bool2Cint(mac.WindowIsTranslucent) diff --git a/v2/pkg/options/mac/preferences.go b/v2/pkg/options/mac/preferences.go index 2c690e7f718..dbc20c805a3 100644 --- a/v2/pkg/options/mac/preferences.go +++ b/v2/pkg/options/mac/preferences.go @@ -13,4 +13,7 @@ type Preferences struct { // A Boolean value that indicates whether to allow people to select or otherwise interact with text. // Set to true by default. TextInteractionEnabled u.Bool + // A Boolean value that indicates whether a web view can display content full screen. + // Set to false by default + FullscreenEnabled u.Bool } diff --git a/website/docs/reference/options.mdx b/website/docs/reference/options.mdx index 67e89707701..a2a19327725 100644 --- a/website/docs/reference/options.mdx +++ b/website/docs/reference/options.mdx @@ -812,6 +812,7 @@ You can specify the webview preferences. type Preferences struct { TabFocusesLinks u.Bool TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool } ``` @@ -819,6 +820,7 @@ type Preferences struct { | -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | | TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | | TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | Example: @@ -827,6 +829,7 @@ Mac: &mac.Options{ Preferences: &mac.Preferences{ TabFocusesLinks: mac.Enabled, TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, } } ``` diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index d49cf6d3a6e..ff94e77de49 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -19,6 +19,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) ### Changed From c0d2c0447c0679ee021ed5de61972bfb803b7ee8 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 4 Oct 2023 22:30:09 +1100 Subject: [PATCH 02/87] chore: update sponsors.svg (#2961) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 26 +++++++++++++++----------- 1 file changed, 15 insertions(+), 11 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index bee039e508a..5c258512197 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -113,46 +113,50 @@ text { Helpers - + - + - + - + - + - + - + - + - + - + - + + + + + From d0b0b5d594e01b83f7e5cf78ec40b3e7b00ec61e Mon Sep 17 00:00:00 2001 From: Nenba Jonathan Date: Fri, 6 Oct 2023 22:57:03 +0100 Subject: [PATCH 03/87] Add French README page (#2943) * add french readme to the projet * add french link to the other readme * fix some typo in the table of contents * fix markdown auto correct * added change in changelog.mdx --------- Co-authored-by: Lea Anthony --- README.es.md | 2 +- README.fr.md | 143 ++++++++++++++++++++++++++++++++ README.ja.md | 2 +- README.ko.md | 2 +- README.md | 2 +- README.pt-br.md | 2 +- README.ru.md | 4 +- README.zh-Hans.md | 2 +- website/src/pages/changelog.mdx | 1 + 9 files changed, 152 insertions(+), 8 deletions(-) create mode 100644 README.fr.md diff --git a/README.es.md b/README.es.md index e34267c1947..16cd67414d5 100644 --- a/README.es.md +++ b/README.es.md @@ -42,7 +42,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) +[Русский](README.ru.md) · [Francais](README.fr.md) diff --git a/README.fr.md b/README.fr.md new file mode 100644 index 00000000000..55379aa466b --- /dev/null +++ b/README.fr.md @@ -0,0 +1,143 @@ +

+
+

+ +

+ Créer des applications de bureau avec Go et les technologies Web. +
+
+ + GitHub + + + + + + Go Reference + + + CodeFactor + + + + + + Awesome + + + Discord + +
+ + Build + + + GitHub tag (latest SemVer pre-release) + +

+ +
+ + + +[English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · +[한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · +[Русский](README.ru.md) · [Francais](README.fr.md) + + + +
+ +## Sommaire + +- [Sommaire](#sommaire) +- [Introduction](#introduction) +- [Fonctionnalités](#fonctionnalités) + - [Feuille de route](#feuille-de-route) +- [Démarrage](#démarrage) +- [Les sponsors](#les-sponsors) +- [Foire aux questions](#foire-aux-questions) +- [Les étoiles au fil du temps](#les-étoiles-au-fil-du-temps) +- [Les contributeurs](#les-contributeurs) +- [License](#license) +- [Inspiration](#inspiration) + +## Introduction + +La méthode traditionnelle pour fournir des interfaces web aux programmes Go consiste à utiliser un serveur web intégré. Wails propose une approche différente : il offre la possibilité d'intégrer à la fois le code Go et une interface web dans un seul binaire. Des outils sont fournis pour vous faciliter la tâche en gérant la création, la compilation et le regroupement des projets. Il ne vous reste plus qu'à faire preuve de créativité! + +## Fonctionnalités + +- Utiliser Go pour le backend +- Utilisez n'importe quelle technologie frontend avec laquelle vous êtes déjà familier pour construire votre interface utilisateur. +- Créez rapidement des interfaces riches pour vos programmes Go à l'aide de modèles prédéfinis. +- Appeler facilement des méthodes Go à partir de Javascript +- Définitions Typescript auto-générées pour vos structures et méthodes Go +- Dialogues et menus natifs +- Prise en charge native des modes sombre et clair +- Prise en charge des effets modernes de translucidité et de "frosted window". +- Système d'événements unifié entre Go et Javascript +- Outil puissant pour générer et construire rapidement vos projets +- Multiplateforme +- Utilise des moteurs de rendu natifs - _pas de navigateur intégré_ ! + +### Feuille de route + +La feuille de route du projet peut être consultée [ici](https://github.com/wailsapp/wails/discussions/1484). Veuillez consulter avant d'ouvrir une demande d'amélioration. + +## Démarrage + +Les instructions d'installation se trouvent sur le site [site officiel](https://wails.io/docs/gettingstarted/installation). + +## Les sponsors + +Ce projet est soutenu par ces personnes aimables et entreprises: + + +

+ +

+ +## Foire aux questions + +- S'agit-il d'une alternative à Electron ? + + Cela dépend de vos besoins. Il est conçu pour permettre aux programmeurs Go de créer facilement des applications de bureau légères ou d'ajouter une interface à leurs applications existantes. Wails offre des éléments natifs tels que des menus et des boîtes de dialogue, il peut donc être considéré comme une alternative légère à electron. + +- À qui s'adresse ce projet ? + + Les programmeurs Go qui souhaitent intégrer une interface HTML/JS/CSS à leurs applications, sans avoir à créer un serveur et à ouvrir un navigateur pour l'afficher. + +- Pourquoi ce nom ?? + + Lorsque j'ai vu WebView, je me suis dit : "Ce que je veux vraiment, c'est un outil pour construire une application WebView, un peu comme Rails l'est pour Ruby". Au départ, il s'agissait donc d'un jeu de mots (Webview on Rails). Il se trouve que c'est aussi un homophone du nom anglais du [Pays](https://en.wikipedia.org/wiki/Wales) d'où je viens. Il s'est donc imposé. + +## Les étoiles au fil du temps + +[![Graphique de l'histoire des étoiles](https://api.star-history.com/svg?repos=wailsapp/wails&type=Date)](https://star-history.com/#wailsapp/wails&Date) + +## Les contributeurs + +La liste des contributeurs devient trop importante pour le readme ! Toutes les personnes extraordinaires qui ont contribué à ce projet ont leur propre page [ici](https://wails.io/credits#contributors). + +## License + +[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fwailsapp%2Fwails.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fwailsapp%2Fwails?ref=badge_large) + +## Inspiration + +Ce projet a été principalement codé sur les albums suivants : + +- [Manic Street Preachers - Resistance Is Futile](https://open.spotify.com/album/1R2rsEUqXjIvAbzM0yHrxA) +- [Manic Street Preachers - This Is My Truth, Tell Me Yours](https://open.spotify.com/album/4VzCL9kjhgGQeKCiojK1YN) +- [The Midnight - Endless Summer](https://open.spotify.com/album/4Krg8zvprquh7TVn9OxZn8) +- [Gary Newman - Savage (Songs from a Broken World)](https://open.spotify.com/album/3kMfsD07Q32HRWKRrpcexr) +- [Steve Vai - Passion & Warfare](https://open.spotify.com/album/0oL0OhrE2rYVns4IGj8h2m) +- [Ben Howard - Every Kingdom](https://open.spotify.com/album/1nJsbWm3Yy2DW1KIc1OKle) +- [Ben Howard - Noonday Dream](https://open.spotify.com/album/6astw05cTiXEc2OvyByaPs) +- [Adwaith - Melyn](https://open.spotify.com/album/2vBE40Rp60tl7rNqIZjaXM) +- [Gwidaith Hen Fran - Cedors Hen Wrach](https://open.spotify.com/album/3v2hrfNGINPLuDP0YDTOjm) +- [Metallica - Metallica](https://open.spotify.com/album/2Kh43m04B1UkVcpcRa1Zug) +- [Bloc Party - Silent Alarm](https://open.spotify.com/album/6SsIdN05HQg2GwYLfXuzLB) +- [Maxthor - Another World](https://open.spotify.com/album/3tklE2Fgw1hCIUstIwPBJF) +- [Alun Tan Lan - Y Distawrwydd](https://open.spotify.com/album/0c32OywcLpdJCWWMC6vB8v) diff --git a/README.ja.md b/README.ja.md index d134d31e908..375a543db28 100644 --- a/README.ja.md +++ b/README.ja.md @@ -44,7 +44,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) +[Русский](README.ru.md) · [Francais](README.fr.md) diff --git a/README.ko.md b/README.ko.md index 6285f93222e..4e72cffa705 100644 --- a/README.ko.md +++ b/README.ko.md @@ -44,7 +44,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) +[Русский](README.ru.md) · [Francais](README.fr.md) diff --git a/README.md b/README.md index 6277db3a84a..6d4427c03b9 100644 --- a/README.md +++ b/README.md @@ -42,7 +42,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) +[Русский](README.ru.md) · [Francais](README.fr.md) diff --git a/README.pt-br.md b/README.pt-br.md index a61e6211691..6cf9c7d621b 100644 --- a/README.pt-br.md +++ b/README.pt-br.md @@ -41,7 +41,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · -[한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) +[한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · [Francais](README.fr.md) diff --git a/README.ru.md b/README.ru.md index d6fcb1ee706..018c644fbd0 100644 --- a/README.ru.md +++ b/README.ru.md @@ -41,7 +41,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · -[한국어](README.ko.md) · [Español](README.es.md) · [Русский](README.ru.md) +[한국어](README.ko.md) · [Español](README.es.md) · [Русский](README.ru.md) · [Francais](README.fr.md) @@ -52,7 +52,7 @@ - [Содержание](#содержание) - [Вступление](#вступление) - [Особенности](#особенности) - - [Roadmap](#roadmap) + - [Roadmap](#roadmap) - [Быстрый старт](#быстрый-старт) - [Спонсоры](#спонсоры) - [FAQ](#faq) diff --git a/README.zh-Hans.md b/README.zh-Hans.md index bd1cb1dabe4..a26639a17cb 100644 --- a/README.zh-Hans.md +++ b/README.zh-Hans.md @@ -44,7 +44,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) +[Русский](README.ru.md) · [Francais](README.fr.md) diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index ff94e77de49..4923de6866f 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -20,6 +20,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) - Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) ### Changed From 1f4b578bcec9843713eee10c7bc7c6adbd35ad8f Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sun, 8 Oct 2023 20:42:57 +1100 Subject: [PATCH 04/87] Bump postcss from 8.4.21 to 8.4.31 in /website (#2963) Bumps [postcss](https://github.com/postcss/postcss) from 8.4.21 to 8.4.31. - [Release notes](https://github.com/postcss/postcss/releases) - [Changelog](https://github.com/postcss/postcss/blob/main/CHANGELOG.md) - [Commits](https://github.com/postcss/postcss/compare/8.4.21...8.4.31) --- updated-dependencies: - dependency-name: postcss dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- website/pnpm-lock.yaml | 322 ++++++++++++++++++++--------------------- 1 file changed, 157 insertions(+), 165 deletions(-) diff --git a/website/pnpm-lock.yaml b/website/pnpm-lock.yaml index 92c035da8f2..b172ea35076 100644 --- a/website/pnpm-lock.yaml +++ b/website/pnpm-lock.yaml @@ -1633,7 +1633,7 @@ packages: '@docusaurus/utils-validation': 2.4.0(@docusaurus/types@2.4.0)(@swc/core@1.3.7) '@slorber/static-site-generator-webpack-plugin': 4.0.7 '@svgr/webpack': 6.5.1 - autoprefixer: 10.4.14(postcss@8.4.21) + autoprefixer: 10.4.14(postcss@8.4.31) babel-loader: 8.3.0(@babel/core@7.21.4)(webpack@5.78.0) babel-plugin-dynamic-import-node: 2.3.3 boxen: 6.2.1 @@ -1647,7 +1647,7 @@ packages: core-js: 3.30.0 css-loader: 6.7.3(webpack@5.78.0) css-minimizer-webpack-plugin: 4.2.2(clean-css@5.3.2)(webpack@5.78.0) - cssnano: 5.1.15(postcss@8.4.21) + cssnano: 5.1.15(postcss@8.4.31) del: 6.1.1 detect-port: 1.5.1 escape-html: 1.0.3 @@ -1661,8 +1661,8 @@ packages: leven: 3.1.0 lodash: 4.17.21 mini-css-extract-plugin: 2.7.5(webpack@5.78.0) - postcss: 8.4.21 - postcss-loader: 7.2.4(@types/node@18.15.11)(postcss@8.4.21)(ts-node@10.9.1)(typescript@5.0.4)(webpack@5.78.0) + postcss: 8.4.31 + postcss-loader: 7.2.4(@types/node@18.15.11)(postcss@8.4.31)(ts-node@10.9.1)(typescript@5.0.4)(webpack@5.78.0) prompts: 2.4.2 react: 17.0.2 react-dev-utils: 12.0.1(typescript@5.0.4)(webpack@5.78.0) @@ -1712,9 +1712,9 @@ packages: resolution: {integrity: sha512-RmdiA3IpsLgZGXRzqnmTbGv43W4OD44PCo+6Q/aYjEM2V57vKCVqNzuafE94jv0z/PjHoXUrjr69SaRymBKYYw==} engines: {node: '>=16.14'} dependencies: - cssnano-preset-advanced: 5.3.10(postcss@8.4.21) - postcss: 8.4.21 - postcss-sort-media-queries: 4.3.0(postcss@8.4.21) + cssnano-preset-advanced: 5.3.10(postcss@8.4.31) + postcss: 8.4.31 + postcss-sort-media-queries: 4.3.0(postcss@8.4.31) tslib: 2.5.0 dev: false @@ -2165,7 +2165,7 @@ packages: infima: 0.2.0-alpha.43 lodash: 4.17.21 nprogress: 0.2.0 - postcss: 8.4.21 + postcss: 8.4.31 prism-react-renderer: 1.3.5(react@17.0.2) prismjs: 1.29.0 react: 17.0.2 @@ -2209,7 +2209,7 @@ packages: '@docusaurus/utils': 2.4.0(@docusaurus/types@2.4.0)(@swc/core@1.3.7) '@docusaurus/utils-common': 2.4.0(@docusaurus/types@2.4.0) '@types/history': 4.7.11 - '@types/react': 18.0.34 + '@types/react': 18.0.35 '@types/react-router-config': 5.0.7 clsx: 1.2.1 parse-numeric-range: 1.3.0 @@ -2303,7 +2303,7 @@ packages: react-dom: ^16.8.4 || ^17.0.0 dependencies: '@types/history': 4.7.11 - '@types/react': 18.0.34 + '@types/react': 18.0.35 commander: 5.1.0 joi: 17.9.1 react: 17.0.2 @@ -3080,14 +3080,6 @@ packages: '@types/react': 18.0.35 dev: false - /@types/react@18.0.34: - resolution: {integrity: sha512-NO1UO8941541CJl1BeOXi8a9dNKFK09Gnru5ZJqkm4Q3/WoQJtHvmwt0VX0SB9YCEwe7TfSSxDuaNmx6H2BAIQ==} - dependencies: - '@types/prop-types': 15.7.5 - '@types/scheduler': 0.16.3 - csstype: 3.1.2 - dev: false - /@types/react@18.0.35: resolution: {integrity: sha512-6Laome31HpetaIUGFWl1VQ3mdSImwxtFZ39rh059a1MNnKGqBpC88J6NJ8n/Is3Qx7CefDGLgf/KhN/sYCf7ag==} dependencies: @@ -3472,7 +3464,7 @@ packages: engines: {node: '>= 4.0.0'} dev: false - /autoprefixer@10.4.14(postcss@8.4.21): + /autoprefixer@10.4.14(postcss@8.4.31): resolution: {integrity: sha512-FQzyfOsTlwVzjHxKEqRIAdJx9niO6VCBCoEwax/VLSoQF29ggECcPuBqUMZ+u8jCZOPSy8b8/8KnuFbp0SaFZQ==} engines: {node: ^10 || ^12 || >=14} hasBin: true @@ -3484,7 +3476,7 @@ packages: fraction.js: 4.2.0 normalize-range: 0.1.2 picocolors: 1.0.0 - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false @@ -4163,13 +4155,13 @@ packages: engines: {node: '>=8'} dev: false - /css-declaration-sorter@6.4.0(postcss@8.4.21): + /css-declaration-sorter@6.4.0(postcss@8.4.31): resolution: {integrity: sha512-jDfsatwWMWN0MODAFuHszfjphEXfNw9JUAhmY4pLu3TyTU+ohUpsbVtbU+1MZn4a47D9kqh03i4eyOm+74+zew==} engines: {node: ^10 || ^12 || >=14} peerDependencies: postcss: ^8.0.9 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false /css-loader@6.7.3(webpack@5.78.0): @@ -4178,12 +4170,12 @@ packages: peerDependencies: webpack: ^5.0.0 dependencies: - icss-utils: 5.1.0(postcss@8.4.21) - postcss: 8.4.21 - postcss-modules-extract-imports: 3.0.0(postcss@8.4.21) - postcss-modules-local-by-default: 4.0.0(postcss@8.4.21) - postcss-modules-scope: 3.0.0(postcss@8.4.21) - postcss-modules-values: 4.0.0(postcss@8.4.21) + icss-utils: 5.1.0(postcss@8.4.31) + postcss: 8.4.31 + postcss-modules-extract-imports: 3.0.0(postcss@8.4.31) + postcss-modules-local-by-default: 4.0.0(postcss@8.4.31) + postcss-modules-scope: 3.0.0(postcss@8.4.31) + postcss-modules-values: 4.0.0(postcss@8.4.31) postcss-value-parser: 4.2.0 semver: 7.4.0 webpack: 5.78.0(@swc/core@1.3.7) @@ -4215,9 +4207,9 @@ packages: optional: true dependencies: clean-css: 5.3.2 - cssnano: 5.1.15(postcss@8.4.21) + cssnano: 5.1.15(postcss@8.4.31) jest-worker: 29.5.0 - postcss: 8.4.21 + postcss: 8.4.31 schema-utils: 4.0.0 serialize-javascript: 6.0.1 source-map: 0.6.1 @@ -4263,77 +4255,77 @@ packages: hasBin: true dev: false - /cssnano-preset-advanced@5.3.10(postcss@8.4.21): + /cssnano-preset-advanced@5.3.10(postcss@8.4.31): resolution: {integrity: sha512-fnYJyCS9jgMU+cmHO1rPSPf9axbQyD7iUhLO5Df6O4G+fKIOMps+ZbU0PdGFejFBBZ3Pftf18fn1eG7MAPUSWQ==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - autoprefixer: 10.4.14(postcss@8.4.21) - cssnano-preset-default: 5.2.14(postcss@8.4.21) - postcss: 8.4.21 - postcss-discard-unused: 5.1.0(postcss@8.4.21) - postcss-merge-idents: 5.1.1(postcss@8.4.21) - postcss-reduce-idents: 5.2.0(postcss@8.4.21) - postcss-zindex: 5.1.0(postcss@8.4.21) + autoprefixer: 10.4.14(postcss@8.4.31) + cssnano-preset-default: 5.2.14(postcss@8.4.31) + postcss: 8.4.31 + postcss-discard-unused: 5.1.0(postcss@8.4.31) + postcss-merge-idents: 5.1.1(postcss@8.4.31) + postcss-reduce-idents: 5.2.0(postcss@8.4.31) + postcss-zindex: 5.1.0(postcss@8.4.31) dev: false - /cssnano-preset-default@5.2.14(postcss@8.4.21): + /cssnano-preset-default@5.2.14(postcss@8.4.31): resolution: {integrity: sha512-t0SFesj/ZV2OTylqQVOrFgEh5uanxbO6ZAdeCrNsUQ6fVuXwYTxJPNAGvGTxHbD68ldIJNec7PyYZDBrfDQ+6A==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - css-declaration-sorter: 6.4.0(postcss@8.4.21) - cssnano-utils: 3.1.0(postcss@8.4.21) - postcss: 8.4.21 - postcss-calc: 8.2.4(postcss@8.4.21) - postcss-colormin: 5.3.1(postcss@8.4.21) - postcss-convert-values: 5.1.3(postcss@8.4.21) - postcss-discard-comments: 5.1.2(postcss@8.4.21) - postcss-discard-duplicates: 5.1.0(postcss@8.4.21) - postcss-discard-empty: 5.1.1(postcss@8.4.21) - postcss-discard-overridden: 5.1.0(postcss@8.4.21) - postcss-merge-longhand: 5.1.7(postcss@8.4.21) - postcss-merge-rules: 5.1.4(postcss@8.4.21) - postcss-minify-font-values: 5.1.0(postcss@8.4.21) - postcss-minify-gradients: 5.1.1(postcss@8.4.21) - postcss-minify-params: 5.1.4(postcss@8.4.21) - postcss-minify-selectors: 5.2.1(postcss@8.4.21) - postcss-normalize-charset: 5.1.0(postcss@8.4.21) - postcss-normalize-display-values: 5.1.0(postcss@8.4.21) - postcss-normalize-positions: 5.1.1(postcss@8.4.21) - postcss-normalize-repeat-style: 5.1.1(postcss@8.4.21) - postcss-normalize-string: 5.1.0(postcss@8.4.21) - postcss-normalize-timing-functions: 5.1.0(postcss@8.4.21) - postcss-normalize-unicode: 5.1.1(postcss@8.4.21) - postcss-normalize-url: 5.1.0(postcss@8.4.21) - postcss-normalize-whitespace: 5.1.1(postcss@8.4.21) - postcss-ordered-values: 5.1.3(postcss@8.4.21) - postcss-reduce-initial: 5.1.2(postcss@8.4.21) - postcss-reduce-transforms: 5.1.0(postcss@8.4.21) - postcss-svgo: 5.1.0(postcss@8.4.21) - postcss-unique-selectors: 5.1.1(postcss@8.4.21) - dev: false - - /cssnano-utils@3.1.0(postcss@8.4.21): + css-declaration-sorter: 6.4.0(postcss@8.4.31) + cssnano-utils: 3.1.0(postcss@8.4.31) + postcss: 8.4.31 + postcss-calc: 8.2.4(postcss@8.4.31) + postcss-colormin: 5.3.1(postcss@8.4.31) + postcss-convert-values: 5.1.3(postcss@8.4.31) + postcss-discard-comments: 5.1.2(postcss@8.4.31) + postcss-discard-duplicates: 5.1.0(postcss@8.4.31) + postcss-discard-empty: 5.1.1(postcss@8.4.31) + postcss-discard-overridden: 5.1.0(postcss@8.4.31) + postcss-merge-longhand: 5.1.7(postcss@8.4.31) + postcss-merge-rules: 5.1.4(postcss@8.4.31) + postcss-minify-font-values: 5.1.0(postcss@8.4.31) + postcss-minify-gradients: 5.1.1(postcss@8.4.31) + postcss-minify-params: 5.1.4(postcss@8.4.31) + postcss-minify-selectors: 5.2.1(postcss@8.4.31) + postcss-normalize-charset: 5.1.0(postcss@8.4.31) + postcss-normalize-display-values: 5.1.0(postcss@8.4.31) + postcss-normalize-positions: 5.1.1(postcss@8.4.31) + postcss-normalize-repeat-style: 5.1.1(postcss@8.4.31) + postcss-normalize-string: 5.1.0(postcss@8.4.31) + postcss-normalize-timing-functions: 5.1.0(postcss@8.4.31) + postcss-normalize-unicode: 5.1.1(postcss@8.4.31) + postcss-normalize-url: 5.1.0(postcss@8.4.31) + postcss-normalize-whitespace: 5.1.1(postcss@8.4.31) + postcss-ordered-values: 5.1.3(postcss@8.4.31) + postcss-reduce-initial: 5.1.2(postcss@8.4.31) + postcss-reduce-transforms: 5.1.0(postcss@8.4.31) + postcss-svgo: 5.1.0(postcss@8.4.31) + postcss-unique-selectors: 5.1.1(postcss@8.4.31) + dev: false + + /cssnano-utils@3.1.0(postcss@8.4.31): resolution: {integrity: sha512-JQNR19/YZhz4psLX/rQ9M83e3z2Wf/HdJbryzte4a3NSuafyp9w/I4U+hx5C2S9g41qlstH7DEWnZaaj83OuEA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /cssnano@5.1.15(postcss@8.4.21): + /cssnano@5.1.15(postcss@8.4.31): resolution: {integrity: sha512-j+BKgDcLDQA+eDifLx0EO4XSA56b7uut3BQFH+wbSaSTuGLuiyTa/wbRYthUXX8LC9mLg+WWKe8h+qJuwTAbHw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - cssnano-preset-default: 5.2.14(postcss@8.4.21) + cssnano-preset-default: 5.2.14(postcss@8.4.31) lilconfig: 2.1.0 - postcss: 8.4.21 + postcss: 8.4.31 yaml: 1.10.2 dev: false @@ -5486,13 +5478,13 @@ packages: safer-buffer: 2.1.2 dev: false - /icss-utils@5.1.0(postcss@8.4.21): + /icss-utils@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==} engines: {node: ^10 || ^12 || >= 14} peerDependencies: postcss: ^8.1.0 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false /ignore@5.2.4: @@ -6580,17 +6572,17 @@ packages: find-up: 3.0.0 dev: false - /postcss-calc@8.2.4(postcss@8.4.21): + /postcss-calc@8.2.4(postcss@8.4.31): resolution: {integrity: sha512-SmWMSJmB8MRnnULldx0lQIyhSNvuDl9HfrZkaqqE/WHAhToYsAvDq+yAsA/kIyINDszOp3Rh0GFoNuH5Ypsm3Q==} peerDependencies: postcss: ^8.2.2 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-selector-parser: 6.0.11 postcss-value-parser: 4.2.0 dev: false - /postcss-colormin@5.3.1(postcss@8.4.21): + /postcss-colormin@5.3.1(postcss@8.4.31): resolution: {integrity: sha512-UsWQG0AqTFQmpBegeLLc1+c3jIqBNB0zlDGRWR+dQ3pRKJL1oeMzyqmH3o2PIfn9MBdNrVPWhDbT769LxCTLJQ==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: @@ -6599,68 +6591,68 @@ packages: browserslist: 4.21.5 caniuse-api: 3.0.0 colord: 2.9.3 - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-convert-values@5.1.3(postcss@8.4.21): + /postcss-convert-values@5.1.3(postcss@8.4.31): resolution: {integrity: sha512-82pC1xkJZtcJEfiLw6UXnXVXScgtBrjlO5CBmuDQc+dlb88ZYheFsjTn40+zBVi3DkfF7iezO0nJUPLcJK3pvA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: browserslist: 4.21.5 - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-discard-comments@5.1.2(postcss@8.4.21): + /postcss-discard-comments@5.1.2(postcss@8.4.31): resolution: {integrity: sha512-+L8208OVbHVF2UQf1iDmRcbdjJkuBF6IS29yBDSiWUIzpYaAhtNl6JYnYm12FnkeCwQqF5LeklOu6rAqgfBZqQ==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-discard-duplicates@5.1.0(postcss@8.4.21): + /postcss-discard-duplicates@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-zmX3IoSI2aoenxHV6C7plngHWWhUOV3sP1T8y2ifzxzbtnuhk1EdPwm0S1bIUNaJ2eNbWeGLEwzw8huPD67aQw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-discard-empty@5.1.1(postcss@8.4.21): + /postcss-discard-empty@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-zPz4WljiSuLWsI0ir4Mcnr4qQQ5e1Ukc3i7UfE2XcrwKK2LIPIqE5jxMRxO6GbI3cv//ztXDsXwEWT3BHOGh3A==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-discard-overridden@5.1.0(postcss@8.4.21): + /postcss-discard-overridden@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-21nOL7RqWR1kasIVdKs8HNqQJhFxLsyRfAnUDm4Fe4t4mCWL9OJiHvlHPjcd8zc5Myu89b/7wZDnOSjFgeWRtw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-discard-unused@5.1.0(postcss@8.4.21): + /postcss-discard-unused@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-KwLWymI9hbwXmJa0dkrzpRbSJEh0vVUd7r8t0yOGPcfKzyJJxFM8kLyC5Ev9avji6nY95pOp1W6HqIrfT+0VGw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-selector-parser: 6.0.11 dev: false - /postcss-loader@7.2.4(@types/node@18.15.11)(postcss@8.4.21)(ts-node@10.9.1)(typescript@5.0.4)(webpack@5.78.0): + /postcss-loader@7.2.4(@types/node@18.15.11)(postcss@8.4.31)(ts-node@10.9.1)(typescript@5.0.4)(webpack@5.78.0): resolution: {integrity: sha512-F88rpxxNspo5hatIc+orYwZDtHFaVFOSIVAx+fBfJC1GmhWbVmPWtmg2gXKE1OxJbneOSGn8PWdIwsZFcruS+w==} engines: {node: '>= 14.15.0'} peerDependencies: @@ -6677,7 +6669,7 @@ packages: cosmiconfig: 8.1.3 cosmiconfig-typescript-loader: 4.3.0(@types/node@18.15.11)(cosmiconfig@8.1.3)(ts-node@10.9.1)(typescript@5.0.4) klona: 2.0.6 - postcss: 8.4.21 + postcss: 8.4.31 semver: 7.4.0 ts-node: 10.9.1(@swc/core@1.3.7)(@types/node@18.15.11)(typescript@5.0.4) typescript: 5.0.4 @@ -6686,29 +6678,29 @@ packages: - '@types/node' dev: false - /postcss-merge-idents@5.1.1(postcss@8.4.21): + /postcss-merge-idents@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-pCijL1TREiCoog5nQp7wUe+TUonA2tC2sQ54UGeMmryK3UFGIYKqDyjnqd6RcuI4znFn9hWSLNN8xKE/vWcUQw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - cssnano-utils: 3.1.0(postcss@8.4.21) - postcss: 8.4.21 + cssnano-utils: 3.1.0(postcss@8.4.31) + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-merge-longhand@5.1.7(postcss@8.4.21): + /postcss-merge-longhand@5.1.7(postcss@8.4.31): resolution: {integrity: sha512-YCI9gZB+PLNskrK0BB3/2OzPnGhPkBEwmwhfYk1ilBHYVAZB7/tkTHFBAnCrvBBOmeYyMYw3DMjT55SyxMBzjQ==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 - stylehacks: 5.1.1(postcss@8.4.21) + stylehacks: 5.1.1(postcss@8.4.31) dev: false - /postcss-merge-rules@5.1.4(postcss@8.4.21): + /postcss-merge-rules@5.1.4(postcss@8.4.31): resolution: {integrity: sha512-0R2IuYpgU93y9lhVbO/OylTtKMVcHb67zjWIfCiKR9rWL3GUk1677LAqD/BcHizukdZEjT8Ru3oHRoAYoJy44g==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: @@ -6716,209 +6708,209 @@ packages: dependencies: browserslist: 4.21.5 caniuse-api: 3.0.0 - cssnano-utils: 3.1.0(postcss@8.4.21) - postcss: 8.4.21 + cssnano-utils: 3.1.0(postcss@8.4.31) + postcss: 8.4.31 postcss-selector-parser: 6.0.11 dev: false - /postcss-minify-font-values@5.1.0(postcss@8.4.21): + /postcss-minify-font-values@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-el3mYTgx13ZAPPirSVsHqFzl+BBBDrXvbySvPGFnQcTI4iNslrPaFq4muTkLZmKlGk4gyFAYUBMH30+HurREyA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-minify-gradients@5.1.1(postcss@8.4.21): + /postcss-minify-gradients@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-VGvXMTpCEo4qHTNSa9A0a3D+dxGFZCYwR6Jokk+/3oB6flu2/PnPXAh2x7x52EkY5xlIHLm+Le8tJxe/7TNhzw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: colord: 2.9.3 - cssnano-utils: 3.1.0(postcss@8.4.21) - postcss: 8.4.21 + cssnano-utils: 3.1.0(postcss@8.4.31) + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-minify-params@5.1.4(postcss@8.4.21): + /postcss-minify-params@5.1.4(postcss@8.4.31): resolution: {integrity: sha512-+mePA3MgdmVmv6g+30rn57USjOGSAyuxUmkfiWpzalZ8aiBkdPYjXWtHuwJGm1v5Ojy0Z0LaSYhHaLJQB0P8Jw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: browserslist: 4.21.5 - cssnano-utils: 3.1.0(postcss@8.4.21) - postcss: 8.4.21 + cssnano-utils: 3.1.0(postcss@8.4.31) + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-minify-selectors@5.2.1(postcss@8.4.21): + /postcss-minify-selectors@5.2.1(postcss@8.4.31): resolution: {integrity: sha512-nPJu7OjZJTsVUmPdm2TcaiohIwxP+v8ha9NehQ2ye9szv4orirRU3SDdtUmKH+10nzn0bAyOXZ0UEr7OpvLehg==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-selector-parser: 6.0.11 dev: false - /postcss-modules-extract-imports@3.0.0(postcss@8.4.21): + /postcss-modules-extract-imports@3.0.0(postcss@8.4.31): resolution: {integrity: sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw==} engines: {node: ^10 || ^12 || >= 14} peerDependencies: postcss: ^8.1.0 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-modules-local-by-default@4.0.0(postcss@8.4.21): + /postcss-modules-local-by-default@4.0.0(postcss@8.4.31): resolution: {integrity: sha512-sT7ihtmGSF9yhm6ggikHdV0hlziDTX7oFoXtuVWeDd3hHObNkcHRo9V3yg7vCAY7cONyxJC/XXCmmiHHcvX7bQ==} engines: {node: ^10 || ^12 || >= 14} peerDependencies: postcss: ^8.1.0 dependencies: - icss-utils: 5.1.0(postcss@8.4.21) - postcss: 8.4.21 + icss-utils: 5.1.0(postcss@8.4.31) + postcss: 8.4.31 postcss-selector-parser: 6.0.11 postcss-value-parser: 4.2.0 dev: false - /postcss-modules-scope@3.0.0(postcss@8.4.21): + /postcss-modules-scope@3.0.0(postcss@8.4.31): resolution: {integrity: sha512-hncihwFA2yPath8oZ15PZqvWGkWf+XUfQgUGamS4LqoP1anQLOsOJw0vr7J7IwLpoY9fatA2qiGUGmuZL0Iqlg==} engines: {node: ^10 || ^12 || >= 14} peerDependencies: postcss: ^8.1.0 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-selector-parser: 6.0.11 dev: false - /postcss-modules-values@4.0.0(postcss@8.4.21): + /postcss-modules-values@4.0.0(postcss@8.4.31): resolution: {integrity: sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==} engines: {node: ^10 || ^12 || >= 14} peerDependencies: postcss: ^8.1.0 dependencies: - icss-utils: 5.1.0(postcss@8.4.21) - postcss: 8.4.21 + icss-utils: 5.1.0(postcss@8.4.31) + postcss: 8.4.31 dev: false - /postcss-normalize-charset@5.1.0(postcss@8.4.21): + /postcss-normalize-charset@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-mSgUJ+pd/ldRGVx26p2wz9dNZ7ji6Pn8VWBajMXFf8jk7vUoSrZ2lt/wZR7DtlZYKesmZI680qjr2CeFF2fbUg==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-normalize-display-values@5.1.0(postcss@8.4.21): + /postcss-normalize-display-values@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-WP4KIM4o2dazQXWmFaqMmcvsKmhdINFblgSeRgn8BJ6vxaMyaJkwAzpPpuvSIoG/rmX3M+IrRZEz2H0glrQNEA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-positions@5.1.1(postcss@8.4.21): + /postcss-normalize-positions@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-6UpCb0G4eofTCQLFVuI3EVNZzBNPiIKcA1AKVka+31fTVySphr3VUgAIULBhxZkKgwLImhzMR2Bw1ORK+37INg==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-repeat-style@5.1.1(postcss@8.4.21): + /postcss-normalize-repeat-style@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-mFpLspGWkQtBcWIRFLmewo8aC3ImN2i/J3v8YCFUwDnPu3Xz4rLohDO26lGjwNsQxB3YF0KKRwspGzE2JEuS0g==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-string@5.1.0(postcss@8.4.21): + /postcss-normalize-string@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-oYiIJOf4T9T1N4i+abeIc7Vgm/xPCGih4bZz5Nm0/ARVJ7K6xrDlLwvwqOydvyL3RHNf8qZk6vo3aatiw/go3w==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-timing-functions@5.1.0(postcss@8.4.21): + /postcss-normalize-timing-functions@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-DOEkzJ4SAXv5xkHl0Wa9cZLF3WCBhF3o1SKVxKQAa+0pYKlueTpCgvkFAHfk+Y64ezX9+nITGrDZeVGgITJXjg==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-unicode@5.1.1(postcss@8.4.21): + /postcss-normalize-unicode@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-qnCL5jzkNUmKVhZoENp1mJiGNPcsJCs1aaRmURmeJGES23Z/ajaln+EPTD+rBeNkSryI+2WTdW+lwcVdOikrpA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: browserslist: 4.21.5 - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-url@5.1.0(postcss@8.4.21): + /postcss-normalize-url@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-5upGeDO+PVthOxSmds43ZeMeZfKH+/DKgGRD7TElkkyS46JXAUhMzIKiCa7BabPeIy3AQcTkXwVVN7DbqsiCew==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: normalize-url: 6.1.0 - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-normalize-whitespace@5.1.1(postcss@8.4.21): + /postcss-normalize-whitespace@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-83ZJ4t3NUDETIHTa3uEg6asWjSBYL5EdkVB0sDncx9ERzOKBVJIUeDO9RyA9Zwtig8El1d79HBp0JEi8wvGQnA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-ordered-values@5.1.3(postcss@8.4.21): + /postcss-ordered-values@5.1.3(postcss@8.4.31): resolution: {integrity: sha512-9UO79VUhPwEkzbb3RNpqqghc6lcYej1aveQteWY+4POIwlqkYE21HKWaLDF6lWNuqCobEAyTovVhtI32Rbv2RQ==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - cssnano-utils: 3.1.0(postcss@8.4.21) - postcss: 8.4.21 + cssnano-utils: 3.1.0(postcss@8.4.31) + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-reduce-idents@5.2.0(postcss@8.4.21): + /postcss-reduce-idents@5.2.0(postcss@8.4.31): resolution: {integrity: sha512-BTrLjICoSB6gxbc58D5mdBK8OhXRDqud/zodYfdSi52qvDHdMwk+9kB9xsM8yJThH/sZU5A6QVSmMmaN001gIg==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false - /postcss-reduce-initial@5.1.2(postcss@8.4.21): + /postcss-reduce-initial@5.1.2(postcss@8.4.31): resolution: {integrity: sha512-dE/y2XRaqAi6OvjzD22pjTUQ8eOfc6m/natGHgKFBK9DxFmIm69YmaRVQrGgFlEfc1HePIurY0TmDeROK05rIg==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: @@ -6926,16 +6918,16 @@ packages: dependencies: browserslist: 4.21.5 caniuse-api: 3.0.0 - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss-reduce-transforms@5.1.0(postcss@8.4.21): + /postcss-reduce-transforms@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-2fbdbmgir5AvpW9RLtdONx1QoYG2/EtqpNQbFASDlixBbAYuTcJ0dECwlqNqH7VbaUnEnh8SrxOe2sRIn24XyQ==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 dev: false @@ -6947,34 +6939,34 @@ packages: util-deprecate: 1.0.2 dev: false - /postcss-sort-media-queries@4.3.0(postcss@8.4.21): + /postcss-sort-media-queries@4.3.0(postcss@8.4.31): resolution: {integrity: sha512-jAl8gJM2DvuIJiI9sL1CuiHtKM4s5aEIomkU8G3LFvbP+p8i7Sz8VV63uieTgoewGqKbi+hxBTiOKJlB35upCg==} engines: {node: '>=10.0.0'} peerDependencies: postcss: ^8.4.16 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 sort-css-media-queries: 2.1.0 dev: false - /postcss-svgo@5.1.0(postcss@8.4.21): + /postcss-svgo@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-D75KsH1zm5ZrHyxPakAxJWtkyXew5qwS70v56exwvw542d9CRtTo78K0WeFxZB4G7JXKKMbEZtZayTGdIky/eA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-value-parser: 4.2.0 svgo: 2.8.0 dev: false - /postcss-unique-selectors@5.1.1(postcss@8.4.21): + /postcss-unique-selectors@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-5JiODlELrz8L2HwxfPnhOWZYWDxVHWL83ufOv84NrcgipI7TaeRsatAhK4Tr2/ZiYldpK/wBvw5BD3qfaK96GA==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 postcss-selector-parser: 6.0.11 dev: false @@ -6982,17 +6974,17 @@ packages: resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} dev: false - /postcss-zindex@5.1.0(postcss@8.4.21): + /postcss-zindex@5.1.0(postcss@8.4.31): resolution: {integrity: sha512-fgFMf0OtVSBR1va1JNHYgMxYk73yhn/qb4uQDq1DLGYolz8gHCyr/sesEuGUaYs58E3ZJRcpoGuPVoB7Meiq9A==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: - postcss: 8.4.21 + postcss: 8.4.31 dev: false - /postcss@8.4.21: - resolution: {integrity: sha512-tP7u/Sn/dVxK2NnruI4H9BG+x+Wxz6oeZ1cJ8P6G/PZY0IKk4k/63TDsQf2kQq3+qoJeLm2kIBUNlZe3zgb4Zg==} + /postcss@8.4.31: + resolution: {integrity: sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==} engines: {node: ^10 || ^12 || >=14} dependencies: nanoid: 3.3.6 @@ -7594,7 +7586,7 @@ packages: dependencies: find-up: 5.0.0 picocolors: 1.0.0 - postcss: 8.4.21 + postcss: 8.4.31 strip-json-comments: 3.1.1 dev: false @@ -8054,14 +8046,14 @@ packages: inline-style-parser: 0.1.1 dev: false - /stylehacks@5.1.1(postcss@8.4.21): + /stylehacks@5.1.1(postcss@8.4.31): resolution: {integrity: sha512-sBpcd5Hx7G6seo7b1LkpttvTz7ikD0LlH5RmdcBNb6fFR0Fl7LQwHDFr300q4cwUqi+IYrFGmsIHieMBfnN/Bw==} engines: {node: ^10 || ^12 || >=14.0} peerDependencies: postcss: ^8.2.15 dependencies: browserslist: 4.21.5 - postcss: 8.4.21 + postcss: 8.4.31 postcss-selector-parser: 6.0.11 dev: false From 4257decf307e3d084677c96c9425804bc591655f Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sun, 8 Oct 2023 20:43:09 +1100 Subject: [PATCH 05/87] Bump postcss from 8.4.18 to 8.4.31 in /v2/internal/frontend/runtime (#2962) Bumps [postcss](https://github.com/postcss/postcss) from 8.4.18 to 8.4.31. - [Release notes](https://github.com/postcss/postcss/releases) - [Changelog](https://github.com/postcss/postcss/blob/main/CHANGELOG.md) - [Commits](https://github.com/postcss/postcss/compare/8.4.18...8.4.31) --- updated-dependencies: - dependency-name: postcss dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .../frontend/runtime/package-lock.json | 38 ++++++++++++------- 1 file changed, 24 insertions(+), 14 deletions(-) diff --git a/v2/internal/frontend/runtime/package-lock.json b/v2/internal/frontend/runtime/package-lock.json index f0b59c22773..f6535549db6 100644 --- a/v2/internal/frontend/runtime/package-lock.json +++ b/v2/internal/frontend/runtime/package-lock.json @@ -1211,10 +1211,16 @@ "dev": true }, "node_modules/nanoid": { - "version": "3.3.4", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz", - "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==", + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.6.tgz", + "integrity": "sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==", "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], "bin": { "nanoid": "bin/nanoid.cjs" }, @@ -1404,9 +1410,9 @@ } }, "node_modules/postcss": { - "version": "8.4.18", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.18.tgz", - "integrity": "sha512-Wi8mWhncLJm11GATDaQKobXSNEYGUHeQLiQqDFG1qQ5UTDPTEvKw0Xt5NsTpktGTwLps3ByrWsBrG0rB8YQ9oA==", + "version": "8.4.31", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz", + "integrity": "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==", "dev": true, "funding": [ { @@ -1416,10 +1422,14 @@ { "type": "tidelift", "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" } ], "dependencies": { - "nanoid": "^3.3.4", + "nanoid": "^3.3.6", "picocolors": "^1.0.0", "source-map-js": "^1.0.2" }, @@ -2829,9 +2839,9 @@ "dev": true }, "nanoid": { - "version": "3.3.4", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz", - "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==", + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.6.tgz", + "integrity": "sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==", "dev": true }, "nice-try": { @@ -2964,12 +2974,12 @@ "dev": true }, "postcss": { - "version": "8.4.18", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.18.tgz", - "integrity": "sha512-Wi8mWhncLJm11GATDaQKobXSNEYGUHeQLiQqDFG1qQ5UTDPTEvKw0Xt5NsTpktGTwLps3ByrWsBrG0rB8YQ9oA==", + "version": "8.4.31", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz", + "integrity": "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==", "dev": true, "requires": { - "nanoid": "^3.3.4", + "nanoid": "^3.3.6", "picocolors": "^1.0.0", "source-map-js": "^1.0.2" } From 79db5a4657a0488511c8c6e0511eac006789cbb3 Mon Sep 17 00:00:00 2001 From: Sean Date: Sun, 8 Oct 2023 02:45:13 -0700 Subject: [PATCH 06/87] docs: troubleshooting tips for cross-account admin perms and "Microsoft Edge can't read or write to its data directory" error (#2931) Co-authored-by: Lea Anthony --- website/docs/guides/troubleshooting.mdx | 148 ++++++++++++++++++++++++ 1 file changed, 148 insertions(+) diff --git a/website/docs/guides/troubleshooting.mdx b/website/docs/guides/troubleshooting.mdx index 1ce18f2d5f2..cf22c99070f 100644 --- a/website/docs/guides/troubleshooting.mdx +++ b/website/docs/guides/troubleshooting.mdx @@ -203,6 +203,153 @@ you can make the webview background transparent using the following config: }) ``` +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + ## WebView2 installation succeeded, but the wails doctor command shows that it is not installed If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the @@ -213,3 +360,4 @@ Source: https://github.com/wailsapp/wails/issues/2917 ## WebVie2wProcess failed with kind If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + From 90c9186dc67941aa800d58d79ce2e281ed507d64 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Mon, 9 Oct 2023 19:15:14 +1100 Subject: [PATCH 07/87] chore: update sponsors.svg (#2967) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 20 ++++++++++++++------ 1 file changed, 14 insertions(+), 6 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 5c258512197..6136a8a3eb8 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -39,16 +39,20 @@ text { Iain Buying Breakfast - tc-hib - + tc-hib + - Tai Groot - + Tai Groot + - Michael - + Michael + + + + Tom Wu + Bironou @@ -61,6 +65,10 @@ text { igops + + + Prashanth + Buying Coffee From a774459cd742e365e73d2b443018e911a0d32198 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=B5=84=E6=B7=B1=E8=8F=9C=E9=B8=9F?= Date: Thu, 12 Oct 2023 20:20:11 +0800 Subject: [PATCH 08/87] Update templates.mdx (#2976) Add a template for Wails with Vite + Vue + TypeScript + NavieUI + Pinia. --- website/versioned_docs/version-v2.6.0/community/templates.mdx | 1 + 1 file changed, 1 insertion(+) diff --git a/website/versioned_docs/version-v2.6.0/community/templates.mdx b/website/versioned_docs/version-v2.6.0/community/templates.mdx index e1afd456849..c25af548eaf 100644 --- a/website/versioned_docs/version-v2.6.0/community/templates.mdx +++ b/website/versioned_docs/version-v2.6.0/community/templates.mdx @@ -29,6 +29,7 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for - [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - A template using JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) - [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - A template using TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, Composition API with <script setup>) - [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Wails template based on Naive UI (A Vue 3 Component Library) +- [wails-vite-vue-ts-navie-pinia](https://github.com/id88/wails-vite-vue-ts-navie-pinia) - A template for Wails with Vite + Vue + TypeScript + NavieUI + Pinia ## Angular From 763783389850c8f92785db7b6c20921458f0719c Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sat, 14 Oct 2023 08:34:15 +1100 Subject: [PATCH 09/87] chore: update sponsors.svg (#2974) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 34 ++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 6136a8a3eb8..29b0c226336 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -114,57 +114,57 @@ text { - + - -Helpers - - + - + + + +Helpers - + - + - + - + - + - + - + - + - + - + - + From d4500e89e28907b96d46bac720f540ba3c4e90b0 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sat, 14 Oct 2023 11:43:03 +1100 Subject: [PATCH 10/87] Bump golang.org/x/net from 0.10.0 to 0.17.0 in /v2 (#2979) Bumps [golang.org/x/net](https://github.com/golang/net) from 0.10.0 to 0.17.0. - [Commits](https://github.com/golang/net/compare/v0.10.0...v0.17.0) --- updated-dependencies: - dependency-name: golang.org/x/net dependency-type: direct:production ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- v2/go.mod | 10 +++++----- v2/go.sum | 22 ++++++++++------------ 2 files changed, 15 insertions(+), 17 deletions(-) diff --git a/v2/go.mod b/v2/go.mod index 60bf75abc1e..308d2476d26 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -38,8 +38,8 @@ require ( github.com/wailsapp/mimetype v1.4.1 github.com/wzshiming/ctc v1.2.3 golang.org/x/mod v0.12.0 - golang.org/x/net v0.10.0 - golang.org/x/sys v0.8.0 + golang.org/x/net v0.17.0 + golang.org/x/sys v0.13.0 golang.org/x/tools v0.6.0 ) @@ -87,11 +87,11 @@ require ( github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778 // indirect github.com/yuin/goldmark v1.4.13 // indirect github.com/yuin/goldmark-emoji v1.0.1 // indirect - golang.org/x/crypto v0.9.0 // indirect + golang.org/x/crypto v0.14.0 // indirect golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 // indirect golang.org/x/image v0.5.0 // indirect - golang.org/x/term v0.8.0 // indirect - golang.org/x/text v0.9.0 // indirect + golang.org/x/term v0.13.0 // indirect + golang.org/x/text v0.13.0 // indirect gopkg.in/warnings.v0 v0.1.2 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/v2/go.sum b/v2/go.sum index 7769626feec..7c13330926d 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -119,8 +119,6 @@ github.com/leaanthony/gosod v1.0.3/go.mod h1:BJ2J+oHsQIyIQpnLPjnqFGTMnOZXDbvWtRC github.com/leaanthony/slicer v1.5.0/go.mod h1:FwrApmf8gOrpzEWM2J/9Lh79tyq8KTX5AzRtwV7m4AY= github.com/leaanthony/slicer v1.6.0 h1:1RFP5uiPJvT93TAHi+ipd3NACobkW53yUiBqZheE/Js= github.com/leaanthony/slicer v1.6.0/go.mod h1:o/Iz29g7LN0GqH3aMjWAe90381nyZlDNquK+mtH2Fj8= -github.com/leaanthony/u v1.0.1 h1:tJLpf9EsuCgSB02ojRxg8KRqMgRN6mCTvFwI55kxRFE= -github.com/leaanthony/u v1.0.1/go.mod h1:9+o6hejoRljvZ3BzdYlVL0JYCwtnAsVuN9pVTQcaRfI= github.com/leaanthony/u v1.1.0 h1:2n0d2BwPVXSUq5yhe8lJPHdxevE2qK5G99PMStMZMaI= github.com/leaanthony/u v1.1.0/go.mod h1:9+o6hejoRljvZ3BzdYlVL0JYCwtnAsVuN9pVTQcaRfI= github.com/leaanthony/winicon v1.0.0 h1:ZNt5U5dY71oEoKZ97UVwJRT4e+5xo5o/ieKuHuk8NqQ= @@ -240,8 +238,8 @@ golang.org/x/crypto v0.0.0-20190219172222-a4c6cb3142f2/go.mod h1:6SG95UA2DQfeDnf golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= -golang.org/x/crypto v0.9.0 h1:LF6fAI+IutBocDJ2OT0Q1g8plpYljMZ4+lty+dsqw3g= -golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0= +golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= +golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4= golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 h1:k/i9J1pBpvlfR+9QsetwPyERsqu1GIbi967PQMq3Ivc= golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w= golang.org/x/image v0.0.0-20200430140353-33d19683fad8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= @@ -257,8 +255,8 @@ golang.org/x/net v0.0.0-20210326060303-6b1517762897/go.mod h1:uSPa2vr4CLtc/ILN5o golang.org/x/net v0.0.0-20210505024714-0287a6fb4125/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210614182718-04defd469f4e/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= -golang.org/x/net v0.10.0 h1:X2//UzNDwYmtCLn7To6G58Wr6f5ahEAQgKNzv9Y951M= -golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= +golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= +golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o= @@ -288,21 +286,21 @@ golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.8.0 h1:EBmGv8NaZBZTWvrbjNoL6HVt+IVy3QDQpJs7VRIw3tU= -golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE= +golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210220032956-6a3ed077a48d/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= -golang.org/x/term v0.8.0 h1:n5xxQn2i3PC0yLAbjTpNT85q/Kgzcr2gIoX9OrJUols= -golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= +golang.org/x/term v0.13.0 h1:bb+I9cTfFazGW51MZqBVmZy7+JEJMouUHTUSKVQLBek= +golang.org/x/term v0.13.0/go.mod h1:LTmsnFJwVN6bCy1rVCoS+qHT1HhALEFxKncY3WNNh4U= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= -golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE= -golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= +golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= From 45c6f731323a62fcbad5a28c2720c1e5c9f3ebf4 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sat, 14 Oct 2023 11:43:16 +1100 Subject: [PATCH 11/87] Bump golang.org/x/net in /v2/internal/staticanalysis/test/standard (#2981) Bumps [golang.org/x/net](https://github.com/golang/net) from 0.7.0 to 0.17.0. - [Commits](https://github.com/golang/net/compare/v0.7.0...v0.17.0) --- updated-dependencies: - dependency-name: golang.org/x/net dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- v2/internal/staticanalysis/test/standard/go.mod | 8 ++++---- v2/internal/staticanalysis/test/standard/go.sum | 16 ++++++++-------- 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/v2/internal/staticanalysis/test/standard/go.mod b/v2/internal/staticanalysis/test/standard/go.mod index 56184c62ded..0c7e2cc752c 100644 --- a/v2/internal/staticanalysis/test/standard/go.mod +++ b/v2/internal/staticanalysis/test/standard/go.mod @@ -25,11 +25,11 @@ require ( github.com/valyala/bytebufferpool v1.0.0 // indirect github.com/valyala/fasttemplate v1.2.1 // indirect github.com/wailsapp/mimetype v1.4.1 // indirect - golang.org/x/crypto v0.0.0-20221012134737-56aed061732a // indirect + golang.org/x/crypto v0.14.0 // indirect golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 // indirect - golang.org/x/net v0.7.0 // indirect - golang.org/x/sys v0.5.0 // indirect - golang.org/x/text v0.7.0 // indirect + golang.org/x/net v0.17.0 // indirect + golang.org/x/sys v0.13.0 // indirect + golang.org/x/text v0.13.0 // indirect ) // replace github.com/wailsapp/wails/v2 v2.0.0 => C:\Users\leaan diff --git a/v2/internal/staticanalysis/test/standard/go.sum b/v2/internal/staticanalysis/test/standard/go.sum index 54b3fbb032f..d2c8dd68796 100644 --- a/v2/internal/staticanalysis/test/standard/go.sum +++ b/v2/internal/staticanalysis/test/standard/go.sum @@ -57,13 +57,13 @@ github.com/wailsapp/mimetype v1.4.1 h1:pQN9ycO7uo4vsUUuPeHEYoUkLVkaRntMnHJxVwYhw github.com/wailsapp/mimetype v1.4.1/go.mod h1:9aV5k31bBOv5z6u+QP8TltzvNGJPmNJD4XlAL3U+j3o= github.com/wailsapp/wails/v2 v2.3.1 h1:ZJz+pyIBKyASkgO8JO31NuHO1gTTHmvwiHYHwei1CqM= github.com/wailsapp/wails/v2 v2.3.1/go.mod h1:zlNLI0E2c2qA6miiuAHtp0Bac8FaGH0tlhA19OssR/8= -golang.org/x/crypto v0.0.0-20221012134737-56aed061732a h1:NmSIgad6KjE6VvHciPZuNRTKxGhlPfD6OA87W/PLkqg= -golang.org/x/crypto v0.0.0-20221012134737-56aed061732a/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= +golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= +golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4= golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 h1:3MTrJm4PyNL9NBqvYDSj3DHl46qQakyfqfWo4jgfaEM= golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE= golang.org/x/net v0.0.0-20210505024714-0287a6fb4125/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.7.0 h1:rJrUqqhjsgNp7KqAIc25s9pZnjU7TUcSY7HcVZjdn1g= -golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= +golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= +golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200810151505-1b9f1253b3ed/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -73,12 +73,12 @@ golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211103235746-7861aae1554b/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.5.0 h1:MUK/U/4lj1t1oPg0HfuXDN/Z1wv31ZJ/YcPiGccS4DU= -golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE= +golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.7.0 h1:4BRB4x83lYWy72KwLD/qYDuTu7q9PjSagHvijDw7cLo= -golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= +golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= From 8efa4698cf8a48267c802fd165c1768e0f877106 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sat, 14 Oct 2023 11:43:38 +1100 Subject: [PATCH 12/87] Bump golang.org/x/net from 0.10.0 to 0.17.0 in /v2/examples/customlayout (#2980) Bumps [golang.org/x/net](https://github.com/golang/net) from 0.10.0 to 0.17.0. - [Commits](https://github.com/golang/net/compare/v0.10.0...v0.17.0) --- updated-dependencies: - dependency-name: golang.org/x/net dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- v2/examples/customlayout/go.mod | 11 +++--- v2/examples/customlayout/go.sum | 61 +++++++++++++++------------------ 2 files changed, 34 insertions(+), 38 deletions(-) diff --git a/v2/examples/customlayout/go.mod b/v2/examples/customlayout/go.mod index fcb4ce4fdc7..ccec36a7604 100644 --- a/v2/examples/customlayout/go.mod +++ b/v2/examples/customlayout/go.mod @@ -14,20 +14,23 @@ require ( github.com/leaanthony/go-ansi-parser v1.6.0 // indirect github.com/leaanthony/gosod v1.0.3 // indirect github.com/leaanthony/slicer v1.6.0 // indirect + github.com/leaanthony/u v1.1.0 // indirect github.com/mattn/go-colorable v0.1.13 // indirect github.com/mattn/go-isatty v0.0.19 // indirect github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8 // indirect github.com/pkg/errors v0.9.1 // indirect + github.com/rivo/uniseg v0.4.4 // indirect github.com/samber/lo v1.38.1 // indirect github.com/tkrajina/go-reflector v0.5.6 // indirect github.com/valyala/bytebufferpool v1.0.0 // indirect github.com/valyala/fasttemplate v1.2.2 // indirect + github.com/wailsapp/go-webview2 v1.0.7 // indirect github.com/wailsapp/mimetype v1.4.1 // indirect - golang.org/x/crypto v0.9.0 // indirect + golang.org/x/crypto v0.14.0 // indirect golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 // indirect - golang.org/x/net v0.10.0 // indirect - golang.org/x/sys v0.8.0 // indirect - golang.org/x/text v0.9.0 // indirect + golang.org/x/net v0.17.0 // indirect + golang.org/x/sys v0.13.0 // indirect + golang.org/x/text v0.13.0 // indirect ) replace github.com/wailsapp/wails/v2 v2.1.0 => ../.. diff --git a/v2/examples/customlayout/go.sum b/v2/examples/customlayout/go.sum index b00ead2f35a..fb0946bc54c 100644 --- a/v2/examples/customlayout/go.sum +++ b/v2/examples/customlayout/go.sum @@ -5,71 +5,66 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= -github.com/google/uuid v1.1.2 h1:EVhdT+1Kseyi1/pUmXKaFxYsDNy9RQYkMWRH68J/W7Y= -github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/jchv/go-winloader v0.0.0-20210711035445-715c2860da7e h1:Q3+PugElBCf4PFpxhErSzU3/PY5sFL5Z6rfv4AbGAck= github.com/jchv/go-winloader v0.0.0-20210711035445-715c2860da7e/go.mod h1:alcuEEnZsY1WQsagKhZDsoPCRoOijYqhZvPwLG0kzVs= -github.com/labstack/echo/v4 v4.9.0 h1:wPOF1CE6gvt/kmbMR4dGzWvHMPT+sAEUJOwOTtvITVY= -github.com/labstack/echo/v4 v4.9.0/go.mod h1:xkCDAdFCIf8jsFQ5NnbK7oqaF/yU1A1X20Ltm0OvSks= +github.com/labstack/echo/v4 v4.10.2 h1:n1jAhnq/elIFTHr1EYpiYtyKgx4RW9ccVgkqByZaN2M= github.com/labstack/echo/v4 v4.10.2/go.mod h1:OEyqf2//K1DFdE57vw2DRgWY0M7s65IVQO2FzvI4J5k= -github.com/labstack/gommon v0.3.1 h1:OomWaJXm7xR6L1HmEtGyQf26TEn7V6X88mktX9kee9o= -github.com/labstack/gommon v0.3.1/go.mod h1:uW6kP17uPlLJsD3ijUYn3/M5bAxtlZhMI6m3MFxTMTM= +github.com/labstack/gommon v0.4.0 h1:y7cvthEAEbU0yHOf4axH8ZG2NH8knB9iNSoTO8dyIk8= github.com/labstack/gommon v0.4.0/go.mod h1:uW6kP17uPlLJsD3ijUYn3/M5bAxtlZhMI6m3MFxTMTM= github.com/leaanthony/debme v1.2.1 h1:9Tgwf+kjcrbMQ4WnPcEIUcQuIZYqdWftzZkBr+i/oOc= github.com/leaanthony/debme v1.2.1/go.mod h1:3V+sCm5tYAgQymvSOfYQ5Xx2JCr+OXiD9Jkw3otUjiA= -github.com/leaanthony/go-ansi-parser v1.0.1 h1:97v6c5kYppVsbScf4r/VZdXyQ21KQIfeQOk2DgKxGG4= -github.com/leaanthony/go-ansi-parser v1.0.1/go.mod h1:7arTzgVI47srICYhvgUV4CGd063sGEeoSlych5yeSPM= +github.com/leaanthony/go-ansi-parser v1.6.0 h1:T8TuMhFB6TUMIUm0oRrSbgJudTFw9csT3ZK09w0t4Pg= github.com/leaanthony/go-ansi-parser v1.6.0/go.mod h1:+vva/2y4alzVmmIEpk9QDhA7vLC5zKDTRwfZGOp3IWU= github.com/leaanthony/gosod v1.0.3 h1:Fnt+/B6NjQOVuCWOKYRREZnjGyvg+mEhd1nkkA04aTQ= github.com/leaanthony/gosod v1.0.3/go.mod h1:BJ2J+oHsQIyIQpnLPjnqFGTMnOZXDbvWtRCSG7jGxs4= -github.com/leaanthony/slicer v1.5.0 h1:aHYTN8xbCCLxJmkNKiLB6tgcMARl4eWmH9/F+S/0HtY= github.com/leaanthony/slicer v1.5.0/go.mod h1:FwrApmf8gOrpzEWM2J/9Lh79tyq8KTX5AzRtwV7m4AY= +github.com/leaanthony/slicer v1.6.0 h1:1RFP5uiPJvT93TAHi+ipd3NACobkW53yUiBqZheE/Js= github.com/leaanthony/slicer v1.6.0/go.mod h1:o/Iz29g7LN0GqH3aMjWAe90381nyZlDNquK+mtH2Fj8= +github.com/leaanthony/u v1.1.0 h1:2n0d2BwPVXSUq5yhe8lJPHdxevE2qK5G99PMStMZMaI= +github.com/leaanthony/u v1.1.0/go.mod h1:9+o6hejoRljvZ3BzdYlVL0JYCwtnAsVuN9pVTQcaRfI= github.com/matryer/is v1.4.0 h1:sosSmIWwkYITGrxZ25ULNDeKiMNzFSr4V/eqBQP0PeE= github.com/matryer/is v1.4.0/go.mod h1:8I/i5uYgLzgsgEloJE1U6xx5HkBQpAZvepWuujKwMRU= -github.com/mattn/go-colorable v0.1.11 h1:nQ+aFkoE2TMGc0b68U2OKSexC+eq46+XwZzWXHRmPYs= github.com/mattn/go-colorable v0.1.11/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= +github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= -github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= +github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= -github.com/pkg/browser v0.0.0-20210706143420-7d21f8c997e2 h1:acNfDZXmm28D2Yg/c3ALnZStzNaZMSagpbr96vY6Zjc= -github.com/pkg/browser v0.0.0-20210706143420-7d21f8c997e2/go.mod h1:HKlIX3XHQyzLZPlr7++PzdhaXEj94dEiJgZDTsxEqUI= +github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8 h1:KoWmjvw+nsYOo29YJK9vDA65RGE3NrOnUtO7a+RF9HU= github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8/go.mod h1:HKlIX3XHQyzLZPlr7++PzdhaXEj94dEiJgZDTsxEqUI= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= -github.com/samber/lo v1.27.1 h1:sTXwkRiIFIQG+G0HeAvOEnGjqWeWtI9cg5/n51KrxPg= -github.com/samber/lo v1.27.1/go.mod h1:it33p9UtPMS7z72fP4gw/EIfQB2eI8ke7GR2wc6+Rhg= +github.com/rivo/uniseg v0.4.4 h1:8TfxU8dW6PdqD27gjM8MVNuicgxIjxpm4K7x4jp8sis= +github.com/rivo/uniseg v0.4.4/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= +github.com/samber/lo v1.38.1 h1:j2XEAqXKb09Am4ebOg31SpvzUTTs6EN3VfgeLUhPdXM= github.com/samber/lo v1.38.1/go.mod h1:+m/ZKRl6ClXCE2Lgf3MsQlWfh4bn1bz6CXEOxnEXnEA= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= -github.com/thoas/go-funk v0.9.1 h1:O549iLZqPpTUQ10ykd26sZhzD+rmR5pWhuElrhbC20M= -github.com/tkrajina/go-reflector v0.5.5 h1:gwoQFNye30Kk7NrExj8zm3zFtrGPqOkzFMLuQZg1DtQ= -github.com/tkrajina/go-reflector v0.5.5/go.mod h1:ECbqLgccecY5kPmPmXg1MrHW585yMcDkVl6IvJe64T4= +github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= +github.com/tkrajina/go-reflector v0.5.6 h1:hKQ0gyocG7vgMD2M3dRlYN6WBBOmdoOzJ6njQSepKdE= github.com/tkrajina/go-reflector v0.5.6/go.mod h1:ECbqLgccecY5kPmPmXg1MrHW585yMcDkVl6IvJe64T4= github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= -github.com/valyala/fasttemplate v1.2.1 h1:TVEnxayobAdVkhQfrfes2IzOB6o+z4roRkPF52WA1u4= github.com/valyala/fasttemplate v1.2.1/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= +github.com/valyala/fasttemplate v1.2.2 h1:lxLXG0uE3Qnshl9QyaK6XJxMXlQZELvChBOCmQD0Loo= github.com/valyala/fasttemplate v1.2.2/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= +github.com/wailsapp/go-webview2 v1.0.7 h1:s95+7irJsAsTy1RsjJ6N0cYX7tZ4gP7Uzawds0L2urs= +github.com/wailsapp/go-webview2 v1.0.7/go.mod h1:Uk2BePfCRzttBBjFrBmqKGJd41P6QIHeV9kTgIeOZNo= github.com/wailsapp/mimetype v1.4.1 h1:pQN9ycO7uo4vsUUuPeHEYoUkLVkaRntMnHJxVwYhwHs= github.com/wailsapp/mimetype v1.4.1/go.mod h1:9aV5k31bBOv5z6u+QP8TltzvNGJPmNJD4XlAL3U+j3o= -golang.org/x/crypto v0.1.0 h1:MDRAIl0xIo9Io2xV565hzXHw3zVseKrJKodhohM5CjU= -golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= -golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0= -golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 h1:3MTrJm4PyNL9NBqvYDSj3DHl46qQakyfqfWo4jgfaEM= -golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE= +golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= +golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4= +golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 h1:k/i9J1pBpvlfR+9QsetwPyERsqu1GIbi967PQMq3Ivc= golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w= golang.org/x/net v0.0.0-20210505024714-0287a6fb4125/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.7.0 h1:rJrUqqhjsgNp7KqAIc25s9pZnjU7TUcSY7HcVZjdn1g= -golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= -golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= +golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= +golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200810151505-1b9f1253b3ed/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -79,15 +74,13 @@ golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211103235746-7861aae1554b/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.5.0 h1:MUK/U/4lj1t1oPg0HfuXDN/Z1wv31ZJ/YcPiGccS4DU= -golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE= +golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.7.0 h1:4BRB4x83lYWy72KwLD/qYDuTu7q9PjSagHvijDw7cLo= -golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= -golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= +golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= From 4b376bcf0d1c6128ec4ca64136b561a5cf45c8b5 Mon Sep 17 00:00:00 2001 From: Light Date: Sat, 14 Oct 2023 17:10:33 +1100 Subject: [PATCH 13/87] Add new htmx/templ template (#2984) * Add new htmx/templ template * update template name * Add new template changelog item --- website/docs/community/templates.mdx | 4 ++++ website/src/pages/changelog.mdx | 1 + 2 files changed, 5 insertions(+) diff --git a/website/docs/community/templates.mdx b/website/docs/community/templates.mdx index 6ff8344deb1..1a781c758a2 100644 --- a/website/docs/community/templates.mdx +++ b/website/docs/community/templates.mdx @@ -61,6 +61,10 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + ## Pure JavaScript (Vanilla) - [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 4923de6866f..6e4bd32fd22 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -21,6 +21,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) - Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) - Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) ### Changed From d51268b8d0680430f3a614775b13e6cd2b906d1c Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 15 Oct 2023 20:08:24 +1100 Subject: [PATCH 14/87] Add basic hardware detection to `wails doctor` --- v2/cmd/wails/doctor.go | 39 +++++++++++++++++++++++++++++++++ v2/go.mod | 6 +++++ v2/go.sum | 20 ++++++++++++++++- website/src/pages/changelog.mdx | 1 + 4 files changed, 65 insertions(+), 1 deletion(-) diff --git a/v2/cmd/wails/doctor.go b/v2/cmd/wails/doctor.go index 75d62b2467a..ef73419ecc8 100644 --- a/v2/cmd/wails/doctor.go +++ b/v2/cmd/wails/doctor.go @@ -1,12 +1,15 @@ package main import ( + "fmt" "runtime" "runtime/debug" + "strconv" "strings" "github.com/pterm/pterm" + "github.com/jaypipes/ghw" "github.com/wailsapp/wails/v2/cmd/wails/flags" "github.com/wailsapp/wails/v2/internal/colour" "github.com/wailsapp/wails/v2/internal/system" @@ -79,6 +82,42 @@ func diagnoseEnvironment(f *flags.Doctor) error { {pterm.Bold.Sprint("Architecture"), runtime.GOARCH}, } + // Probe CPU + cpus, _ := ghw.CPU() + if cpus != nil { + prefix := "CPU" + for idx, cpu := range cpus.Processors { + if len(cpus.Processors) > 1 { + prefix = "CPU " + strconv.Itoa(idx+1) + } + systemTabledata = append(systemTabledata, []string{prefix, cpu.Model}) + } + } else { + systemTabledata = append(systemTabledata, []string{"CPU", "Unknown"}) + } + + // Probe GPU + gpu, _ := ghw.GPU(ghw.WithDisableWarnings()) + if gpu != nil { + prefix := "GPU" + for idx, card := range gpu.GraphicsCards { + if len(gpu.GraphicsCards) > 1 { + prefix = "GPU " + strconv.Itoa(idx+1) + " " + } + details := fmt.Sprintf("%s (%s) - Driver: %s", card.DeviceInfo.Product.Name, card.DeviceInfo.Vendor.Name, card.DeviceInfo.Driver) + systemTabledata = append(systemTabledata, []string{prefix, details}) + } + } else { + systemTabledata = append(systemTabledata, []string{"GPU", "Unknown"}) + } + + memory, _ := ghw.Memory() + if memory != nil { + systemTabledata = append(systemTabledata, []string{"Memory", strconv.Itoa(int(memory.TotalPhysicalBytes/1024/1024/1024)) + "GB"}) + } else { + systemTabledata = append(systemTabledata, []string{"Memory", "Unknown"}) + } + err = pterm.DefaultTable.WithBoxed().WithData(systemTabledata).Render() if err != nil { return err diff --git a/v2/go.mod b/v2/go.mod index 308d2476d26..891928653bc 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -15,6 +15,7 @@ require ( github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 github.com/google/uuid v1.3.0 github.com/jackmordaunt/icns v1.0.0 + github.com/jaypipes/ghw v0.12.0 github.com/labstack/echo/v4 v4.10.2 github.com/labstack/gommon v0.4.0 github.com/leaanthony/clir v1.3.0 @@ -48,17 +49,20 @@ require ( atomicgo.dev/keyboard v0.2.8 // indirect bitbucket.org/creachadair/shell v0.0.7 // indirect github.com/Microsoft/go-winio v0.4.16 // indirect + github.com/StackExchange/wmi v1.2.1 // indirect github.com/alecthomas/chroma v0.10.0 // indirect github.com/aymerick/douceur v0.2.0 // indirect github.com/containerd/console v1.0.3 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/dlclark/regexp2 v1.4.0 // indirect github.com/emirpasic/gods v1.12.0 // indirect + github.com/ghodss/yaml v1.0.0 // indirect github.com/go-git/gcfg v1.5.0 // indirect github.com/go-git/go-billy/v5 v5.2.0 // indirect github.com/gookit/color v1.5.2 // indirect github.com/gorilla/css v1.0.0 // indirect github.com/imdario/mergo v0.3.12 // indirect + github.com/jaypipes/pcidb v1.0.0 // indirect github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect github.com/jchv/go-winloader v0.0.0-20210711035445-715c2860da7e // indirect github.com/kevinburke/ssh_config v0.0.0-20201106050909-4977a11b4351 // indirect @@ -93,5 +97,7 @@ require ( golang.org/x/term v0.13.0 // indirect golang.org/x/text v0.13.0 // indirect gopkg.in/warnings.v0 v0.1.2 // indirect + gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect + howett.net/plist v1.0.0 // indirect ) diff --git a/v2/go.sum b/v2/go.sum index 7c13330926d..c3faec7c0b5 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -12,11 +12,14 @@ github.com/MarvinJWendt/testza v0.2.12/go.mod h1:JOIegYyV7rX+7VZ9r77L/eH6CfJHHzX github.com/MarvinJWendt/testza v0.3.0/go.mod h1:eFcL4I0idjtIx8P9C6KkAuLgATNKpX4/2oUqKc6bF2c= github.com/MarvinJWendt/testza v0.4.2/go.mod h1:mSdhXiKH8sg/gQehJ63bINcCKp7RtYewEjXsvsVUPbE= github.com/MarvinJWendt/testza v0.4.3 h1:u2XaM4IqGp9dsdUmML8/Z791fu4yjQYzOiufOtJwTII= +github.com/MarvinJWendt/testza v0.4.3/go.mod h1:CpXaOfceNEYnLDtNIyTrPPcCpDJYqzZnu2aiA2Wp33U= github.com/Masterminds/semver v1.5.0 h1:H65muMkzWKEuNDnfl9d70GUjFniHKHRbFPGBuZ3QEww= github.com/Masterminds/semver v1.5.0/go.mod h1:MB6lktGJrhw8PrUyiEoblNEGEQ+RzHPF078ddwwvV3Y= github.com/Microsoft/go-winio v0.4.14/go.mod h1:qXqCSQ3Xa7+6tgxaGTIe4Kpcdsi+P8jBhyzoq1bpyYA= github.com/Microsoft/go-winio v0.4.16 h1:FtSW/jqD+l4ba5iPBj9CODVtgfYAD8w2wS923g/cFDk= github.com/Microsoft/go-winio v0.4.16/go.mod h1:XB6nPKklQyQ7GC9LdcBEcBl8PF76WugXOPRXwdLnMv0= +github.com/StackExchange/wmi v1.2.1 h1:VIkavFPXSjcnS+O8yTq7NI32k0R5Aj+v39y29VYDOSA= +github.com/StackExchange/wmi v1.2.1/go.mod h1:rcmrprowKIVzvc+NUiLncP2uuArMWLCbu9SBzvHz7e8= github.com/acarl005/stripansi v0.0.0-20180116102854-5a71ef0e047d h1:licZJFw2RwpHMqeKTCYkitsPqHNxTmd4SNR5r94FGM8= github.com/acarl005/stripansi v0.0.0-20180116102854-5a71ef0e047d/go.mod h1:asat636LX7Bqt5lYEZ27JNDcqxfjdBQuJ/MM4CN/Lzo= github.com/alcortesm/tgz v0.0.0-20161220082320-9c5fe88206d7 h1:uSoVVbwJiQipAclBbw+8quDsfcvFjOpI5iCf4p/cqCs= @@ -51,6 +54,8 @@ github.com/flytam/filenamify v1.0.0 h1:ewx6BY2dj7U6h2zGPJmt33q/BjkSf/YsY/woQvnUN github.com/flytam/filenamify v1.0.0/go.mod h1:Dzf9kVycwcsBlr2ATg6uxjqiFgKGH+5SKFuhdeP5zu8= github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= +github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk= +github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= github.com/gliderlabs/ssh v0.2.2 h1:6zsha5zo/TWhRhwqCD3+EarCAgZ2yN28ipRnGPnwkI0= github.com/gliderlabs/ssh v0.2.2/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0= github.com/go-git/gcfg v1.5.0 h1:Q5ViNfGF8zFgyJWPqYwA7qGFoMTEiBmdlkcfRmpIMa4= @@ -63,12 +68,14 @@ github.com/go-git/go-git-fixtures/v4 v4.0.2-0.20200613231340-f56387b50c12 h1:PbK github.com/go-git/go-git-fixtures/v4 v4.0.2-0.20200613231340-f56387b50c12/go.mod h1:m+ICp2rF3jDhFgEZ/8yziagdT1C+ZpZcrJjappBCDSw= github.com/go-git/go-git/v5 v5.3.0 h1:8WKMtJR2j8RntEXR/uvTKagfEt4GYlwQ7mntE4+0GWc= github.com/go-git/go-git/v5 v5.3.0/go.mod h1:xdX4bWJ48aOrdhnl2XqHYstHbbp6+LFS4r4X+lNVprw= +github.com/go-ole/go-ole v1.2.5/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= +github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4= github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ= github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= @@ -83,10 +90,15 @@ github.com/imdario/mergo v0.3.12 h1:b6R2BslTbIEToALKP7LxUvijTsNI9TAe80pLWN2g/HU= github.com/imdario/mergo v0.3.12/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA= github.com/jackmordaunt/icns v1.0.0 h1:RYSxplerf/l/DUd09AHtITwckkv/mqjVv4DjYdPmAMQ= github.com/jackmordaunt/icns v1.0.0/go.mod h1:7TTQVEuGzVVfOPPlLNHJIkzA6CoV7aH1Dv9dW351oOo= +github.com/jaypipes/ghw v0.12.0 h1:xU2/MDJfWmBhJnujHY9qwXQLs3DBsf0/Xa9vECY0Tho= +github.com/jaypipes/ghw v0.12.0/go.mod h1:jeJGbkRB2lL3/gxYzNYzEDETV1ZJ56OKr+CSeSEym+g= +github.com/jaypipes/pcidb v1.0.0 h1:vtZIfkiCUE42oYbJS0TAq9XSfSmcsgo9IdxSm9qzYU8= +github.com/jaypipes/pcidb v1.0.0/go.mod h1:TnYUvqhPBzCKnH34KrIX22kAeEbDCSRJ9cqLRCuNDfk= github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 h1:BQSFePA1RWJOlocH6Fxy8MmwDt+yVQYULKfN0RoTN8A= github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99/go.mod h1:1lJo3i6rXxKeerYnT8Nvf0QmHCRC1n8sfWVwXF2Frvo= github.com/jchv/go-winloader v0.0.0-20210711035445-715c2860da7e h1:Q3+PugElBCf4PFpxhErSzU3/PY5sFL5Z6rfv4AbGAck= github.com/jchv/go-winloader v0.0.0-20210711035445-715c2860da7e/go.mod h1:alcuEEnZsY1WQsagKhZDsoPCRoOijYqhZvPwLG0kzVs= +github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= github.com/jessevdk/go-flags v1.5.0/go.mod h1:Fw0T6WPc1dYxT4mKEZRfG5kJhaTDP9pj1c2EWnYs/m4= github.com/kevinburke/ssh_config v0.0.0-20201106050909-4977a11b4351 h1:DowS9hvgyYSX4TO5NpyC606/Z4SxnNYbT+WX27or6Ck= github.com/kevinburke/ssh_config v0.0.0-20201106050909-4977a11b4351/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM= @@ -94,6 +106,7 @@ github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa02 github.com/klauspost/cpuid/v2 v2.0.10/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= github.com/klauspost/cpuid/v2 v2.0.12/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= github.com/klauspost/cpuid/v2 v2.1.0 h1:eyi1Ad2aNJMW95zcSbmGg7Cg6cq3ADwLpMAP96d8rF0= +github.com/klauspost/cpuid/v2 v2.1.0/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= @@ -260,6 +273,7 @@ golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o= +golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -318,11 +332,15 @@ gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EV gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= +gopkg.in/yaml.v1 v1.0.0-20140924161607-9f9df34309c0/go.mod h1:WDnlLJ4WF5VGsH/HVa3CI79GS0ol3YnhVnKP89i0kNg= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +howett.net/plist v1.0.0 h1:7CrbWYbPPO/PyNy38b2EB/+gYbjCe2DXBxgtOOZbSQM= +howett.net/plist v1.0.0/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g= diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 6e4bd32fd22..08b6e36dfe9 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -22,6 +22,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) - Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor` ### Changed From b5406cb13ece5c92cd565736bcd0db0759c8c687 Mon Sep 17 00:00:00 2001 From: Carl Seleborg Date: Sat, 21 Oct 2023 07:30:48 +0200 Subject: [PATCH 15/87] Add missing runtime.UnlockOSThread() (#2994) Co-authored-by: Lea Anthony --- v2/internal/frontend/desktop/windows/frontend.go | 1 + 1 file changed, 1 insertion(+) diff --git a/v2/internal/frontend/desktop/windows/frontend.go b/v2/internal/frontend/desktop/windows/frontend.go index a753bbd3013..c388d738472 100644 --- a/v2/internal/frontend/desktop/windows/frontend.go +++ b/v2/internal/frontend/desktop/windows/frontend.go @@ -211,6 +211,7 @@ func (f *Frontend) WindowCenter() { func (f *Frontend) WindowSetAlwaysOnTop(b bool) { runtime.LockOSThread() + defer runtime.UnlockOSThread() f.mainWindow.SetAlwaysOnTop(b) } From b9fc72bbc8f1723d889cd31741eaa2dcb0a3a1bd Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sat, 21 Oct 2023 16:33:39 +1100 Subject: [PATCH 16/87] Bump @babel/traverse from 7.21.4 to 7.23.2 in /website (#2992) Bumps [@babel/traverse](https://github.com/babel/babel/tree/HEAD/packages/babel-traverse) from 7.21.4 to 7.23.2. - [Release notes](https://github.com/babel/babel/releases) - [Changelog](https://github.com/babel/babel/blob/main/CHANGELOG.md) - [Commits](https://github.com/babel/babel/commits/v7.23.2/packages/babel-traverse) --- updated-dependencies: - dependency-name: "@babel/traverse" dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Lea Anthony --- website/pnpm-lock.yaml | 126 +++++++++++++++++++++++++++++++++++------ 1 file changed, 108 insertions(+), 18 deletions(-) diff --git a/website/pnpm-lock.yaml b/website/pnpm-lock.yaml index b172ea35076..44e36e6568d 100644 --- a/website/pnpm-lock.yaml +++ b/website/pnpm-lock.yaml @@ -185,6 +185,14 @@ packages: '@babel/highlight': 7.18.6 dev: false + /@babel/code-frame@7.22.13: + resolution: {integrity: sha512-XktuhWlJ5g+3TJXc5upd9Ks1HutSArik6jf2eAjYFyIOf4ej3RN+184cZbzDvbPnuTJIUhPKKJE3cIsYTiAT3w==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/highlight': 7.22.20 + chalk: 2.4.2 + dev: false + /@babel/compat-data@7.21.4: resolution: {integrity: sha512-/DYyDpeCfaVinT40FPGdkkb+lYSKvsVuMjDAG7jPOWWiM1ibOaB9CXJAlc4d1QpP/U2q2P9jbrSlClKSErd55g==} engines: {node: '>=6.9.0'} @@ -200,7 +208,7 @@ packages: '@babel/helpers': 7.21.0 '@babel/parser': 7.21.4 '@babel/template': 7.20.7 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@babel/types': 7.21.4 convert-source-map: 1.9.0 debug: 4.3.4 @@ -226,7 +234,7 @@ packages: '@babel/helpers': 7.21.0 '@babel/parser': 7.21.4 '@babel/template': 7.20.7 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@babel/types': 7.21.4 convert-source-map: 1.9.0 debug: 4.3.4 @@ -247,6 +255,16 @@ packages: jsesc: 2.5.2 dev: false + /@babel/generator@7.23.0: + resolution: {integrity: sha512-lN85QRR+5IbYrMWM6Y4pE/noaQtg4pNiqeNGX60eqOfo6gtEj6uw/JagelB8vVztSd7R6M5n1+PQkDbHbBRU4g==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.0 + '@jridgewell/gen-mapping': 0.3.3 + '@jridgewell/trace-mapping': 0.3.18 + jsesc: 2.5.2 + dev: false + /@babel/helper-annotate-as-pure@7.18.6: resolution: {integrity: sha512-duORpUiYrEpzKIop6iNbjnwKLAKnJ47csTyRACyEmWj0QdUrm5aqNJGHSSEQSUAvNW0ojX0dOmK9dZduvkfeXA==} engines: {node: '>=6.9.0'} @@ -327,6 +345,11 @@ packages: engines: {node: '>=6.9.0'} dev: false + /@babel/helper-environment-visitor@7.22.20: + resolution: {integrity: sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==} + engines: {node: '>=6.9.0'} + dev: false + /@babel/helper-explode-assignable-expression@7.18.6: resolution: {integrity: sha512-eyAYAsQmB80jNfg4baAtLeWAQHfHFiR483rzFK+BhETlGZaQC9bsfrugfXDCbRHLQbIA7U5NxhhOxN7p/dWIcg==} engines: {node: '>=6.9.0'} @@ -342,6 +365,14 @@ packages: '@babel/types': 7.21.4 dev: false + /@babel/helper-function-name@7.23.0: + resolution: {integrity: sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/template': 7.22.15 + '@babel/types': 7.23.0 + dev: false + /@babel/helper-hoist-variables@7.18.6: resolution: {integrity: sha512-UlJQPkFqFULIcyW5sbzgbkxn2FKRgwWiRexcuaR8RNJRy8+LLveqPjwZV/bwrLZCN0eUHD/x8D0heK1ozuoo6Q==} engines: {node: '>=6.9.0'} @@ -349,6 +380,13 @@ packages: '@babel/types': 7.21.4 dev: false + /@babel/helper-hoist-variables@7.22.5: + resolution: {integrity: sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.0 + dev: false + /@babel/helper-member-expression-to-functions@7.21.0: resolution: {integrity: sha512-Muu8cdZwNN6mRRNG6lAYErJ5X3bRevgYR2O8wN0yn7jJSnGDu6eG59RfT29JHxGUovyfrh6Pj0XzmR7drNVL3Q==} engines: {node: '>=6.9.0'} @@ -373,7 +411,7 @@ packages: '@babel/helper-split-export-declaration': 7.18.6 '@babel/helper-validator-identifier': 7.19.1 '@babel/template': 7.20.7 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@babel/types': 7.21.4 transitivePeerDependencies: - supports-color @@ -418,7 +456,7 @@ packages: '@babel/helper-member-expression-to-functions': 7.21.0 '@babel/helper-optimise-call-expression': 7.18.6 '@babel/template': 7.20.7 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@babel/types': 7.21.4 transitivePeerDependencies: - supports-color @@ -445,16 +483,33 @@ packages: '@babel/types': 7.21.4 dev: false + /@babel/helper-split-export-declaration@7.22.6: + resolution: {integrity: sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/types': 7.23.0 + dev: false + /@babel/helper-string-parser@7.19.4: resolution: {integrity: sha512-nHtDoQcuqFmwYNYPz3Rah5ph2p8PFeFCsZk9A/48dPc/rGocJ5J3hAAZ7pb76VWX3fZKu+uEr/FhH5jLx7umrw==} engines: {node: '>=6.9.0'} dev: false + /@babel/helper-string-parser@7.22.5: + resolution: {integrity: sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw==} + engines: {node: '>=6.9.0'} + dev: false + /@babel/helper-validator-identifier@7.19.1: resolution: {integrity: sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==} engines: {node: '>=6.9.0'} dev: false + /@babel/helper-validator-identifier@7.22.20: + resolution: {integrity: sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==} + engines: {node: '>=6.9.0'} + dev: false + /@babel/helper-validator-option@7.21.0: resolution: {integrity: sha512-rmL/B8/f0mKS2baE9ZpyTcTavvEuWhTTW8amjzXNvYG4AwBsqTLikfXsEofsJEfKHf+HQVQbFOHy6o+4cnC/fQ==} engines: {node: '>=6.9.0'} @@ -466,7 +521,7 @@ packages: dependencies: '@babel/helper-function-name': 7.21.0 '@babel/template': 7.20.7 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@babel/types': 7.21.4 transitivePeerDependencies: - supports-color @@ -477,7 +532,7 @@ packages: engines: {node: '>=6.9.0'} dependencies: '@babel/template': 7.20.7 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@babel/types': 7.21.4 transitivePeerDependencies: - supports-color @@ -492,6 +547,15 @@ packages: js-tokens: 4.0.0 dev: false + /@babel/highlight@7.22.20: + resolution: {integrity: sha512-dkdMCN3py0+ksCgYmGG8jKeGA/8Tk+gJwSYYlFGxG5lmhfKNoAy004YpLxpS1W2J8m/EK2Ew+yOs9pVRwO89mg==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/helper-validator-identifier': 7.22.20 + chalk: 2.4.2 + js-tokens: 4.0.0 + dev: false + /@babel/parser@7.21.4: resolution: {integrity: sha512-alVJj7k7zIxqBZ7BTRhz0IqJFxW1VJbm6N8JbcYhQ186df9ZBPbZBmWSqAMXwHGsCJdYks7z/voa3ibiS5bCIw==} engines: {node: '>=6.0.0'} @@ -500,6 +564,14 @@ packages: '@babel/types': 7.21.4 dev: false + /@babel/parser@7.23.0: + resolution: {integrity: sha512-vvPKKdMemU85V9WE/l5wZEmImpCtLqbnTvqDS2U1fJ96KrxoW7KrXhNsNCblQlg8Ck4b85yxdTyelsMUgFUXiw==} + engines: {node: '>=6.0.0'} + hasBin: true + dependencies: + '@babel/types': 7.23.0 + dev: false + /@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression@7.18.6(@babel/core@7.21.4): resolution: {integrity: sha512-Dgxsyg54Fx1d4Nge8UnvTrED63vrwOdPmyvPzlNN/boaliRP54pm3pGzZD1SJUwrBA+Cs/xdG8kXX6Mn/RfISQ==} engines: {node: '>=6.9.0'} @@ -1510,18 +1582,27 @@ packages: '@babel/types': 7.21.4 dev: false - /@babel/traverse@7.21.4: - resolution: {integrity: sha512-eyKrRHKdyZxqDm+fV1iqL9UAHMoIg0nDaGqfIOd8rKH17m5snv7Gn4qgjBoFfLz9APvjFU/ICT00NVCv1Epp8Q==} + /@babel/template@7.22.15: + resolution: {integrity: sha512-QPErUVm4uyJa60rkI73qneDacvdvzxshT3kksGqlGWYdOTIUOwJ7RDUL8sGqslY1uXWSL6xMFKEXDS3ox2uF0w==} engines: {node: '>=6.9.0'} dependencies: - '@babel/code-frame': 7.21.4 - '@babel/generator': 7.21.4 - '@babel/helper-environment-visitor': 7.18.9 - '@babel/helper-function-name': 7.21.0 - '@babel/helper-hoist-variables': 7.18.6 - '@babel/helper-split-export-declaration': 7.18.6 - '@babel/parser': 7.21.4 - '@babel/types': 7.21.4 + '@babel/code-frame': 7.22.13 + '@babel/parser': 7.23.0 + '@babel/types': 7.23.0 + dev: false + + /@babel/traverse@7.23.2: + resolution: {integrity: sha512-azpe59SQ48qG6nu2CzcMLbxUudtN+dOM9kDbUqGq3HXUJRlo7i8fvPoxQUzYgLZ4cMVmuZgm8vvBpNeRhd6XSw==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/code-frame': 7.22.13 + '@babel/generator': 7.23.0 + '@babel/helper-environment-visitor': 7.22.20 + '@babel/helper-function-name': 7.23.0 + '@babel/helper-hoist-variables': 7.22.5 + '@babel/helper-split-export-declaration': 7.22.6 + '@babel/parser': 7.23.0 + '@babel/types': 7.23.0 debug: 4.3.4 globals: 11.12.0 transitivePeerDependencies: @@ -1537,6 +1618,15 @@ packages: to-fast-properties: 2.0.0 dev: false + /@babel/types@7.23.0: + resolution: {integrity: sha512-0oIyUfKoI3mSqMvsxBdclDwxXKXAUA8v/apZbc+iSyARYou1o8ZGDxbUYyLFoW2arqS2jDGqJuZvv1d/io1axg==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/helper-string-parser': 7.22.5 + '@babel/helper-validator-identifier': 7.22.20 + to-fast-properties: 2.0.0 + dev: false + /@colors/colors@1.5.0: resolution: {integrity: sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==} engines: {node: '>=0.1.90'} @@ -1623,7 +1713,7 @@ packages: '@babel/preset-typescript': 7.21.4(@babel/core@7.21.4) '@babel/runtime': 7.21.0 '@babel/runtime-corejs3': 7.21.0 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@docusaurus/cssnano-preset': 2.4.0 '@docusaurus/logger': 2.4.0 '@docusaurus/mdx-loader': 2.4.0(@docusaurus/types@2.4.0)(@swc/core@1.3.7)(react-dom@17.0.2)(react@17.0.2) @@ -1734,7 +1824,7 @@ packages: react-dom: ^16.8.4 || ^17.0.0 dependencies: '@babel/parser': 7.21.4 - '@babel/traverse': 7.21.4 + '@babel/traverse': 7.23.2 '@docusaurus/logger': 2.4.0 '@docusaurus/utils': 2.4.0(@docusaurus/types@2.4.0)(@swc/core@1.3.7) '@mdx-js/mdx': 1.6.22 From 3e938a10aa64ed93fa73fc36fb24f29e5f9cd8e2 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sat, 21 Oct 2023 16:43:21 +1100 Subject: [PATCH 17/87] Bump undici from 5.21.0 to 5.26.3 in /scripts/sponsors (#2987) Bumps [undici](https://github.com/nodejs/undici) from 5.21.0 to 5.26.3. - [Release notes](https://github.com/nodejs/undici/releases) - [Commits](https://github.com/nodejs/undici/compare/v5.21.0...v5.26.3) --- updated-dependencies: - dependency-name: undici dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Lea Anthony --- scripts/sponsors/package-lock.json | 37 +++++++++++------------------- 1 file changed, 13 insertions(+), 24 deletions(-) diff --git a/scripts/sponsors/package-lock.json b/scripts/sponsors/package-lock.json index 5153e9cf85b..e3523a1d983 100644 --- a/scripts/sponsors/package-lock.json +++ b/scripts/sponsors/package-lock.json @@ -20,6 +20,14 @@ "url": "https://github.com/sponsors/antfu" } }, + "node_modules/@fastify/busboy": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@fastify/busboy/-/busboy-2.0.0.tgz", + "integrity": "sha512-JUFJad5lv7jxj926GPgymrWQxxjPYuJNiNjNMzqT+HiuP6Vl3dk5xzG+8sTX96np0ZAluvaMzPsjhHZ5rNuNQQ==", + "engines": { + "node": ">=14" + } + }, "node_modules/ajv": { "version": "6.12.6", "resolved": "https://registry.npmmirror.com/ajv/-/ajv-6.12.6.tgz", @@ -196,17 +204,6 @@ "ieee754": "^1.1.13" } }, - "node_modules/busboy": { - "version": "1.6.0", - "resolved": "https://registry.npmmirror.com/busboy/-/busboy-1.6.0.tgz", - "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==", - "dependencies": { - "streamsearch": "^1.1.0" - }, - "engines": { - "node": ">=10.16.0" - } - }, "node_modules/camelcase": { "version": "4.1.0", "resolved": "https://registry.npmmirror.com/camelcase/-/camelcase-4.1.0.tgz", @@ -1584,14 +1581,6 @@ "node": ">=0.10.0" } }, - "node_modules/streamsearch": { - "version": "1.1.0", - "resolved": "https://registry.npmmirror.com/streamsearch/-/streamsearch-1.1.0.tgz", - "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==", - "engines": { - "node": ">=10.0.0" - } - }, "node_modules/string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -1781,14 +1770,14 @@ } }, "node_modules/undici": { - "version": "5.21.0", - "resolved": "https://registry.npmjs.org/undici/-/undici-5.21.0.tgz", - "integrity": "sha512-HOjK8l6a57b2ZGXOcUsI5NLfoTrfmbOl90ixJDl0AEFG4wgHNDQxtZy15/ZQp7HhjkpaGlp/eneMgtsu1dIlUA==", + "version": "5.26.3", + "resolved": "https://registry.npmjs.org/undici/-/undici-5.26.3.tgz", + "integrity": "sha512-H7n2zmKEWgOllKkIUkLvFmsJQj062lSm3uA4EYApG8gLuiOM0/go9bIoC3HVaSnfg4xunowDE2i9p8drkXuvDw==", "dependencies": { - "busboy": "^1.6.0" + "@fastify/busboy": "^2.0.0" }, "engines": { - "node": ">=12.18" + "node": ">=14.0" } }, "node_modules/universalify": { From b42a18be6b1498f4448aef2221597f622d5e5b52 Mon Sep 17 00:00:00 2001 From: Mike Date: Sat, 21 Oct 2023 06:47:02 +0100 Subject: [PATCH 18/87] feat: add golangci workflow with some linters (#2957) This implements the golangci-lint workflow to new PRs. It includes a limited number of enabled linters including: - errcheck - errname - gofmt - gofumpt - gosimple - misspell Signed-off-by: mikeee Co-authored-by: Lea Anthony --- .github/workflows/pr.yml | 23 +++++ v2/.golangci.yml | 162 ++++++++++++++++++++++++++++++++ v2/Taskfile.yaml | 10 ++ website/src/pages/changelog.mdx | 3 +- 4 files changed, 197 insertions(+), 1 deletion(-) create mode 100644 v2/.golangci.yml diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml index 6db750b73ef..46acb8ee434 100644 --- a/.github/workflows/pr.yml +++ b/.github/workflows/pr.yml @@ -26,6 +26,29 @@ jobs: run: | echo "::warning::Feature branch does not contain any changes to the website." + lint_go: + name: Run Go Linters + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Setup Go + uses: actions/setup-go@v4 + with: + go-version: "1.21" + + - name: Update go modules + working-directory: ./v2 + run: go mod tidy + + - name: Run Linter + uses: golangci/golangci-lint-action@v3 + with: + version: v1.54 + working-directory: ./v2 + args: --timeout=10m0s --config ./.golangci.yml + test_go: name: Run Go Tests runs-on: ${{ matrix.os }} diff --git a/v2/.golangci.yml b/v2/.golangci.yml new file mode 100644 index 00000000000..66b77ba7ff6 --- /dev/null +++ b/v2/.golangci.yml @@ -0,0 +1,162 @@ +# Options for analysis runner. +run: + # Custom concurrency value + concurrency: 4 + + # Execution timeout + timeout: 10m + + # Exit code when an issue is found. + issues-exit-code: 1 + + # Inclusion of test files + tests: false + + modules-download-mode: readonly + + allow-parallel-runners: false + + go: '1.21' + + +output: + # Runner output format + format: tab + + # Print line of issue code + print-issued-lines: false + + # Append linter to the output + print-linter-name: true + + # Separate issues by line + uniq-by-line: true + + # Output path prefixing + path-prefix: "" + + # Sort results + sort-results: true + + +# Specific linter configs +linters-settings: + errcheck: + check-type-assertions: false + check-blank: false + ignore: fmt:.* + disable-default-exclusions: false + + gofmt: + simplify: true + + gofumpt: + extra-rules: false + +linters: + fast: false + # Enable all available linters. + enable-all: true + # Disable specific linters + disable: + - asasalint + - asciicheck + - bidichk + - bodyclose + - containedctx + - contextcheck + - cyclop + - deadcode + - decorder + - depguard + - dogsled + - dupl + - dupword + - durationcheck + - errchkjson + - errorlint + - execinquery + - exhaustive + - exhaustivestruct + - exhaustruct + - exportloopref + - forbidigo + - forcetypeassert + - funlen + - gci + - ginkgolinter + - gocheckcompilerdirectives + - gochecknoglobals + - gochecknoinits + - gocognit + - goconst + - gocritic + - gocyclo + - godot + - godox + - goerr113 + - goheader + - goimports + - golint + - gomnd + - gomoddirectives + - gomodguard + - goprintffuncname + - gosec + - gosmopolitan + - govet + - grouper + - ifshort + - importas + - ineffassign + - interfacebloat + - interfacer + - ireturn + - lll + - loggercheck + - maintidx + - makezero + - maligned + - mirror + - musttag + - nakedret + - nestif + - nilerr + - nilnil + - nlreturn + - noctx + - nolintlint + - nonamedreturns + - nosnakecase + - nosprintfhostport + - paralleltest + - prealloc + - predeclared + - promlinter + - reassign + - revive + - rowserrcheck + - scopelint + - sqlclosecheck + - staticcheck + - structcheck + - stylecheck + - tagalign + - tagliatelle + - tenv + - testableexamples + - testpackage + - thelper + - tparallel + - typecheck + - unconvert + - unparam + - unused + - usestdlibvars + - varcheck + - varnamelen + - wastedassign + - whitespace + - wrapcheck + - wsl + - zerologlint \ No newline at end of file diff --git a/v2/Taskfile.yaml b/v2/Taskfile.yaml index 526dd097d04..d1893732b44 100644 --- a/v2/Taskfile.yaml +++ b/v2/Taskfile.yaml @@ -3,6 +3,16 @@ version: "3" tasks: + download: + summary: Run go mod tidy + cmds: + - go mod tidy + + lint: + summary: Run golangci-lint + cmds: + - golangci-lint run ./... --timeout=3m -v + release: summary: Release a new version of Task. Call with `task v2:release -- ` dir: tools/release diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 08b6e36dfe9..8b1829ef67e 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -21,8 +21,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) - Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) - Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) -- Added CPU/GPU/Memory detection for `wails doctor` +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c ### Changed From a9fb0713a1397468293cd9f79ac34dbbdfa55579 Mon Sep 17 00:00:00 2001 From: TuffenDuffen <50796186+TuffenDuffen@users.noreply.github.com> Date: Sat, 21 Oct 2023 07:50:02 +0200 Subject: [PATCH 19/87] Update zypper.go (#2941) * Update zypper.go * Update changelog.mdx --- v2/internal/system/packagemanager/zypper.go | 1 + website/src/pages/changelog.mdx | 1 + 2 files changed, 2 insertions(+) diff --git a/v2/internal/system/packagemanager/zypper.go b/v2/internal/system/packagemanager/zypper.go index c486b53e1c3..efaeb0b1b8d 100644 --- a/v2/internal/system/packagemanager/zypper.go +++ b/v2/internal/system/packagemanager/zypper.go @@ -45,6 +45,7 @@ func (z *Zypper) Packages() packagemap { }, "npm": []*Package{ {Name: "npm10", SystemPackage: true}, + {Name: "npm20", SystemPackage: true}, }, "docker": []*Package{ {Name: "docker", SystemPackage: true, Optional: true}, diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 8b1829ef67e..7da2b65f832 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -34,6 +34,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) ## v2.6.0 - 2023-09-06 From ee4ce1d3ef48c5ea142798e2de5ccaa9e9fbefb2 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sat, 21 Oct 2023 20:24:56 +1100 Subject: [PATCH 20/87] Add mac compile error troubleshooting tip --- website/docs/guides/troubleshooting.mdx | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/website/docs/guides/troubleshooting.mdx b/website/docs/guides/troubleshooting.mdx index cf22c99070f..3bff4e44ffa 100644 --- a/website/docs/guides/troubleshooting.mdx +++ b/website/docs/guides/troubleshooting.mdx @@ -173,6 +173,28 @@ If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-se Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + -- ## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" From 63c9baa466ca0cf2dc6369be53c5bad72876a666 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=A4=9A=E5=95=A6b=E6=A2=A6?= <1715109585@qq.com> Date: Sat, 21 Oct 2023 17:29:29 +0800 Subject: [PATCH 21/87] Add tips to solve the problem that macos cannot be compiled. (#2925) * Add tips to solve the problem that macos cannot be compiled. * Add tips to solve the problem that macos cannot be compiled. --------- Co-authored-by: Lea Anthony --- .../current/guides/manual-builds.mdx | 1 + 1 file changed, 1 insertion(+) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx index 83de0fdfd8f..72028ea6b12 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx @@ -80,6 +80,7 @@ Wails CLI 为项目做了很多繁重的工作,但有时需要手动构建项 #### 手动步骤 - 对于 dev 构建,最少的命令是:`go build -tags dev -gcflags "all=-N -l"` + - 在macos上需要加上环境变量 `export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l"` - 对于生产构建,最少的命令是:`go build -tags desktop,production -ldflags "-w -s -H windowsgui"` - 确保在与 `.syso` 文件相同的目录中进行编译 From 42708e7f40b616f719b1dcf2e181623cb6bf7ff7 Mon Sep 17 00:00:00 2001 From: Yuki Shindo Date: Sat, 21 Oct 2023 18:50:39 +0900 Subject: [PATCH 22/87] Update README.ja.md (#2764) Co-authored-by: Lea Anthony --- README.ja.md | 63 ++++++++++++++-------------------------------------- 1 file changed, 17 insertions(+), 46 deletions(-) diff --git a/README.ja.md b/README.ja.md index 375a543db28..8d29c7d1cc2 100644 --- a/README.ja.md +++ b/README.ja.md @@ -54,17 +54,16 @@ - [目次](#目次) - [はじめに](#はじめに) - - [公式サイト](#公式サイト) - - [ロードマップ](#ロードマップ) - [特徴](#特徴) -- [スポンサー](#スポンサー) + - [ロードマップ](#ロードマップ) - [始め方](#始め方) +- [スポンサー](#スポンサー) - [FAQ](#faq) - [スター数の推移](#スター数の推移) - [コントリビューター](#コントリビューター) -- [特記事項](#特記事項) -- [スペシャルサンクス](#スペシャルサンクス) - [ライセンス](#ライセンス) +- [インスピレーション](#インスピレーション) + ## はじめに @@ -72,44 +71,35 @@ Go プログラムにウェブインタフェースを提供する従来の方 Wails では Go のコードとウェブフロントエンドを単一のバイナリにまとめる機能を提供します。 また、プロジェクトの作成、コンパイル、ビルドを行うためのツールが提供されています。あなたがすべきことは創造性を発揮することです! -### 公式サイト - -Version 2: - -Wails v2 が 3 つのプラットフォームでベータ版としてリリースされました。興味のある方は[新しいウェブサイト](https://wails.io)をご覧ください。 - -レガシー版 v1: - -レガシー版 v1 のドキュメントは[https://wails.app](https://wails.app)で見ることができます。 - -### ロードマップ - -プロジェクトのロードマップは[こちら](https://github.com/wailsapp/wails/discussions/1484)になります。 -機能拡張のリクエストを出す前にご覧ください。 - ## 特徴 - バックエンドには Go を利用しています - 使い慣れたフロントエンド技術を利用して UI を構築できます -- あらかじめ用意されたテンプレートを利用することで、リッチなフロントエンドを備えた Go プログラムを作成できます +- あらかじめ用意されたテンプレートを利用することで、リッチなフロントエンドを備えた Go プログラムを素早く作成できます - JavaScript から Go のメソッドを簡単に呼び出すことができます - あなたの書いた Go の構造体やメソットに応じた TypeScript の定義が自動生成されます - ネイティブのダイアログとメニューが利用できます +- ネイティブなダーク/ライトモードをサポートします - モダンな半透明や「frosted window」エフェクトをサポートしています - Go と JavaScript 間で統一されたイベント・システムを備えています - プロジェクトを素早く生成して構築する強力な cli ツールを用意しています - マルチプラットフォームに対応しています - ネイティブなレンダリングエンジンを使用しています - _つまりブラウザを埋め込んでいるわけではありません!_ -## スポンサー +### ロードマップ -このプロジェクトは、以下の方々・企業によって支えられています。 - +プロジェクトのロードマップは[こちら](https://github.com/wailsapp/wails/discussions/1484)になります。 +機能拡張のリクエストを出す前にご覧ください。 ## 始め方 インストール方法は[公式サイト](https://wails.io/docs/gettingstarted/installation)に掲載されています。 +## スポンサー + +このプロジェクトは、以下の方々・企業によって支えられています。 + + ## FAQ - Electron の代替品になりますか? @@ -137,13 +127,11 @@ Wails v2 が 3 つのプラットフォームでベータ版としてリリー 貢献してくれた方のリストが大きくなりすぎて、readme に入りきらなくなりました! このプロジェクトに貢献してくれた素晴らしい方々のページは[こちら](https://wails.io/credits#contributors)です。 -## 特記事項 +## ライセンス -このプロジェクトは以下の方々の協力がなければ、実現しなかったと思います。 +[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fwailsapp%2Fwails.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fwailsapp%2Fwails?ref=badge_large) -- [Dustin Krysak](https://wiki.ubuntu.com/bashfulrobot) - 彼のサポートとフィードバックはとても大きいものでした。 -- [Serge Zaitsev](https://github.com/zserge) - Wails のウィンドウで使用している[Webview](https://github.com/zserge/webview)の作者です。 -- [Byron](https://github.com/bh90210) - 時には Byron が一人でこのプロジェクトを存続させてくれたこともありました。彼の素晴らしいインプットがなければ v1 に到達することはなかったでしょう。 +## インスピレーション プロジェクトを進める際に、以下のアルバムたちも支えてくれています。 @@ -161,20 +149,3 @@ Wails v2 が 3 つのプラットフォームでベータ版としてリリー - [Maxthor - Another World](https://open.spotify.com/album/3tklE2Fgw1hCIUstIwPBJF) - [Alun Tan Lan - Y Distawrwydd](https://open.spotify.com/album/0c32OywcLpdJCWWMC6vB8v) -## スペシャルサンクス - -

-
- このプロジェクトを後援し、WailsをApple Siliconに移植する取り組みを支援してくれた Paceとても感謝しています!

- パワフルで素早く簡単に使えるプロジェクト管理ツールをお探しなら、ぜひチェックしてみてください!

-

- -

- ライセンスを提供していただいたJetBrains社に感謝します!

- ロゴをクリックして、感謝の気持ちを伝えてください!

- -

- -## ライセンス - -[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fwailsapp%2Fwails.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fwailsapp%2Fwails?ref=badge_large) From 6c46f6b41ca29b1dbd58d006df66bf6437e227ad Mon Sep 17 00:00:00 2001 From: Andrey Pshenkin Date: Sat, 21 Oct 2023 20:44:38 +0100 Subject: [PATCH 23/87] Implement file association Open a file from Finder/Explorer (#2918) * implement MacOS openFile/openFiles events * wip: windows file association * fix macro import * add file icon copy * try copy icon * keep only required part of scripts * update config schema * fix json * set fileAssociation for mac via config * proper iconName handling * add fileAssociation icon generator * fix file association icons bundle * don't break compatibility * remove mimeType as not supported linux for now * add documentation * adjust config schema * restore formatting * remove unused option in file association * get rid of openFiles mac os. change configuration structure * remove unused channel * fix documentation * fix typo --------- Co-authored-by: Lea Anthony --- .../frontend/desktop/darwin/AppDelegate.h | 4 +- .../frontend/desktop/darwin/AppDelegate.m | 10 +- .../frontend/desktop/darwin/frontend.go | 21 ++ v2/internal/project/project.go | 19 +- .../buildassets/build/darwin/Info.dev.plist | 21 +- v2/pkg/buildassets/build/darwin/Info.plist | 21 +- .../build/windows/installer/project.nsi | 16 +- .../build/windows/installer/wails_tools.nsh | 49 ++++- v2/pkg/commands/build/packager.go | 46 ++++- v2/pkg/options/mac/mac.go | 3 +- website/docs/guides/file-association.mdx | 184 ++++++++++++++++++ website/src/pages/changelog.mdx | 1 + website/static/schemas/config.v2.json | 158 ++++++++++++--- 13 files changed, 492 insertions(+), 61 deletions(-) create mode 100644 website/docs/guides/file-association.mdx diff --git a/v2/internal/frontend/desktop/darwin/AppDelegate.h b/v2/internal/frontend/desktop/darwin/AppDelegate.h index e2dd841c91e..7b533ad5ff7 100644 --- a/v2/internal/frontend/desktop/darwin/AppDelegate.h +++ b/v2/internal/frontend/desktop/darwin/AppDelegate.h @@ -11,7 +11,7 @@ #import #import "WailsContext.h" -@interface AppDelegate : NSResponder +@interface AppDelegate : NSResponder @property bool alwaysOnTop; @property bool startHidden; @@ -20,4 +20,6 @@ @end +extern void HandleOpenFile(char *); + #endif /* AppDelegate_h */ diff --git a/v2/internal/frontend/desktop/darwin/AppDelegate.m b/v2/internal/frontend/desktop/darwin/AppDelegate.m index b66979e76af..6a9f5a4c379 100644 --- a/v2/internal/frontend/desktop/darwin/AppDelegate.m +++ b/v2/internal/frontend/desktop/darwin/AppDelegate.m @@ -11,9 +11,17 @@ #import "AppDelegate.h" @implementation AppDelegate +-(BOOL)application:(NSApplication *)sender openFile:(NSString *)filename +{ + const char* utf8FileName = filename.UTF8String; + HandleOpenFile((char*)utf8FileName); + return YES; +} + - (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)sender { - return NO; + return NO; } + - (void)applicationWillFinishLaunching:(NSNotification *)aNotification { [NSApp setActivationPolicy:NSApplicationActivationPolicyRegular]; if (self.alwaysOnTop) { diff --git a/v2/internal/frontend/desktop/darwin/frontend.go b/v2/internal/frontend/desktop/darwin/frontend.go index a6a0808fd7f..66e5dc0cf91 100644 --- a/v2/internal/frontend/desktop/darwin/frontend.go +++ b/v2/internal/frontend/desktop/darwin/frontend.go @@ -38,6 +38,7 @@ const startURL = "wails://wails/" var messageBuffer = make(chan string, 100) var requestBuffer = make(chan webview.Request, 100) var callbackBuffer = make(chan uint, 10) +var openFilepathBuffer = make(chan string, 100) type Frontend struct { @@ -107,15 +108,23 @@ func NewFrontend(ctx context.Context, appoptions *options.App, myLogger *logger. go result.startMessageProcessor() go result.startCallbackProcessor() + go result.startFileOpenProcessor() return result } +func (f *Frontend) startFileOpenProcessor() { + for filePath := range openFilepathBuffer { + f.ProcessOpenFileEvent(filePath) + } +} + func (f *Frontend) startMessageProcessor() { for message := range messageBuffer { f.processMessage(message) } } + func (f *Frontend) startRequestProcessor() { for request := range requestBuffer { f.assets.ServeWebViewRequest(request) @@ -355,6 +364,12 @@ func (f *Frontend) processMessage(message string) { } +func (f *Frontend) ProcessOpenFileEvent(filePath string) { + if f.frontendOptions.Mac != nil && f.frontendOptions.Mac.OnFileOpen != nil { + f.frontendOptions.Mac.OnFileOpen(filePath) + } +} + func (f *Frontend) Callback(message string) { escaped, err := json.Marshal(message) if err != nil { @@ -398,3 +413,9 @@ func processCallback(callbackID uint) { func processURLRequest(_ unsafe.Pointer, wkURLSchemeTask unsafe.Pointer) { requestBuffer <- webview.NewRequest(wkURLSchemeTask) } + +//export HandleOpenFile +func HandleOpenFile(filePath *C.char) { + goFilepath := C.GoString(filePath) + openFilepathBuffer <- goFilepath +} diff --git a/v2/internal/project/project.go b/v2/internal/project/project.go index 023ca1dfed2..a0bf518a9c9 100644 --- a/v2/internal/project/project.go +++ b/v2/internal/project/project.go @@ -216,11 +216,20 @@ type Author struct { } type Info struct { - CompanyName string `json:"companyName"` - ProductName string `json:"productName"` - ProductVersion string `json:"productVersion"` - Copyright *string `json:"copyright"` - Comments *string `json:"comments"` + CompanyName string `json:"companyName"` + ProductName string `json:"productName"` + ProductVersion string `json:"productVersion"` + Copyright *string `json:"copyright"` + Comments *string `json:"comments"` + FileAssociations []FileAssociation `json:"fileAssociations"` +} + +type FileAssociation struct { + Ext string `json:"ext"` + Name string `json:"name"` + Description string `json:"description"` + IconName string `json:"iconName"` + Role string `json:"role"` } type Bindings struct { diff --git a/v2/pkg/buildassets/build/darwin/Info.dev.plist b/v2/pkg/buildassets/build/darwin/Info.dev.plist index 02e7358ee86..9d4096f4fb2 100644 --- a/v2/pkg/buildassets/build/darwin/Info.dev.plist +++ b/v2/pkg/buildassets/build/darwin/Info.dev.plist @@ -23,10 +23,29 @@ true NSHumanReadableCopyright {{.Info.Copyright}} + {{if .Info.FileAssociations}} + CFBundleDocumentTypes + + {{range .Info.FileAssociations}} + + CFBundleTypeExtensions + + {{.Ext}} + + CFBundleTypeName + {{.Name}} + CFBundleTypeRole + {{.Role}} + CFBundleTypeIconFile + {{.IconName}} + + {{end}} + + {{end}} NSAppTransportSecurity NSAllowsLocalNetworking - \ No newline at end of file + diff --git a/v2/pkg/buildassets/build/darwin/Info.plist b/v2/pkg/buildassets/build/darwin/Info.plist index e7819a7e877..079bdafa962 100644 --- a/v2/pkg/buildassets/build/darwin/Info.plist +++ b/v2/pkg/buildassets/build/darwin/Info.plist @@ -23,5 +23,24 @@ true NSHumanReadableCopyright {{.Info.Copyright}} + {{if .Info.FileAssociations}} + CFBundleDocumentTypes + + {{range .Info.FileAssociations}} + + CFBundleTypeExtensions + + {{.Ext}} + + CFBundleTypeName + {{.Name}} + CFBundleTypeRole + {{.Role}} + CFBundleTypeIconFile + {{.IconName}} + + {{end}} + + {{end}} - \ No newline at end of file + diff --git a/v2/pkg/buildassets/build/windows/installer/project.nsi b/v2/pkg/buildassets/build/windows/installer/project.nsi index 13cc4f023bc..f18f2b5dfb9 100644 --- a/v2/pkg/buildassets/build/windows/installer/project.nsi +++ b/v2/pkg/buildassets/build/windows/installer/project.nsi @@ -3,10 +3,10 @@ Unicode true #### ## Please note: Template replacements don't work in this file. They are provided with default defines like ## mentioned underneath. -## If the keyword is not defined, "wails_tools.nsh" will populate them with the values from ProjectInfo. -## If they are defined here, "wails_tools.nsh" will not touch them. This allows to use this project.nsi manually +## If the keyword is not defined, "wails_tools.nsh" will populate them with the values from ProjectInfo. +## If they are defined here, "wails_tools.nsh" will not touch them. This allows to use this project.nsi manually ## from outside of Wails for debugging and development of the installer. -## +## ## For development first make a wails nsis build to populate the "wails_tools.nsh": ## > wails build --target windows/amd64 --nsis ## Then you can call makensis on this file with specifying the path to your binary: @@ -17,7 +17,7 @@ Unicode true ## For a installer with both architectures: ## > makensis -DARG_WAILS_AMD64_BINARY=..\..\bin\app-amd64.exe -DARG_WAILS_ARM64_BINARY=..\..\bin\app-arm64.exe #### -## The following information is taken from the ProjectInfo file, but they can be overwritten here. +## The following information is taken from the ProjectInfo file, but they can be overwritten here. #### ## !define INFO_PROJECTNAME "MyProject" # Default "{{.Name}}" ## !define INFO_COMPANYNAME "MyCompany" # Default "{{.Info.CompanyName}}" @@ -85,16 +85,18 @@ Section !insertmacro wails.webview2runtime SetOutPath $INSTDIR - + !insertmacro wails.files CreateShortcut "$SMPROGRAMS\${INFO_PRODUCTNAME}.lnk" "$INSTDIR\${PRODUCT_EXECUTABLE}" CreateShortCut "$DESKTOP\${INFO_PRODUCTNAME}.lnk" "$INSTDIR\${PRODUCT_EXECUTABLE}" + !insertmacro wails.associateFiles + !insertmacro wails.writeUninstaller SectionEnd -Section "uninstall" +Section "uninstall" !insertmacro wails.setShellContext RMDir /r "$AppData\${PRODUCT_EXECUTABLE}" # Remove the WebView2 DataPath @@ -104,5 +106,7 @@ Section "uninstall" Delete "$SMPROGRAMS\${INFO_PRODUCTNAME}.lnk" Delete "$DESKTOP\${INFO_PRODUCTNAME}.lnk" + !insertmacro wails.unassociateFiles + !insertmacro wails.deleteUninstaller SectionEnd diff --git a/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh b/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh index 467c349ac53..bc79fb395ac 100644 --- a/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh +++ b/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh @@ -163,17 +163,58 @@ RequestExecutionLevel "${REQUEST_EXECUTION_LEVEL}" Goto ok ${EndIf} ${EndIf} - + SetDetailsPrint both DetailPrint "${WAILS_INSTALL_WEBVIEW_DETAILPRINT}" SetDetailsPrint listonly - + InitPluginsDir CreateDirectory "$pluginsdir\webview2bootstrapper" SetOutPath "$pluginsdir\webview2bootstrapper" File "tmp\MicrosoftEdgeWebview2Setup.exe" ExecWait '"$pluginsdir\webview2bootstrapper\MicrosoftEdgeWebview2Setup.exe" /silent /install' - + SetDetailsPrint both ok: -!macroend \ No newline at end of file +!macroend + +# Copy of APP_ASSOCIATE and APP_UNASSOCIATE macros from here https://gist.github.com/nikku/281d0ef126dbc215dd58bfd5b3a5cd5b +!macro APP_ASSOCIATE EXT FILECLASS DESCRIPTION ICON COMMANDTEXT COMMAND + ; Backup the previously associated file class + ReadRegStr $R0 SHELL_CONTEXT "Software\Classes\.${EXT}" "" + WriteRegStr SHELL_CONTEXT "Software\Classes\.${EXT}" "${FILECLASS}_backup" "$R0" + + WriteRegStr SHELL_CONTEXT "Software\Classes\.${EXT}" "" "${FILECLASS}" + + WriteRegStr SHELL_CONTEXT "Software\Classes\${FILECLASS}" "" `${DESCRIPTION}` + WriteRegStr SHELL_CONTEXT "Software\Classes\${FILECLASS}\DefaultIcon" "" `${ICON}` + WriteRegStr SHELL_CONTEXT "Software\Classes\${FILECLASS}\shell" "" "open" + WriteRegStr SHELL_CONTEXT "Software\Classes\${FILECLASS}\shell\open" "" `${COMMANDTEXT}` + WriteRegStr SHELL_CONTEXT "Software\Classes\${FILECLASS}\shell\open\command" "" `${COMMAND}` +!macroend + +!macro APP_UNASSOCIATE EXT FILECLASS + ; Backup the previously associated file class + ReadRegStr $R0 SHELL_CONTEXT "Software\Classes\.${EXT}" `${FILECLASS}_backup` + WriteRegStr SHELL_CONTEXT "Software\Classes\.${EXT}" "" "$R0" + + DeleteRegKey SHELL_CONTEXT `Software\Classes\${FILECLASS}` +!macroend + +!macro wails.associateFiles + ; Create file associations + {{range .Info.FileAssociations}} + !insertmacro APP_ASSOCIATE "{{.Ext}}" "{{.Name}}" "{{.Description}}" "$INSTDIR\{{.IconName}}.ico" "Open with ${INFO_PRODUCTNAME}" "$INSTDIR\${PRODUCT_EXECUTABLE} $\"%1$\"" + + File "..\{{.IconName}}.ico" + {{end}} +!macroend + +!macro wails.unassociateFiles + ; Delete app associations + {{range .Info.FileAssociations}} + !insertmacro APP_UNASSOCIATE "{{.Ext}}" "{{.Name}}" + + Delete "$INSTDIR\{{.IconName}}.ico" + {{end}} +!macroend diff --git a/v2/pkg/commands/build/packager.go b/v2/pkg/commands/build/packager.go index 92ce37e9056..beca47c5bea 100644 --- a/v2/pkg/commands/build/packager.go +++ b/v2/pkg/commands/build/packager.go @@ -6,6 +6,7 @@ import ( "github.com/leaanthony/winicon" "github.com/tc-hib/winres" "github.com/tc-hib/winres/version" + "github.com/wailsapp/wails/v2/internal/project" "image" "os" "path/filepath" @@ -95,12 +96,20 @@ func packageApplicationForDarwin(options *Options) error { return err } - // Generate Icons - err = processApplicationIcon(options, resourceDir) + // Generate App Icon + err = processDarwinIcon(options.ProjectData, "appicon", resourceDir, "iconfile") if err != nil { return err } + // Generate FileAssociation Icons + for _, fileAssociation := range options.ProjectData.Info.FileAssociations { + err = processDarwinIcon(options.ProjectData, fileAssociation.IconName, resourceDir, "") + if err != nil { + return err + } + } + options.CompiledBinary = packedBinaryPath return nil @@ -124,8 +133,8 @@ func processPList(options *Options, contentsDirectory string) error { return os.WriteFile(targetFile, content, 0644) } -func processApplicationIcon(options *Options, resourceDir string) (err error) { - appIcon, err := buildassets.ReadFile(options.ProjectData, "appicon.png") +func processDarwinIcon(projectData *project.Project, iconName string, resourceDir string, destIconName string) (err error) { + appIcon, err := buildassets.ReadFile(projectData, iconName+".png") if err != nil { return err } @@ -135,7 +144,11 @@ func processApplicationIcon(options *Options, resourceDir string) (err error) { return err } - tgtBundle := filepath.Join(resourceDir, "iconfile.icns") + if destIconName == "" { + destIconName = iconName + } + + tgtBundle := filepath.Join(resourceDir, destIconName+".icns") dest, err := os.Create(tgtBundle) if err != nil { return err @@ -151,13 +164,21 @@ func processApplicationIcon(options *Options, resourceDir string) (err error) { } func packageApplicationForWindows(options *Options) error { - // Generate icon + // Generate app icon var err error - err = generateIcoFile(options) + err = generateIcoFile(options, "appicon", "icon") if err != nil { return err } + // Generate FileAssociation Icons + for _, fileAssociation := range options.ProjectData.Info.FileAssociations { + err = generateIcoFile(options, fileAssociation.IconName, "") + if err != nil { + return err + } + } + // Create syso file err = compileResources(options) if err != nil { @@ -171,13 +192,18 @@ func packageApplicationForLinux(_ *Options) error { return nil } -func generateIcoFile(options *Options) error { - content, err := buildassets.ReadFile(options.ProjectData, "appicon.png") +func generateIcoFile(options *Options, iconName string, destIconName string) error { + content, err := buildassets.ReadFile(options.ProjectData, iconName+".png") if err != nil { return err } + + if destIconName == "" { + destIconName = iconName + } + // Check ico file exists already - icoFile := buildassets.GetLocalPath(options.ProjectData, "windows/icon.ico") + icoFile := buildassets.GetLocalPath(options.ProjectData, "windows/"+destIconName+".ico") if !fs.FileExists(icoFile) { if dir := filepath.Dir(icoFile); !fs.DirExists(dir) { if err := fs.MkDirs(dir, 0755); err != nil { diff --git a/v2/pkg/options/mac/mac.go b/v2/pkg/options/mac/mac.go index 9838b145b1f..ecd2923fbbc 100644 --- a/v2/pkg/options/mac/mac.go +++ b/v2/pkg/options/mac/mac.go @@ -22,6 +22,7 @@ type Options struct { WindowIsTranslucent bool Preferences *Preferences //ActivationPolicy ActivationPolicy - About *AboutInfo + About *AboutInfo + OnFileOpen func(filePath string) `json:"-"` //URLHandlers map[string]func(string) } diff --git a/website/docs/guides/file-association.mdx b/website/docs/guides/file-association.mdx new file mode 100644 index 00000000000..ff3ca6e6b53 --- /dev/null +++ b/website/docs/guides/file-association.mdx @@ -0,0 +1,184 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + + +## Set Up File Association: +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +|:------------|:---------------------------------------------------------------------------------------------------------------------------------------------------| +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + + +### Windows +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` +5. Configure nfpm to use your scripts and files. Example: +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` +6. Build your .deb package using nfpm: +```sh +nfpm pkg --packager deb --target . +``` +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, +new instance of app is launched and file path is passed as argument to your app. +To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 7da2b65f832..1b0fe038d61 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -18,6 +18,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) +– Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) - Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) - Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) diff --git a/website/static/schemas/config.v2.json b/website/static/schemas/config.v2.json index 22a74d56258..7ab40d95fbd 100644 --- a/website/static/schemas/config.v2.json +++ b/website/static/schemas/config.v2.json @@ -72,10 +72,17 @@ "frontend:dev:serverUrl": { "type": "string", "description": "URL to a 3rd party dev server to be used to serve assets (eg. Vite). If this is set to 'auto', then the devServerUrl will be inferred from the Vite output", - "examples": [ "auto", "http://localhost:3000" ], + "examples": [ + "auto", + "http://localhost:3000" + ], "oneOf": [ - { "format": "uri" }, - { "const": "auto" } + { + "format": "uri" + }, + { + "const": "auto" + } ] }, "wailsjsdir": { @@ -87,7 +94,9 @@ "version": { "description": "Project config version", "default": "2", - "enum": [ "2" ] + "enum": [ + "2" + ] }, "outputfilename": { "type": "string", @@ -123,7 +132,9 @@ "type": "object", "description": "The application author", "properties": { - "name": { "type": "string" }, + "name": { + "type": "string" + }, "email": { "type": "string", "format": "email" @@ -156,6 +167,39 @@ "type": "string", "description": "A short comment for the app", "default": "Built using Wails (https://wails.io)" + }, + "fileAssociations": { + "type": "array", + "description": "File associations for the app", + "items": { + "type": "object", + "properties": { + "ext": { + "type": "string", + "description": "The extension (minus the leading period). e.g. png" + }, + "name": { + "type": "string", + "description": "The name. e.g. PNG File" + }, + "description": { + "type": "string", + "description": "Windows-only. The description. It is displayed on the `Type` column on Windows Explorer." + }, + "iconName": { + "type": "string", + "description": "The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows)" + }, + "role": { + "description": "macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole.", + "allOf": [ + { + "$ref": "#/definitions/BundleTypeRole" + } + ] + } + } + } } } }, @@ -185,7 +229,9 @@ } }, "dependencies": { - "garbleargs": ["obfuscated"] + "garbleargs": [ + "obfuscated" + ] }, "definitions": { "OsHook": { @@ -203,11 +249,21 @@ "description": "Build hooks for different targets.", "additionalProperties": false, "properties": { - "{GOOS}/{GOARCH}": { "$ref": "#/definitions/OsArchHook" }, - "{GOOS}/*": { "$ref": "#/definitions/OsHook" }, - "windows/*": { "$ref": "#/definitions/OsHook" }, - "linux/*": { "$ref": "#/definitions/OsHook" }, - "darwin/*": { "$ref": "#/definitions/OsHook" }, + "{GOOS}/{GOARCH}": { + "$ref": "#/definitions/OsArchHook" + }, + "{GOOS}/*": { + "$ref": "#/definitions/OsHook" + }, + "windows/*": { + "$ref": "#/definitions/OsHook" + }, + "linux/*": { + "$ref": "#/definitions/OsHook" + }, + "darwin/*": { + "$ref": "#/definitions/OsHook" + }, "*/*": { "type": "string", "description": "Executed at build level before/after a build" @@ -225,26 +281,66 @@ "description": "Executed at build level before/after a build of the specific platform" } } + }, + "BundleTypeRole": { + "description": "macOS-only. Corresponds to CFBundleTypeRole", + "oneOf": [ + { + "description": "CFBundleTypeRole.Editor. Files can be read and edited.", + "type": "string", + "enum": [ + "Editor" + ] + }, + { + "description": "CFBundleTypeRole.Viewer. Files can be read.", + "type": "string", + "enum": [ + "Viewer" + ] + }, + { + "description": "CFBundleTypeRole.Shell", + "type": "string", + "enum": [ + "Shell" + ] + }, + { + "description": "CFBundleTypeRole.QLGenerator", + "type": "string", + "enum": [ + "QLGenerator" + ] + }, + { + "description": "CFBundleTypeRole.None", + "type": "string", + "enum": [ + "None" + ] + } + ] } }, - "bindings": { - "type": "object", - "description": "Bindings configurations", - "properties": { - "ts_generation": { - "type": "object", - "description": "model.ts file generation config", - "properties": { - "prefix": { - "type": "string", - "description": "All generated JavaScript entities will be prefixed with this value" - }, - "suffix": { - "type": "string", - "description": "All generated JavaScript entities will be suffixed with this value" - } - } - } - } - } + "bindings": { + "type": "object", + "description": "Bindings configurations", + "properties": { + "ts_generation": { + "type": "object", + "description": "model.ts file generation config", + "properties": { + "prefix": { + "type": "string", + "description": "All generated JavaScript entities will be prefixed with this value" + }, + "suffix": { + "type": "string", + "description": "All generated JavaScript entities will be suffixed with this value" + } + } + } + } + } } From 30d17a760d2cde21f11f8d108ff789f63b07ff9d Mon Sep 17 00:00:00 2001 From: stffabi Date: Mon, 23 Oct 2023 11:35:18 +0200 Subject: [PATCH 24/87] [assetserver] Inject runtime/IPC into all index html files (#2203) It will also be injected into all html files returned when requesting a folder path. --- v2/pkg/assetserver/assetserver.go | 95 ++++++++++++++++++----------- v2/pkg/assetserver/body_recorder.go | 61 ++++++++++++++++++ website/src/pages/changelog.mdx | 1 + 3 files changed, 121 insertions(+), 36 deletions(-) create mode 100644 v2/pkg/assetserver/body_recorder.go diff --git a/v2/pkg/assetserver/assetserver.go b/v2/pkg/assetserver/assetserver.go index 625c3f24509..7fd6508ea10 100644 --- a/v2/pkg/assetserver/assetserver.go +++ b/v2/pkg/assetserver/assetserver.go @@ -5,7 +5,6 @@ import ( "fmt" "math/rand" "net/http" - "net/http/httptest" "strings" "golang.org/x/net/html" @@ -111,23 +110,60 @@ func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) { return } - header := rw.Header() if d.servingFromDisk { - header.Add(HeaderCacheControl, "no-cache") + rw.Header().Add(HeaderCacheControl, "no-cache") + } + + handler := d.handler + if req.Method != http.MethodGet { + handler.ServeHTTP(rw, req) + return } path := req.URL.Path - switch path { - case "", "/", "/index.html": - recorder := httptest.NewRecorder() - d.handler.ServeHTTP(recorder, req) - for k, v := range recorder.HeaderMap { - header[k] = v + if path == runtimeJSPath { + d.writeBlob(rw, path, d.runtimeJS) + + } else if path == runtimePath && d.runtimeHandler != nil { + d.runtimeHandler.HandleRuntimeCall(rw, req) + + } else if path == ipcJSPath { + content := d.runtime.DesktopIPC() + if d.ipcJS != nil { + content = d.ipcJS(req) + } + d.writeBlob(rw, path, content) + + } else if script, ok := d.pluginScripts[path]; ok { + d.writeBlob(rw, path, []byte(script)) + + } else if d.isRuntimeInjectionMatch(path) { + recorder := &bodyRecorder{ + ResponseWriter: rw, + doRecord: func(code int, h http.Header) bool { + if code == http.StatusNotFound { + return true + } + + if code != http.StatusOK { + return false + } + + return strings.Contains(h.Get(HeaderContentType), "text/html") + }} + + handler.ServeHTTP(recorder, req) + + body := recorder.Body() + if body == nil { + // The body has been streamed and not recorded, we are finished + return } - switch recorder.Code { + code := recorder.Code() + switch code { case http.StatusOK: - content, err := d.processIndexHTML(recorder.Body.Bytes()) + content, err := d.processIndexHTML(body.Bytes()) if err != nil { d.serveError(rw, err, "Unable to processIndexHTML") return @@ -138,34 +174,12 @@ func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) { d.writeBlob(rw, indexHTML, defaultHTML) default: - rw.WriteHeader(recorder.Code) - - } - - case runtimeJSPath: - d.writeBlob(rw, path, d.runtimeJS) + rw.WriteHeader(code) - case runtimePath: - if d.runtimeHandler != nil { - d.runtimeHandler.HandleRuntimeCall(rw, req) - } else { - d.handler.ServeHTTP(rw, req) } - case ipcJSPath: - content := d.runtime.DesktopIPC() - if d.ipcJS != nil { - content = d.ipcJS(req) - } - d.writeBlob(rw, path, content) - - default: - // Check if this is a plugin script - if script, ok := d.pluginScripts[path]; ok { - d.writeBlob(rw, path, []byte(script)) - return - } - d.handler.ServeHTTP(rw, req) + } else { + handler.ServeHTTP(rw, req) } } @@ -229,3 +243,12 @@ func (d *AssetServer) logError(message string, args ...interface{}) { d.logger.Error("[AssetServer] "+message, args...) } } + +func (AssetServer) isRuntimeInjectionMatch(path string) bool { + if path == "" { + path = "/" + } + + return strings.HasSuffix(path, "/") || + strings.HasSuffix(path, "/"+indexHTML) +} diff --git a/v2/pkg/assetserver/body_recorder.go b/v2/pkg/assetserver/body_recorder.go new file mode 100644 index 00000000000..fa3bc1e7cb3 --- /dev/null +++ b/v2/pkg/assetserver/body_recorder.go @@ -0,0 +1,61 @@ +package assetserver + +import ( + "bytes" + "net/http" +) + +type bodyRecorder struct { + http.ResponseWriter + doRecord func(code int, header http.Header) bool + + body *bytes.Buffer + code int + wroteHeader bool +} + +func (rw *bodyRecorder) Write(buf []byte) (int, error) { + rw.writeHeader(buf, http.StatusOK) + if rw.body != nil { + return rw.body.Write(buf) + } + return rw.ResponseWriter.Write(buf) +} + +func (rw *bodyRecorder) WriteHeader(code int) { + rw.writeHeader(nil, code) +} + +func (rw *bodyRecorder) Code() int { + return rw.code +} + +func (rw *bodyRecorder) Body() *bytes.Buffer { + return rw.body +} + +func (rw *bodyRecorder) writeHeader(buf []byte, code int) { + if rw.wroteHeader { + return + } + + if rw.doRecord != nil { + header := rw.Header() + if len(buf) != 0 { + if _, hasType := header[HeaderContentType]; !hasType { + header.Set(HeaderContentType, http.DetectContentType(buf)) + } + } + + if rw.doRecord(code, header) { + rw.body = bytes.NewBuffer(nil) + } + } + + if rw.body == nil { + rw.ResponseWriter.WriteHeader(code) + } + + rw.code = code + rw.wroteHeader = true +} diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 1b0fe038d61..10f80d73749 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -25,6 +25,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) ### Changed From a59f8b2cf30728db7868f02dc85f2e414ad8d709 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Mon, 23 Oct 2023 20:38:03 +1100 Subject: [PATCH 25/87] docs: sync translations (#2893) Co-authored-by: leaanthony --- .../current/community/templates.mdx | 4 + .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/file-association.mdx | 199 ++++ .../current/guides/troubleshooting.mdx | 182 +++- .../current/reference/cli.mdx | 62 +- .../current/reference/options.mdx | 44 +- .../changelog.mdx | 27 +- .../current/community/templates.mdx | 6 +- .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/file-association.mdx | 199 ++++ .../current/guides/troubleshooting.mdx | 182 +++- .../current/reference/cli.mdx | 62 +- .../current/reference/options.mdx | 44 +- .../current/reference/runtime/window.mdx | 2 +- .../version-v2.6.0.json | 16 +- .../changelog.mdx | 27 +- .../current/community/templates.mdx | 4 + .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/file-association.mdx | 199 ++++ .../current/guides/troubleshooting.mdx | 182 +++- .../current/reference/cli.mdx | 62 +- .../current/reference/options.mdx | 44 +- .../changelog.mdx | 27 +- .../2021-09-27-v2-beta1-release-notes.mdx | 90 +- .../2021-11-08-v2-beta2-release-notes.mdx | 88 +- .../2022-02-22-v2-beta3-release-notes.mdx | 62 +- .../2022-09-22-v2-release-notes.mdx | 92 +- .../2023-01-17-v3-roadmap.mdx | 154 ++-- .../current/community/links.mdx | 2 +- .../community/showcase/bulletinboard.mdx | 2 +- .../current/community/showcase/emailit.mdx | 2 +- .../current/community/showcase/filehound.mdx | 2 +- .../current/community/showcase/hiposter.mdx | 2 +- .../community/showcase/modalfilemanager.mdx | 4 +- .../current/community/showcase/scriptbar.mdx | 2 +- .../current/community/showcase/warmine.mdx | 8 +- .../current/community/templates.mdx | 20 +- .../current/gettingstarted/development.mdx | 2 +- .../current/guides/angular.mdx | 6 +- .../guides/application-development.mdx | 8 +- .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/dynamic-assets.mdx | 28 +- .../current/guides/file-association.mdx | 199 ++++ .../current/guides/frameless.mdx | 12 +- .../current/guides/local-development.mdx | 34 +- .../current/guides/mac-appstore.mdx | 74 +- .../current/guides/migrating.mdx | 66 +- .../current/guides/mouse-buttons.mdx | 4 +- .../current/guides/obfuscated.mdx | 20 +- .../current/guides/overscroll.mdx | 4 +- .../current/guides/routing.mdx | 2 +- .../current/guides/sveltekit.mdx | 100 +- .../current/guides/templates.mdx | 74 +- .../current/guides/troubleshooting.mdx | 268 +++++- .../current/reference/cli.mdx | 62 +- .../current/reference/menus.mdx | 86 +- .../current/reference/options.mdx | 44 +- .../version-v2.3.1.json | 16 +- .../changelog.mdx | 27 +- .../current/community/templates.mdx | 4 + .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/file-association.mdx | 199 ++++ .../current/guides/troubleshooting.mdx | 182 +++- .../current/reference/cli.mdx | 62 +- .../current/reference/options.mdx | 44 +- .../changelog.mdx | 27 +- website/i18n/vi/code.json | 431 +++++++++ .../2021-09-27-v2-beta1-release-notes.mdx | 161 ++++ .../2021-11-08-v2-beta2-release-notes.mdx | 170 ++++ .../2022-02-22-v2-beta3-release-notes.mdx | 114 +++ .../2022-09-22-v2-release-notes.mdx | 98 ++ .../2023-01-17-v3-roadmap.mdx | 184 ++++ .../authors.yml | 10 + .../options.json | 14 + .../current.json | 38 + .../current/community/links.mdx | 26 + .../community/showcase/bulletinboard.mdx | 10 + .../current/community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 12 + .../current/community/showcase/filehound.mdx | 16 + .../current/community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../current/community/showcase/october.mdx | 14 + .../current/community/showcase/optimus.mdx | 10 + .../current/community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 12 + .../current/community/showcase/riftshare.mdx | 21 + .../current/community/showcase/scriptbar.mdx | 10 + .../current/community/showcase/surge.mdx | 10 + .../current/community/showcase/wally.mdx | 10 + .../current/community/showcase/warmine.mdx | 19 + .../current/community/showcase/wombat.mdx | 10 + .../current/community/showcase/ytd.mdx | 10 + .../current/community/templates.mdx | 69 ++ .../current/gettingstarted/building.mdx | 22 + .../current/gettingstarted/development.mdx | 16 + .../current/gettingstarted/firstproject.mdx | 130 +++ .../current/gettingstarted/installation.mdx | 90 ++ .../current/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/dynamic-assets.mdx | 136 +++ .../current/guides/file-association.mdx | 199 ++++ .../current/guides/frameless.mdx | 87 ++ .../current/guides/frontend.mdx | 72 ++ .../current/guides/ides.mdx | 127 +++ .../current/guides/linux-distro-support.mdx | 103 +++ .../current/guides/linux.mdx | 18 + .../current/guides/local-development.mdx | 55 ++ .../current/guides/mac-appstore.mdx | 97 ++ .../current/guides/manual-builds.mdx | 95 ++ .../current/guides/migrating.mdx | 191 ++++ .../current/guides/mouse-buttons.mdx | 25 + .../current/guides/obfuscated.mdx | 40 + .../current/guides/overscroll.mdx | 10 + .../current/guides/routing.mdx | 47 + .../current/guides/signing.mdx | 387 ++++++++ .../current/guides/sveltekit.mdx | 153 ++++ .../current/guides/templates.mdx | 97 ++ .../current/guides/troubleshooting.mdx | 368 ++++++++ .../current/guides/vscode.mdx | 82 ++ .../current/guides/windows-installer.mdx | 58 ++ .../current/guides/windows.mdx | 61 ++ .../current/howdoesitwork.mdx | 369 ++++++++ .../current/introduction.mdx | 75 ++ .../current/reference/cli.mdx | 240 +++++ .../current/reference/menus.mdx | 230 +++++ .../current/reference/options.mdx | 853 ++++++++++++++++++ .../current/reference/project-config.mdx | 92 ++ .../current/reference/runtime/browser.mdx | 13 + .../current/reference/runtime/clipboard.mdx | 23 + .../current/reference/runtime/dialog.mdx | 302 +++++++ .../current/reference/runtime/events.mdx | 37 + .../current/reference/runtime/intro.mdx | 85 ++ .../current/reference/runtime/log.mdx | 130 +++ .../current/reference/runtime/menu.mdx | 25 + .../current/reference/runtime/screen.mdx | 38 + .../current/reference/runtime/window.mdx | 227 +++++ .../current/tutorials/dogsapi.mdx | 245 +++++ .../current/tutorials/helloworld.mdx | 123 +++ .../version-v2.3.1.json | 38 + .../version-v2.4.0.json | 38 + .../version-v2.5.0.json | 38 + .../version-v2.6.0.json | 38 + .../changelog.mdx | 751 +++++++++++++++ .../docusaurus-plugin-content-pages/coc.mdx | 83 ++ .../community-guide.mdx | 142 +++ .../credits.mdx | 268 ++++++ .../docusaurus-plugin-content-pages/faq.mdx | 9 + .../markdown-page.md | 7 + .../vi/docusaurus-theme-classic/footer.json | 62 ++ .../vi/docusaurus-theme-classic/navbar.json | 46 + .../current/community/templates.mdx | 4 + .../current/guides/crossplatform-build.mdx | 66 ++ .../current/guides/file-association.mdx | 199 ++++ .../current/guides/manual-builds.mdx | 1 - .../current/guides/troubleshooting.mdx | 182 +++- .../current/reference/cli.mdx | 62 +- .../current/reference/options.mdx | 44 +- .../changelog.mdx | 27 +- 162 files changed, 13107 insertions(+), 801 deletions(-) create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx create mode 100644 website/i18n/vi/code.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/authors.yml create mode 100644 website/i18n/vi/docusaurus-plugin-content-blog/options.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/links.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/mollywallet.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/october.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/optimus.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/portfall.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/restic-browser.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/riftshare.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/surge.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wally.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wombat.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/ytd.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/templates.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/building.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/firstproject.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/installation.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/angular.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/application-development.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frameless.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frontend.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/ides.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux-distro-support.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/local-development.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/migrating.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/overscroll.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/routing.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/signing.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/templates.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/vscode.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows-installer.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/howdoesitwork.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/cli.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/menus.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/options.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/project-config.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/browser.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/clipboard.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/dialog.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/events.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/intro.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/log.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/menu.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/screen.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/dogsapi.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/helloworld.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/version-v2.3.1.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/version-v2.4.0.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/version-v2.5.0.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/version-v2.6.0.json create mode 100644 website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-pages/coc.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-pages/community-guide.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-pages/faq.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-pages/markdown-page.md create mode 100644 website/i18n/vi/docusaurus-theme-classic/footer.json create mode 100644 website/i18n/vi/docusaurus-theme-classic/navbar.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx index b4ab74e2363..d1b9913b1ec 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -60,6 +60,10 @@ Si vous n'êtes pas sûr d'un modèle, inspectez `package.json` et `wails.json` - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Développez votre application GUI avec de la programmation fonctionnelle et une configuration de développement en direct :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine les puissances :muscle: d'Elm + Tailwind CSS + Wails ! Rechargement automatique pris en charge. +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + ## Pure JavaScript (Vanilla) - [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - Un modèle avec rien que du JavaScript, du HTML et du CSS de base diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx index 5b319f9a447..1c3b6a9cc52 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -159,6 +159,28 @@ If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-se Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + -- ## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" @@ -185,4 +207,162 @@ This is due to the default background of the webview being white. If you want to WebviewIsTransparent: true, }, }) -``` \ No newline at end of file +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/cli.mdx index 1c9bea69494..30721c11204 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/cli.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -49,35 +49,35 @@ Si vous n'êtes pas sûr d'un modèle, inspectez les fichiers `package.json` et `wails build` est utilisé pour compiler votre projet vers un binaire prêt à la production. -| Option | Description | Par défaut | -|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -| -clean | Nettoie le répertoire `build/bin` | | -| -compiler "compiler" | Utiliser un autre compilateur pour compiler, par exemple go1.15beta1 | go | -| -debug | Conserve les informations de débogage dans l'application et affiche la console de débogage. Permet l'utilisation des outils de développement dans la fenêtre de l'application | | -| -devtools | Permet l'utilisation des devtools dans la fenêtre d'application en production (quand -debug n'est pas utilisé) | | -| -dryrun | Affiche la commande build sans l'exécuter | | -| -f | Forcer la compilation de l'application | | -| -garbleargs | Arguments à passer à garble | `-literals -tiny -seed=random` | -| -ldflags "flags" | Options supplémentaires à passer au compilateur | | -| -m | Permet d'ignorer mod tidy avant la compilation | | -| -nopackage | Ne pas empaqueter l'application | | -| -nocolour | Désactive la couleur des logs dans le terminal | | -| -nosyncgomod | Ne pas synchroniser go.mod avec la version Wails | | -| -nsis | Génère l'installateur NSIS pour Windows | | -| -o filename | Nom du fichier de sortie | | -| -obfuscated | Cacher le code de l'application en utilisant [garble](https://github.com/burrowers/garble) | | -| -platform | Construit pour les [plates-formes](../reference/cli.mdx#platforms) données (séparées par des virgules) par exemple. `windows/arm64`. Notez que si vous ne donnez pas l'architecture, `runtime.GOARCH` est utilisé. | platform = le contenu de la variable d'environnement `GOOS` si elle existe, autrement `runtime.GOOS`.
arch = le contenu de la variable d'environnement `GOARCH` si elle existe, autrement `runtime.GOARCH`. | -| -race | Construire avec le détecteur Go race | | -| -s | Ignorer la construction du frontend | | -| -skipbindings | Ignorer la génération des liaisons | | -| -tags "extra tags" | Options de compilation à passer au compilateur Go. Doivent être entre guillemets. Séparés par un espace ou une virgule (pas les deux) | | -| -trimpath | Supprimer tous les chemins vers les fichiers système de l'exécutable final. | | -| -u | Met à jour le `go.mod de votre projet` pour utiliser la même version de Wails que le CLI | | -| -upx | Compresser le binaire final en utilisant "upx" | | -| -upxflags | Options à passer à upx | | -| -v int | Niveau de verbosité (0 - silencieux, 1 - par défaut, 2 - verbeux) | 1 | -| -webview2 | Stratégie d'installation WebView2 : download,embed,browser,error | download | -| -windowsconsole | Garder la fenêtre de la console lors de la construction d'une version pour Windows | | +| Option | Description | Par défaut | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Nettoie le répertoire `build/bin` | | +| -compiler "compiler" | Utiliser un autre compilateur pour compiler, par exemple go1.15beta1 | go | +| -debug | Conserve les informations de débogage dans l'application et affiche la console de débogage. Permet l'utilisation des outils de développement dans la fenêtre de l'application | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Affiche la commande build sans l'exécuter | | +| -f | Forcer la compilation de l'application | | +| -garbleargs | Arguments à passer à garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Options supplémentaires à passer au compilateur | | +| -m | Permet d'ignorer mod tidy avant la compilation | | +| -nopackage | Ne pas empaqueter l'application | | +| -nocolour | Désactive la couleur des logs dans le terminal | | +| -nosyncgomod | Ne pas synchroniser go.mod avec la version Wails | | +| -nsis | Génère l'installateur NSIS pour Windows | | +| -o filename | Nom du fichier de sortie | | +| -obfuscated | Cacher le code de l'application en utilisant [garble](https://github.com/burrowers/garble) | | +| -platform | Construit pour les [plates-formes](../reference/cli.mdx#platforms) données (séparées par des virgules) par exemple. `windows/arm64`. Notez que si vous ne donnez pas l'architecture, `runtime.GOARCH` est utilisé. | platform = le contenu de la variable d'environnement `GOOS` si elle existe, autrement `runtime.GOOS`.
arch = le contenu de la variable d'environnement `GOARCH` si elle existe, autrement `runtime.GOARCH`. | +| -race | Construire avec le détecteur Go race | | +| -s | Ignorer la construction du frontend | | +| -skipbindings | Ignorer la génération des liaisons | | +| -tags "extra tags" | Options de compilation à passer au compilateur Go. Doivent être entre guillemets. Séparés par un espace ou une virgule (pas les deux) | | +| -trimpath | Supprimer tous les chemins vers les fichiers système de l'exécutable final. | | +| -u | Met à jour le `go.mod de votre projet` pour utiliser la même version de Wails que le CLI | | +| -upx | Compresser le binaire final en utilisant "upx" | | +| -upxflags | Options à passer à upx | | +| -v int | Niveau de verbosité (0 - silencieux, 1 - par défaut, 2 - verbeux) | 1 | +| -webview2 | Stratégie d'installation WebView2 : download,embed,browser,error | download | +| -windowsconsole | Garder la fenêtre de la console lors de la construction d'une version pour Windows | | Pour une description détaillée des options `webview2` , veuillez vous référer au Guide de [Windows](../guides/windows.mdx). @@ -167,8 +167,8 @@ Your system is ready for Wails development! - Un serveur web est lancé sur `http://localhost:34115` qui sert votre application (et pas seulement le frontend) sur http. Cela vous permet d'utiliser les extensions de développement de votre navigateur favori - Tous les assets de l'application sont chargés à partir du disque. Si elles sont modifiées, l'application se rechargera automatiquement (pas de recompilation). Tous les navigateurs connectés rechargeront également - Un module JS est généré fournissant les éléments suivants : - - Les méthodes Javascript permettant d'appeler vos méthodes Go avec JSDoc autogénérée, vous fournissant des indications sur les méthodes - - Les versions TypeScript de vos structures Go, qui peuvent être construites et transmises à vos méthodes +- Les méthodes Javascript permettant d'appeler vos méthodes Go avec JSDoc autogénérée, vous fournissant des indications sur les méthodes +- Les versions TypeScript de vos structures Go, qui peuvent être construites et transmises à vos méthodes - Un second module JS est généré qui fournit une déclaration des méthodes et structures pour l'exécutable - Sur macOS, il regroupera l'application dans un fichier `.app` et l'exécutera. Il utilisera un `build/darwin/Info.dev.plist` pour le développement. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/options.mdx index 1d3166ee89d..a80e3526732 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/options.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -366,7 +366,7 @@ Nom: CSSDragValue
Type: `string` EnableDefaultContextMenu active le menu contextuel par défaut du navigateur en production. -Par défaut, le menu contextuel par défaut du navigateur n'est disponible qu'en développement et dans un `-debug` ou `-devtools` [build](../reference/cli.mdx#build) avec l'inspecteur de devtools, En utilisant cette option, vous pouvez activer le menu contextuel par défaut dans `production` alors que l'inspecteur devtools ne sera pas disponible à moins que le drapeau `-devtools` ne soit utilisé. +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. Lorsque cette option est activée, par défaut, le menu contextuel ne sera affiché que pour du texte (où Couper/Copier/Coller est nécessaire), pour remplacer ce comportement, vous pouvez utiliser la propriété CSS `--default-contextmenu` sur n'importe quel élément HTML (y compris le corps ``) avec les valeurs suivantes : @@ -593,6 +593,12 @@ Définir ceci à `true` désactivera l'accélération matérielle GPU pour la we Nom: WebviewGpuIsDisabled
Type: `bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + ### Mac Ceci définit [les options spécifiques à Mac](#mac). @@ -688,6 +694,42 @@ Définir ceci à `true` rendra l'arrière-plan de la fenêtre translucide. Souve Nom: WindowIsTranslucent
Type: `bool` +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Nom | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Exemple: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### About Cette configuration vous permet de définir le titre, le message et l'icône pour l'élément de menu "À propos" dans le menu de l'application créé par le rôle "AppMenu". diff --git a/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx index f8f9c2ffaca..b1b51b2277c 100644 --- a/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx @@ -13,6 +13,28 @@ Le format est basé sur [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) ## [Unreleased] +### Ajouts + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### Changements + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### Corrections + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + ## v2.6.0 - 2023-09-06 ### Modifications importantes @@ -27,6 +49,7 @@ Le format est basé sur [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) - Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) - Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) - Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) - Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) - Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) @@ -44,6 +67,7 @@ Le format est basé sur [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) - Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) - Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) - Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) ### Changements @@ -680,9 +704,6 @@ Experimental: &options.Experimental{ - [v2, nsis] On dirait que / comme séparateur de chemin ne fonctionne que pour certaines directives de manière cross-platform par [@stffabi](https://github.com/stffabi) dans #1227 - import des modèles sur la définition de bindings par [@adalessa](https://github.com/adalessa) dans #123 - - 1 - - Utilisation de la recherche locale sur le site web par [@leaanthony](https://github.com/leaanthony) en #1234 - Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 - Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/templates.mdx index 9fc4a8d92af..3dc92d2fbd9 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -47,7 +47,7 @@ sidebar_position: 1 - [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - Svelteを使用したテンプレート - [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - SvelteおよびViteを使用したテンプレート - [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - TailwindCSS v3を含んだ、SvelteおよびViteを使用したテンプレート -- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - Svelte v4.2.0、Vite、TailwindCSS v3.3.3を使用したテンプレート - [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - SvelteKitを使用したテンプレート ## Solid @@ -60,6 +60,10 @@ sidebar_position: 1 - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - 関数型プログラミングと**高速な**ホットリロードを使ったGUIアプリ開発 :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Elm + Tailwind CSS + Wailsのパワー:muscle:を組み合わせたテンプレート (ホットリロードサポートあり) +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - インタラクティビティを実現するためのピュアなhtmxに、コンポーネントおよびフォームを作成するためのテンプレートが合わさったテンプレート + ## ピュアJavaScript (バニラ) - [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - 基本的なJavaScript、HTML、CSSのみを含むテンプレート diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx index 0746f225c8c..eb93618657c 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -159,6 +159,28 @@ XCodeコマンドラインツールの再インストールが引き続き失敗 出典: https://github.com/wailsapp/wails/issues/1806 および https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 +## My application won't compile on Mac + +次のようなエラーが発生する場合: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + -- ## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" @@ -185,4 +207,162 @@ This is due to the default background of the webview being white. If you want to WebviewIsTransparent: true, }, }) -``` \ No newline at end of file +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/cli.mdx index 6c83ddd938d..3a7ff0a1ae5 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/cli.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -49,35 +49,35 @@ WailsではGitHubでホストされているリモートテンプレートをサ `wails build`は、プロジェクトを本番配布用のバイナリにコンパイルするときに使用します。 -| フラグ | 説明 | デフォルト | -|:-------------------- |:-------------------------------------------------------------------------------------------------------------------------------- |:--------------------------------------------------------------------------------------------------------------------------------------------------- | -| -clean | `build/bin`ディレクトリをクリーンする | | -| -compiler "compiler" | 違うGoコンパイラを使用する。例: go1.15beta1 | go | -| -debug | アプリケーションのデバッグ情報を保持し、デバッグコンソールを表示する。 これにより、アプリケーションウィンドウで開発者ツールを使用することを許可できます。 | | -| -devtools | 本番用のアプリケーションウィンドウにおいて開発者ツールの使用を許可する (-debugが使用されていないとき) | | -| -dryrun | 実際には実行せずにbuildコマンドの結果を表示する | | -| -f | アプリケーションを強制的にビルド | | -| -garbleargs | garbleへ渡す引数 | `-literals -tiny -seed=random` | -| -ldflags "flags" | コンパイラに渡す追加のldflags | | -| -m | コンパイル前のmod tidyの実行をスキップする | | -| -nopackage | アプリケーションをパッケージ化しない | | -| -nocolour | 出力文字に色をつけない | | -| -nosyncgomod | go.modとWailsのバージョンを同期させない | | -| -nsis | Windows向けのNSISインストーラを生成する | | -| -o filename | 出力ファイル名 | | -| -obfuscated | [garble](https://github.com/burrowers/garble)を使用してアプリケーションを難読化する | | -| -platform | 指定された[プラットフォーム](../reference/cli.mdx#platforms)(カンマ区切り) 向けにビルドする。例: `windows/arm64`。 アーキテクチャを指定しない場合は、`runtime.GOARCH`の値が使用されます。 | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | -| -race | Goのrace detectorを使用してビルドする | | -| -s | フロントエンドのビルドをスキップ | | -| -skipbindings | バインディングの生成をスキップする | | -| -tags "extra tags" | Goコンパイラに渡すビルドタグ。 値は引用符で囲んでください。 また、スペースまたはカンマで区切ってください(両方は使用しないでください)。 | | -| -trimpath | 実行可能ファイルから、すべてのファイルシステムパスを削除する | | -| -u | プロジェクトの`go.mod`を更新し、CLIと同じバージョンのWailsを使用する | | -| -upx | "upx"を使用して最終的にバイナリを圧縮する | | -| -upxflags | upxに渡すフラグ | | -| -v int | 詳細度レベル (0 - サイレント, 1 - デフォルト, 2 - 詳細) | 1 | -| -webview2 | WebView2インストーラーのストラテジ: download,embed,browser,error | download | -| -windowsconsole | Windiws向けビルドでコンソールウィンドウを維持する | | +| フラグ | 説明 | デフォルト | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | `build/bin`ディレクトリをクリーンする | | +| -compiler "compiler" | 違うGoコンパイラを使用する。例: go1.15beta1 | go | +| -debug | アプリケーションのデバッグ情報を保持し、デバッグコンソールを表示する。 これにより、アプリケーションウィンドウで開発者ツールを使用することを許可できます。 | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | 実際には実行せずにbuildコマンドの結果を表示する | | +| -f | アプリケーションを強制的にビルド | | +| -garbleargs | garbleへ渡す引数 | `-literals -tiny -seed=random` | +| -ldflags "flags" | コンパイラに渡す追加のldflags | | +| -m | コンパイル前のmod tidyの実行をスキップする | | +| -nopackage | アプリケーションをパッケージ化しない | | +| -nocolour | 出力文字に色をつけない | | +| -nosyncgomod | go.modとWailsのバージョンを同期させない | | +| -nsis | Windows向けのNSISインストーラを生成する | | +| -o filename | 出力ファイル名 | | +| -obfuscated | [garble](https://github.com/burrowers/garble)を使用してアプリケーションを難読化する | | +| -platform | 指定された[プラットフォーム](../reference/cli.mdx#platforms)(カンマ区切り) 向けにビルドする。例: `windows/arm64`。 アーキテクチャを指定しない場合は、`runtime.GOARCH`の値が使用されます。 | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Goのrace detectorを使用してビルドする | | +| -s | フロントエンドのビルドをスキップ | | +| -skipbindings | バインディングの生成をスキップする | | +| -tags "extra tags" | Goコンパイラに渡すビルドタグ。 値は引用符で囲んでください。 また、スペースまたはカンマで区切ってください(両方は使用しないでください)。 | | +| -trimpath | 実行可能ファイルから、すべてのファイルシステムパスを削除する | | +| -u | プロジェクトの`go.mod`を更新し、CLIと同じバージョンのWailsを使用する | | +| -upx | "upx"を使用して最終的にバイナリを圧縮する | | +| -upxflags | upxに渡すフラグ | | +| -v int | 詳細度レベル (0 - サイレント, 1 - デフォルト, 2 - 詳細) | 1 | +| -webview2 | WebView2インストーラーのストラテジ: download,embed,browser,error | download | +| -windowsconsole | Windiws向けビルドでコンソールウィンドウを維持する | | `webview2`フラグの詳細については、[Windows](../guides/windows.mdx)ガイドをご覧ください。 @@ -166,8 +166,8 @@ Your system is ready for Wails development! - `http://localhost:34115`でWebサーバが起動し、HTTP経由でアプリケーション(フロントエンドだけではありません)が提供されます。 これにより、任意のブラウザ拡張機能を使用することができます - すべてのアプリケーションアセットはディスクから読み込まれます。 アセットが変更された場合、アプリケーションは自動的に、リビルドではなくリロードされます。 接続されているすべてのブラウザもリロードされます - 以下のものを含むJSモジュールが生成されます: - - GoメソッドのJavaScriptラッパー (コードヒントに有用なJSDocも自動付与されています) - - Goの構造体のTypeScriptバージョン (構造体のインスタンスを生成したり、Goメソッドの引数として渡したりすることができます) +- GoメソッドのJavaScriptラッパー (コードヒントに有用なJSDocも自動付与されています) +- Goの構造体のTypeScriptバージョン (構造体のインスタンスを生成したり、Goメソッドの引数として渡したりすることができます) - 別のJSモジュールとして、ランタイムのラッパーおよびTS定義も生成されます - macOSの場合、アプリケーションは`.app`ファイルにバンドルされて実行されます。 これには、開発用の`build/darwin/Info.dev.plist`を使用します。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/options.mdx index 8f9a73bb97e..2f53915a0ea 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/options.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -366,7 +366,7 @@ func (b *App) beforeClose(ctx context.Context) (prevent bool) { EnableDefaultContextMenuは、本番環境において、ブラウザのデフォルトコンテキストメニューを有効にします。 -通常、ブラウザのデフォルトコンテキストメニューは、開発環境での動作時、または`-debug`・`-devtools`フラグをつけて開発者ツールを有効にして[ビルド](../reference/cli.mdx#build)したときのみ利用できますが、本オプションを使うと、`-devtools`フラグをつけない限り開発者ツールは使用できませんが、`本番`環境でもコンテキストメニューを有効にすることができます。 +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. このオプションを有効にすると、デフォルトでは、テキストに関するコンテキスト(切り取り/コピー/貼り付け) のみがコンテキストメニューに表示されます。この動作をオーバーライドするには、`--default-contextmenu`というCSSプロパティを任意のHTML要素(`body`含む) で以下の値と共に使用してください: @@ -593,6 +593,12 @@ Windowsがローパワーモード(サスペンド/休止状態) から復帰し 名前: WebviewGpuIsDisabled
データ型: `bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + ### Mac [Mac固有のオプション](#mac)を定義します。 @@ -688,6 +694,42 @@ Mac: &mac.Options{ 名前: WindowIsTranslucent
データ型: `bool` +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| 名前 | 説明 | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +例: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### About "AppMenu"ロールで作成されたアプリケーションメニューの中にある"About"メニュー項目において、タイトル、メッセージ、アイコンを設定できます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx index d506cce7c83..ea6dd358d5b 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx @@ -202,7 +202,7 @@ Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: ### WindowPrint -Opens tha native print dialog. +ネイティブな印刷ダイアログを開きます。 Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.6.0.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.6.0.json index a75900f45a1..3d895d3ebe5 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.6.0.json +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.6.0.json @@ -4,35 +4,35 @@ "description": "The label for version v2.6.0" }, "sidebar.docs.category.Getting Started": { - "message": "Getting Started", + "message": "はじめよう", "description": "The label for category Getting Started in sidebar docs" }, "sidebar.docs.category.Reference": { - "message": "Reference", + "message": "リファレンス", "description": "The label for category Reference in sidebar docs" }, "sidebar.docs.category.Runtime": { - "message": "Runtime", + "message": "ランタイム", "description": "The label for category Runtime in sidebar docs" }, "sidebar.docs.category.Community": { - "message": "Community", + "message": "コミュニティ", "description": "The label for category Community in sidebar docs" }, "sidebar.docs.category.Showcase": { - "message": "Showcase", + "message": "事例紹介", "description": "The label for category Showcase in sidebar docs" }, "sidebar.docs.category.Guides": { - "message": "Guides", + "message": "ガイド", "description": "The label for category Guides in sidebar docs" }, "sidebar.docs.category.Tutorials": { - "message": "Tutorials", + "message": "チュートリアル", "description": "The label for category Tutorials in sidebar docs" }, "sidebar.docs.link.Contributing": { - "message": "Contributing", + "message": "コントリビューション", "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" } } diff --git a/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx index 3cda7c1c716..5828a733f3a 100644 --- a/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx @@ -13,6 +13,28 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), ## [Unreleased] +### Added + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### Changed + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### Fixed + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + ## v2.6.0 - 2023-09-06 ### Breaking Changes @@ -27,6 +49,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) - Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) - Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) - Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) - Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) @@ -44,6 +67,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) - Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) - Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) ### Changed @@ -680,9 +704,6 @@ Experimental: &options.Experimental{ - [v2, nsis] Seems like / as path separator works only for some directives in a cross platform way by [@stffabi](https://github.com/stffabi) in #1227 - import models on binding definition by [@adalessa](https://github.com/adalessa) in #123 - - 1 - - Use local search on website by [@leaanthony](https://github.com/leaanthony) in #1234 - Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 - Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/templates.mdx index 3e000705ff4..6ecbac37ee8 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -60,6 +60,10 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + ## Pure JavaScript (Vanilla) - [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx index 28482ac3dd9..2b5b7a25840 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -159,6 +159,28 @@ If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-se Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + -- ## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" @@ -185,4 +207,162 @@ This is due to the default background of the webview being white. If you want to WebviewIsTransparent: true, }, }) -``` \ No newline at end of file +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/cli.mdx index a4817110fc8..c7aea10a64d 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/cli.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -49,35 +49,35 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for `wails build` is used for compiling your project to a production-ready binary. -| Flag | Description | Default | -|:-------------------- |:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:--------------------------------------------------------------------------------------------------------------------------------------------------- | -| -clean | Cleans the `build/bin` directory | | -| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | -| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | -| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used) | | -| -dryrun | Prints the build command without executing it | | -| -f | Force build application | | -| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | -| -ldflags "flags" | Additional ldflags to pass to the compiler | | -| -m | Skip mod tidy before compile | | -| -nopackage | Do not package application | | -| -nocolour | Disable colour in output | | -| -nosyncgomod | Do not sync go.mod with the Wails version | | -| -nsis | Generate NSIS installer for Windows | | -| -o filename | Output filename | | -| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | -| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | -| -race | Build with Go's race detector | | -| -s | Skip building the frontend | | -| -skipbindings | Skip bindings generation | | -| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | -| -trimpath | Remove all file system paths from the resulting executable. | | -| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | -| -upx | Compress final binary using "upx" | | -| -upxflags | Flags to pass to upx | | -| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | -| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | -| -windowsconsole | Keep the console window for Windows builds | | +| Flag | Description | Default | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Cleans the `build/bin` directory | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Force build application | | +| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -m | Skip mod tidy before compile | | +| -nopackage | Do not package application | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o filename | Output filename | | +| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | +| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Build with Go's race detector | | +| -s | Skip building the frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | +| -trimpath | Remove all file system paths from the resulting executable. | | +| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | +| -upx | Compress final binary using "upx" | | +| -upxflags | Flags to pass to upx | | +| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | +| -windowsconsole | Keep the console window for Windows builds | | For a detailed description of the `webview2` flag, please refer to the [Windows](../guides/windows.mdx) Guide. @@ -166,8 +166,8 @@ Your system is ready for Wails development! - A webserver is started on `http://localhost:34115` which serves your application (not just frontend) over http. This allows you to use your favourite browser development extensions - All application assets are loaded from disk. If they are changed, the application will automatically reload (not rebuild). All connected browsers will also reload - A JS module is generated that provides the following: - - JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting - - TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods - A second JS module is generated that provides a wrapper + TS declaration for the runtime - On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/options.mdx index e1bb970a127..46d1c071b60 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/options.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -366,7 +366,7 @@ Name: CSSDragValue
Type: `string` EnableDefaultContextMenu enables the browser's default context-menu in production. -By default, the browser's default context-menu is only available in development and in a `-debug` or `-devtools` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : @@ -593,6 +593,12 @@ Setting this to `true` will disable GPU hardware acceleration for the webview. Name: WebviewGpuIsDisabled
Type: `bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + ### Mac This defines [Mac specific options](#mac). @@ -688,6 +694,42 @@ Setting this to `true` will make the window background translucent. Often combin Name: WindowIsTranslucent
Type: `bool` +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### About This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. diff --git a/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx index 1af17f00c29..c4f2e844564 100644 --- a/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx @@ -13,6 +13,28 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), ## [Unreleased] +### Added + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### Changed + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### Fixed + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + ## v2.6.0 - 2023-09-06 ### Breaking Changes @@ -27,6 +49,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) - Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) - Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) - Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) - Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) @@ -44,6 +67,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) - Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) - Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) ### Changed @@ -680,9 +704,6 @@ Experimental: &options.Experimental{ - [v2, nsis] Seems like / as path separator works only for some directives in a cross platform way by [@stffabi](https://github.com/stffabi) in #1227 - import models on binding definition by [@adalessa](https://github.com/adalessa) in #123 - - 1 - - Use local search on website by [@leaanthony](https://github.com/leaanthony) in #1234 - Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 - Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 diff --git a/website/i18n/pt/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx b/website/i18n/pt/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx index d01b542b153..e8a637cfda1 100644 --- a/website/i18n/pt/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx @@ -1,6 +1,6 @@ --- slug: wails-v2-beta-for-windows -title: Wails v2 Beta for Windows +title: Wails v2 Beta para Windows authors: - leaanthony tags: @@ -19,19 +19,19 @@ tags:
``` -When I first announced Wails on Reddit, just over 2 years ago from a train in Sydney, I did not expect it to get much attention. A few days later, a prolific tech vlogger released a tutorial video, gave it a positive review and from that point on, interest in the project has skyrocketed. +Quando eu anunciei o pela primeira vez o Wails no Reddit, justamente a 2 anos atrás dentro de um trem em Sidney, Eu não esperava que isso teria muita atenção. Alguns dias mais tarde, um vlogger tecnológico prolífico lançou um vídeo de tutorial, deu uma revisão positiva desde então o interesse pelo projeto disparou. -It was clear that people were excited about adding web frontends to their Go projects, and almost immediately pushed the project beyond the proof of concept that I had created. At the time, Wails used the [webview](https://github.com/webview/webview) project to handle the frontend, and the only option for Windows was the IE11 renderer. Many bug reports were rooted in this limitation: poor JavaScript/CSS support and no dev tools to debug it. This was a frustrating development experience but there wasn't much that could have been done to rectify it. +Estava claro que as pessoas estavam animadas para adicionar frontend ‘web’ nas suas aplicações em Go, e quase imediatamente o projeto foi para além da prova de conceito que criei. No momento, Wails usava o projeto [webview](https://github.com/webview/webview) para lidar com o frontend, e a única opção para Windows era o renderizador IE11. Muitos relatórios de erros tiveram root nessa limitação: suporte pobre de JavaScript/CSS e nenhuma ferramenta de desenvolvimento para depurá-lo. Esta foi uma experiência de desenvolvimento frustrante, mas não havia muito que pudesse ter sido feito para corrigi-lo. -For a long time, I'd firmly believed that Microsoft would eventually have to sort out their browser situation. The world was moving on, frontend development was booming and IE wasn't cutting it. When Microsoft announced the move to using Chromium as the basis for their new browser direction, I knew it was only a matter of time until Wails could use it, and move the Windows developer experience to the next level. +Por muito tempo, eu acreditava firmemente que a Microsoft acabaria por ter que resolver a situação do navegador. O mundo estava avançando, o desenvolvimento frondend estava crescendo e o IE não estava acompanhando. Quando a Microsoft anunciou o movimento para usar o Chromium como base para a nova direção do navegador. Eu sabia que era apenas uma questão de tempo até que as Wails pudessem usá-lo, e mova a experiência de desenvolvedor do Windows para o próximo nível. -Today, I am pleased to announce: **Wails v2 Beta for Windows**! There's a huge amount to unpack in this release, so grab a drink, take a seat and we'll begin... +Hoje, estou contente em anunciar: **Wails v2 Beta para Windows**! Há uma enorme quantidade para descompactar esta atualização, então tome uma bebida, sente-se e nós começaremos... -### No CGO Dependency! +### Sem dependência CGO! -No, I'm not joking: _No_ _CGO_ _dependency_ 🤯! The thing about Windows is that, unlike MacOS and Linux, it doesn't come with a default compiler. In addition, CGO requires a mingw compiler and there's a ton of different installation options. Removing the CGO requirement has massively simplified setup, as well as making debugging an awful lot easier. Whilst I have put a fair bit of effort in getting this working, the majority of the credit should go to [John Chadwick](https://github.com/jchv) for not only starting a couple of projects to make this possible, but also being open to someone taking those projects and building on them. Credit also to [Tad Vizbaras](https://github.com/tadvi) whose [winc](https://github.com/tadvi/winc) project started me down this path. +Não, não estou brincando: _Não_ _CGO_ _dependência_🤯! O problema do Windows é que, diferentemente do MacOS e do Linux, ele não vem com um compilador padrão. Além disso, CGO requer um compilador mingw e há uma tonelada de diferentes opções de instalação. A remoção do requisito de CGO tem uma configuração simplificada massivamente, bem como tornar a depuração muito mais fácil. Embora eu tenha me esforçado bastante para fazer isso funcionar, a maioria dos o crédito deveria ir para [John Chadwick](https://github.com/jchv) por não apenas iniciar alguns projetos para fazer isso possível, mas também estar aberto a alguém que assuma esses projetos e os desenvolva. Crédito também a [Tad Vizbaras](https://github.com/tadvi) cujo projeto [winc](https://github.com/tadvi/winc) me iniciou por este caminho. -### WebView2 Chromium Renderer +### Renderizador Chromium WebView2 ```mdx-code-block
@@ -44,15 +44,15 @@ No, I'm not joking: _No_ _CGO_ _dependency_ 🤯! The thing about Windows is tha
``` -Finally, Windows developers get a first class rendering engine for their applications! Gone are the days of contorting your frontend code to work on Windows. On top of that, you get a first-class developer tools experience! +Finalmente, os desenvolvedores do Windows recebem um mecanismo de renderização de primeira classe para as suas aplicações! Já se foram dias de contorno seu código frontend para funcionar no Windows. Além disso, você tem uma experiência de ferramentas de desenvolvedor de primeira classe! -The WebView2 component does, however, have a requirement to have the `WebView2Loader.dll` sitting alongside the binary. This makes distribution just that little bit more painful than we gophers are used to. All solutions and libraries (that I know of) that use WebView2 have this dependency. +O componente WebView2 tem, no entanto, um requisito para que o `WebView2Loader.dll` fique lado a lado com o binário. Isso faz com que a distribuição seja um pouco mais dolorosa do que nós gophers gostamos de ver. Todas as soluções e bibliotecas (que eu conheço) que usam WebView2 têm essa dependência. -However, I'm really excited to announce that Wails applications _have no such requirement_! Thanks to the wizardry of [John Chadwick](https://github.com/jchv), we are able to bundle this dll inside the binary and get Windows to load it as if it were present on disk. +No entanto, estou muito animado em anunciar que os aplicativos Wails _não têm tal exigência_! Obrigado à magia de [John Chadwick](https://github.com/jchv), nós somos capazes de agregar essa barra dentro do binário e fazer com que o Windows carregue como se estivesse presente no disco. -Gophers rejoice! The single binary dream lives on! +Gophers se alegram! O sonho binário único continua vivo! -### New Features +### Novos Recursos ```mdx-code-block
@@ -65,11 +65,11 @@ Gophers rejoice! The single binary dream lives on!
``` -There were a lot of requests for native menu support. Wails has finally got you covered. Application menus are now available and include support for most native menu features. This includes standard menu items, checkboxes, radio groups, submenus and separators. +Houve muitas solicitações para o suporte ao menu nativo. Wails finalmente ajudou você. Os menus de aplicativo agora estão disponíveis e incluem suporte para a maioria dos recursos do menu nativo. Isto inclui itens de menu padrão, caixas de seleção, grupos de rádio, submenus e separadores. -There were a huge number of requests in v1 for the ability to have greater control of the window itself. I'm happy to announce that there's new runtime APIs specifically for this. It's feature-rich and supports multi-monitor configurations. There is also an improved dialogs API: Now, you can have modern, native dialogs with rich configuration to cater for all your dialog needs. +Houve um grande número de pedidos na v1 no sentido de se conseguir um maior controlo da própria janela. Estou feliz em anunciar que há novas APIs de tempo de execução especificamente para isso. Ele é rico em recursos e suporta configurações multi-monitores. Há também uma API de diálogos aprimorada: agora você pode ter diálogos com configuração avançada para atender a todas as suas necessidades de diálogo. -There is now the option to generate IDE configuration along with your project. This means that if you open your project in a supported IDE, it will already be configured for building and debugging your application. Currently VSCode is supported but we hope to support other IDEs such as Goland soon. +Agora há a opção de gerar a configuração do IDE junto com o seu projeto. Isto significa que se você abrir o seu projeto em um IDE suportado, ele já será configurado para construir e depurar sua aplicação. Atualmente o VSCode é suportado mas esperamos dar suporte a outros IDEs como o Goland em breve. ```mdx-code-block
@@ -82,19 +82,19 @@ There is now the option to generate IDE configuration along with your project. T
``` -### No requirement to bundle assets +### Nenhum requisito para empacotar assets -A huge pain-point of v1 was the need to condense your entire application down to single JS & CSS files. I'm happy to announce that for v2, there is no requirement to bundle assets, in any way, shape or form. Want to load a local image? Use an `` tag with a local src path. Want to use a cool font? Copy it in and add the path to it in your CSS. +Um grande problema da v1 foi a necessidade de condensar todo o seu aplicativo em um único JS & Arquivos CSS. Estou feliz em anunciar que para v2, não há nenhum requisito de agrupar assets, de qualquer forma. Quer carregar uma imagem local? Use uma tag `` com um caminho de src local. Quer usar uma fonte legal? Copie ele e adicione o caminho para ele em seu CSS. -> Wow, that sounds like a webserver... +> Uau, isso soa como um servidor web... -Yes, it works just like a webserver, except it isn't. +Sim, funciona como um servidor web, mas não é. -> So how do I include my assets? +> Então, como incluo meus assets? -You just pass a single `embed.FS` that contains all your assets into your application configuration. They don't even need to be in the top directory - Wails will just work it out for you. +Você apenas passa um único `embed.FS` que contém todos os seus assets na configuração da sua aplicação. Eles nem precisam estar no diretório superior - o Wails resolverá isso para você. -### New Development Experience +### Nova Experiência de Desenvolvimento ```mdx-code-block
@@ -107,20 +107,20 @@ You just pass a single `embed.FS` that contains all your assets into your applic
``` -Now that assets don't need to be bundled, it's enabled a whole new development experience. The new `wails dev` command will build and run your application, but instead of using the assets in the `embed.FS`, it loads them directly from disk. +Agora que os ativos não precisam ser agrupados, foi possível uma experiência de desenvolvimento totalmente nova. O novo comando `wail dev` irá construir e executar seu aplicativo, mas em vez de usar os ativos do `incorporados. S`, carrega-os diretamente do disco. -It also provides the additional features: +Ele também fornece os recursos adicionais: -- Hot reload - Any changes to frontend assets will trigger and auto reload of the application frontend -- Auto rebuild - Any changes to your Go code will rebuild and relaunch your application +- Carregamento automatico - Quaisquer mudanças nos recursos do frontend irão ativar e recarregar automaticamente o frontend do aplicativo +- Reconstrução automática - Qualquer alteração ao seu código Go irá reconstruir e reiniciar seu aplicativo -In addition to this, a webserver will start on port 34115. This will serve your application to any browser that connects to it. All connected web browsers will respond to system events like hot reload on asset change. +Além disso, um servidor web iniciará na porta 34115. Isso servirá seu aplicativo para qualquer navegador que conecta a ele. Todos os navegadores web conectados responderão a eventos do sistema como recarregar rapidamente na alteração de ativos. -In Go, we are used to dealing with structs in our applications. It's often useful to send structs to our frontend and use them as state in our application. In v1, this was a very manual process and a bit of a burden on the developer. I'm happy to announce that in v2, any application run in dev mode will automatically generate TypeScript models for all structs that are input or output parameters to bound methods. This enables seamless interchange of data models between the two worlds. +Em Go, estamos acostumados a lidar com estruturas em nossas aplicações. Muitas vezes é útil enviar estruturas para nosso frontend e use-os como estado em nosso aplicativo. Na v1, este foi um processo muito manual e um pouco de sobrecarga para o desenvolvedor. Tenho prazer em anunciar isso em v2, qualquer aplicativo executado no modo de desenvolvimento irá gerar automaticamente modelos TypeScript para todas as construções que são entradas ou parâmetros de saída para métodos vinculados. Isso permite uma troca perfeita de dados modelos entre os dois mundos. -In addition to this, another JS module is dynamically generated wrapping all your bound methods. This provides JSDoc for your methods, providing code completion and hinting in your IDE. It's really cool when you get data models auto-imported when hitting tab in an auto-generated module wrapping your Go code! +Além disso, outro módulo JS é gerado dinamicamente todos os seus métodos de vinculação. Isso fornece JSDoc para seus métodos, fornecendo a conclusão de código e dicas em seu IDE. É muito legal quando você obtém modelos de dados auto-importado quando atinge a aba em um módulo gerado automaticamente embrulhando o seu código Go! -### Remote Templates +### Modelos Remotos ```mdx-code-block
@@ -133,29 +133,29 @@ In addition to this, another JS module is dynamically generated wrapping all you
``` -Getting an application up and running quickly was always a key goal for the Wails project. When we launched, we tried to cover a lot of the modern frameworks at the time: react, vue and angular. The world of frontend development is very opinionated, fast moving and hard to keep on top of! As a result, we found our base templates getting out of date pretty quickly and this caused a maintenance headache. It also meant that we didn't have cool modern templates for the latest and greatest tech stacks. +Colocar um aplicativo em funcionamento rapidamente sempre foi um objetivo importante do projeto Wails. Quando lançamos, tentamos cobrir muitos dos frameworks modernos no momento: react, vue e angular. O mundo do desenvolvimento frontend é muito teimoso, rápido e difícil de manter no controle! Como resultado, descobrimos que nossos modelos básicos estavam bastante desatualizados rapidamente e isso causou uma dor de cabeça de manutenção. Também significava que não tínhamos modelos modernos legais para os mais recentes e os maiores cumes tecnológicos. -With v2, I wanted to empower the community by giving you the ability to create and host templates yourselves, rather than rely on the Wails project. So now you can create projects using community supported templates! I hope this will inspire developers to create a vibrant ecosystem of project templates. I'm really quite excited about what our developer community can create! +Com a v2, eu queria capacitar a comunidade, dando a vocês a capacidade de criar e hospedar modelos por conta própria, em vez de do que confiar no projeto Wails. Então agora você pode criar projetos usando templates suportados pela comunidade! Espero que isto vá inspirar os desenvolvedores a criar um ecossistema vibrante de modelos de projeto. Estou realmente animado com o que a comunidade de desenvolvedores pode criar! -### In Conclusion +### Em conclusão -Wails v2 represents a new foundation for the project. The aim of this release is to get feedback on the new approach, and to iron out any bugs before a full release. Your input would be most welcome. Please direct any feedback to the [v2 Beta](https://github.com/wailsapp/wails/discussions/828) discussion board. +As trilhas v2 representam uma nova fundação para o projeto. O objetivo desta versão é obter feedback sobre a nova abordagem e aperfeiçoar quaisquer erros antes de uma versão completa. Sua opinião seria muito bem-vinda. Por favor, direcione qualquer feedback para o fórum de discussão [Beta](https://github.com/wailsapp/wails/discussions/828). -There were many twists and turns, pivots and u-turns to get to this point. This was due partly to early technical decisions that needed changing, and partly because some core problems we had spent time building workarounds for were fixed upstream: Go’s embed feature is a good example. Fortunately, everything came together at the right time, and today we have the very best solution that we can have. I believe the wait has been worth it - this would not have been possible even 2 months ago. +Houve muitas reviravoltas, pivots e reviravoltas para chegar a este ponto. Isto deveu-se em parte a decisões técnicas iniciais isso precisava ser mudado, e em parte porque alguns problemas principais para os quais passamos tempo criando soluções alternativas foram corrigidos no upstream: O recurso de incorporação do Go é um bom exemplo. Felizmente, tudo se juntou no momento certo, e hoje nós temos a melhor solução que podemos ter. Eu acredito que a espera tem valido a pena - isto não teria sido possível até 2 meses atrás. -I also need to give a huge thank you :pray: to the following people because without them, this release just wouldn't exist: +Eu também preciso dar um enorme agradecimento :pray: às seguintes pessoas, porque sem elas essa liberação não existiria: -- [Misite Bao](https://github.com/misitebao) - An absolute workhorse on the Chinese translations and an incredible bug finder. -- [John Chadwick](https://github.com/jchv) - His amazing work on [go-webview2](https://github.com/jchv/go-webview2) and [go-winloader](https://github.com/jchv/go-winloader) have made the Windows version we have today possible. -- [Tad Vizbaras](https://github.com/tadvi) - Experimenting with his [winc](https://github.com/tadvi/winc) project was the first step down the path to a pure Go Wails. -- [Mat Ryer](https://github.com/matryer) - His support, encouragement and feedback has really helped drive the project forward. +- [Misite Bao](https://github.com/misitebao) - Um cavalo de trabalho absoluto sobre as traduções chinesas e uma incrível busca de bugs. +- [John Chadwick](https://github.com/jchv) - Seu excelente trabalho no [go-webview2](https://github.com/jchv/go-webview2) e [go-winloader](https://github.com/jchv/go-winloader) tornaram a versão do Windows que temos hoje possível. +- [Tad Vizbaras](https://github.com/tadvi) - Experimentar seu projeto [winc](https://github.com/tadvi/winc) foi o primeiro passo para um Go Wails puro. +- [Mat Ryer](https://github.com/matryer) - Seu apoio, encorajamento e feedback realmente ajudaram a impulsionar o projeto. -And finally, I'd like to give a special thank you to all the [project sponsors](/credits#sponsors), including [JetBrains](https://www.jetbrains.com?from=Wails), whose support drive the project in many ways behind the scenes. +E finalmente, eu gostaria de agradecer especialmente a todos os patrocinadores [do projeto](/credits#sponsors), incluindo [JetBrains](https://www.jetbrains.com?from=Wails), cujo suporte dirige o projeto de muitas formas nos bastidores. -I look forward to seeing what people build with Wails in this next exciting phase of the project! +Estou ansioso para ver o que as pessoas construirão com Wails nesta próxima fase emocionante do projeto! Lea. -PS: MacOS and Linux users need not feel left out - porting to this new foundation is actively under way and most of the hard work has already been done. Hang in there! +PS: MacOS e usuários do Linux não precisam de se sentir esquecidos - o facto de se portar para esta nova fundação está activamente em curso e a maior parte do trabalho árduo já foi feito. Aguenta firme! -PPS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! +PPS: Se você ou sua empresa consideram o Wails útil, considere [patrocinar o projeto](https://github.com/sponsors/leaanthony). Obrigado! diff --git a/website/i18n/pt/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx b/website/i18n/pt/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx index 86d44616f40..74af6aa277d 100644 --- a/website/i18n/pt/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx @@ -1,6 +1,6 @@ --- slug: wails-v2-beta-for-mac -title: Wails v2 Beta for MacOS +title: Wails v2 Beta para MacOS authors: - leaanthony tags: @@ -19,13 +19,13 @@ tags:
``` -Today marks the first beta release of Wails v2 for Mac! It's taken quite a while to get to this point and I'm hoping that today's release will give you something that's reasonably useful. There have been a number of twists and turns to get to this point and I'm hoping, with your help, to iron out the crinkles and get the Mac port polished for the final v2 release. +Hoje marca a primeira versão beta do Wails v2 para Mac! Demorou um pouco para chegar a este ponto e espero que o lançamento de hoje lhe dará algo razoavelmente útil. Houve uma série de reviravoltas e turnos para chegar a este ponto e estou esperando, com a sua ajuda, para passar as rugas e polir a porta Mac para a versão final do v2. -You mean this isn't ready for production? For your use case, it may well be ready, but there are still a number of known issues so keep your eye on [this project board](https://github.com/wailsapp/wails/projects/7) and if you would like to contribute, you'd be very welcome! +Quer dizer que isso não está pronto para a produção? Para o seu caso de uso, ele pode estar pronto, mas ainda há vários problemas conhecidos, então fique de olho [neste quadro de projeto](https://github.com/wailsapp/wails/projects/7) e se desejar gostaria de contribuir, será muito bem-vindo! -So what's new for Wails v2 for Mac vs v1? Hint: It's pretty similar to the Windows Beta :wink: +Então, o que há de novo para Wails v2 para Mac vs v1? Dica: É bem parecido com o Windows Beta :wink: -### New Features +### Novos Recursos ```mdx-code-block
@@ -38,47 +38,47 @@ So what's new for Wails v2 for Mac vs v1? Hint: It's pretty similar to the Windo
``` -There were a lot of requests for native menu support. Wails has finally got you covered. Application menus are now available and include support for most native menu features. This includes standard menu items, checkboxes, radio groups, submenus and separators. +Houve muitas solicitações para o suporte ao menu nativo. Wails finalmente ajudou você. Os menus de aplicativo agora estão disponíveis e incluem suporte para a maioria dos recursos do menu nativo. Isto inclui itens de menu padrão, caixas de seleção, grupos de rádio, submenus e separadores. -There were a huge number of requests in v1 for the ability to have greater control of the window itself. I'm happy to announce that there's new runtime APIs specifically for this. It's feature-rich and supports multi-monitor configurations. There is also an improved dialogs API: Now, you can have modern, native dialogs with rich configuration to cater for all your dialog needs. +Houve um grande número de pedidos na v1 no sentido de se conseguir um maior controlo da própria janela. Estou feliz em anunciar que há novas APIs de tempo de execução especificamente para isso. Ele é rico em recursos e suporta configurações multi-monitores. Há também uma API de diálogos aprimorada: agora você pode ter diálogos com configuração avançada para atender a todas as suas necessidades de diálogo. -### Mac Specific Options +### Opções especificas para Mac -In addition to the normal application options, Wails v2 for Mac also brings some Mac extras: +Além das opções normais do aplicativo, o Wails v2 para Mac também traz alguns extras: -- Make your window all funky and translucent, like all the pretty swift apps! -- Highly customisable titlebar -- We support the NSAppearance options for the application -- Simple config to auto-create an "About" menu +- Faça sua janela totalmente engraçada e translucente, como todos os aplicativos muito rápidos! +- Barra de títulos altamente customizável +- Nós suportamos as opções NSAppearance para o aplicativo +- Configuração simples para criar automaticamente um menu "Sobre" -### No requirement to bundle assets +### Nenhum requisito para empacotar assets -A huge pain-point of v1 was the need to condense your entire application down to single JS & CSS files. I'm happy to announce that for v2, there is no requirement to bundle assets, in any way, shape or form. Want to load a local image? Use an `` tag with a local src path. Want to use a cool font? Copy it in and add the path to it in your CSS. +Um grande problema da v1 foi a necessidade de condensar todo o seu aplicativo em um único JS & Arquivos CSS. Estou feliz em anunciar que para v2, não há nenhum requisito de agrupar assets, de qualquer forma. Quer carregar uma imagem local? Use uma tag `` com um caminho de src local. Quer usar uma fonte legal? Copie ele e adicione o caminho para ele em seu CSS. -> Wow, that sounds like a webserver... +> Uau, isso soa como um servidor web... -Yes, it works just like a webserver, except it isn't. +Sim, funciona como um servidor web, mas não é. -> So how do I include my assets? +> Então, como incluo meus assets? -You just pass a single `embed.FS` that contains all your assets into your application configuration. They don't even need to be in the top directory - Wails will just work it out for you. +Você apenas passa um único `embed.FS` que contém todos os seus assets na configuração da sua aplicação. Eles nem precisam estar no diretório superior - o Wails resolverá isso para você. -### New Development Experience +### Nova Experiência de Desenvolvimento -Now that assets don't need to be bundled, it's enabled a whole new development experience. The new `wails dev` command will build and run your application, but instead of using the assets in the `embed.FS`, it loads them directly from disk. +Agora que os ativos não precisam ser agrupados, foi possível uma experiência de desenvolvimento totalmente nova. O novo comando `wail dev` irá construir e executar seu aplicativo, mas em vez de usar os ativos do `incorporados. S`, carrega-os diretamente do disco. -It also provides the additional features: +Ele também fornece os recursos adicionais: -- Hot reload - Any changes to frontend assets will trigger and auto reload of the application frontend -- Auto rebuild - Any changes to your Go code will rebuild and relaunch your application +- Carregamento automatico - Quaisquer mudanças nos recursos do frontend irão ativar e recarregar automaticamente o frontend do aplicativo +- Reconstrução automática - Qualquer alteração ao seu código Go irá reconstruir e reiniciar seu aplicativo -In addition to this, a webserver will start on port 34115. This will serve your application to any browser that connects to it. All connected web browsers will respond to system events like hot reload on asset change. +Além disso, um servidor web iniciará na porta 34115. Isso servirá seu aplicativo para qualquer navegador que conecta a ele. Todos os navegadores web conectados responderão a eventos do sistema como recarregar rapidamente na alteração de ativos. -In Go, we are used to dealing with structs in our applications. It's often useful to send structs to our frontend and use them as state in our application. In v1, this was a very manual process and a bit of a burden on the developer. I'm happy to announce that in v2, any application run in dev mode will automatically generate TypeScript models for all structs that are input or output parameters to bound methods. This enables seamless interchange of data models between the two worlds. +Em Go, estamos acostumados a lidar com estruturas em nossas aplicações. Muitas vezes é útil enviar estruturas para nosso frontend e use-os como estado em nosso aplicativo. Na v1, este foi um processo muito manual e um pouco de sobrecarga para o desenvolvedor. Tenho prazer em anunciar isso em v2, qualquer aplicativo executado no modo de desenvolvimento irá gerar automaticamente modelos TypeScript para todas as construções que são entradas ou parâmetros de saída para métodos vinculados. Isso permite uma troca perfeita de dados modelos entre os dois mundos. -In addition to this, another JS module is dynamically generated wrapping all your bound methods. This provides JSDoc for your methods, providing code completion and hinting in your IDE. It's really cool when you get data models auto-imported when hitting tab in an auto-generated module wrapping your Go code! +Além disso, outro módulo JS é gerado dinamicamente todos os seus métodos de vinculação. Isso fornece JSDoc para seus métodos, fornecendo a conclusão de código e dicas em seu IDE. É muito legal quando você obtém modelos de dados auto-importado quando atinge a aba em um módulo gerado automaticamente embrulhando o seu código Go! -### Remote Templates +### Modelos Remotos ```mdx-code-block
@@ -91,13 +91,13 @@ In addition to this, another JS module is dynamically generated wrapping all you
``` -Getting an application up and running quickly was always a key goal for the Wails project. When we launched, we tried to cover a lot of the modern frameworks at the time: react, vue and angular. The world of frontend development is very opinionated, fast moving and hard to keep on top of! As a result, we found our base templates getting out of date pretty quickly and this caused a maintenance headache. It also meant that we didn't have cool modern templates for the latest and greatest tech stacks. +Colocar um aplicativo em funcionamento rapidamente sempre foi um objetivo importante do projeto Wails. Quando lançamos, tentamos cobrir muitos dos frameworks modernos no momento: react, vue e angular. O mundo do desenvolvimento frontend é muito teimoso, rápido e difícil de manter no controle! Como resultado, descobrimos que nossos modelos básicos estavam bastante desatualizados rapidamente e isso causou uma dor de cabeça de manutenção. Também significava que não tínhamos modelos modernos legais para os mais recentes e os maiores cumes tecnológicos. -With v2, I wanted to empower the community by giving you the ability to create and host templates yourselves, rather than rely on the Wails project. So now you can create projects using community supported templates! I hope this will inspire developers to create a vibrant ecosystem of project templates. I'm really quite excited about what our developer community can create! +Com a v2, eu queria capacitar a comunidade, dando a vocês a capacidade de criar e hospedar modelos por conta própria, em vez de do que confiar no projeto Wails. Então agora você pode criar projetos usando templates suportados pela comunidade! Espero que isto vá inspirar os desenvolvedores a criar um ecossistema vibrante de modelos de projeto. Estou realmente animado com o que a comunidade de desenvolvedores pode criar! -### Native M1 Support +### Suporte Nativo M1 -Thanks to the amazing support of [Mat Ryer](https://github.com/matryer/), the Wails project now supports M1 native builds: +Graças ao incrível apoio do [Ryer](https://github.com/matryer/)Mat, o projeto Wails agora suporta as construções nativas do M1: ```mdx-code-block
@@ -110,7 +110,7 @@ Thanks to the amazing support of [Mat Ryer](https://github.com/matryer/), the Wa
``` -You can also specify `darwin/amd64` as a target too: +Você também pode especificar `darwin/amd64` como um alvo também: ```mdx-code-block
@@ -123,7 +123,7 @@ You can also specify `darwin/amd64` as a target too:
``` -Oh, I almost forgot.... you can also do `darwin/universal`.... :wink: +Ah, eu quase esqueci... você também pode fazer `darwin/universal`.... :wink: ```mdx-code-block
@@ -136,9 +136,9 @@ Oh, I almost forgot.... you can also do `darwin/universal`.... :wink:
``` -### Cross Compilation to Windows +### Compilação para Windows -Because Wails v2 for Windows is pure Go, you can target Windows builds without docker. +Como o Wails v2 para Windows é um Go, você pode direcionar versões do Windows sem o docker. ```mdx-code-block
@@ -151,20 +151,20 @@ Because Wails v2 for Windows is pure Go, you can target Windows builds without d
``` -### WKWebView Renderer +### Renderizador WKWebView -V1 relied on a (now deprecated) WebView component. V2 uses the most recent WKWebKit component so expect the latest and greatest from Apple. +V1 dependeu de um componente WebView (agora obsoleto). V2 usa o componente WKWebKit mais recente então espere o mais recente e o maior da Apple. -### In Conclusion +### Em conclusão -As I'd said in the Windows release notes, Wails v2 represents a new foundation for the project. The aim of this release is to get feedback on the new approach, and to iron out any bugs before a full release. Your input would be most welcome! Please direct any feedback to the [v2 Beta](https://github.com/wailsapp/wails/discussions/828) discussion board. +Como eu disse nas notas de lançamento do Windows, Wails v2 representa uma nova fundação para o projeto. O objetivo desta versão é obter feedback sobre a nova abordagem e aperfeiçoar quaisquer erros antes de uma versão completa. Sua opinião seria muito bem-vinda! Por favor, direcione qualquer feedback para o fórum de discussão [Beta](https://github.com/wailsapp/wails/discussions/828). -And finally, I'd like to give a special thank you to all the [project sponsors](/credits#sponsors), including [JetBrains](https://www.jetbrains.com?from=Wails), whose support drive the project in many ways behind the scenes. +E finalmente, eu gostaria de agradecer especialmente a todos os patrocinadores [do projeto](/credits#sponsors), incluindo [JetBrains](https://www.jetbrains.com?from=Wails), cujo suporte dirige o projeto de muitas formas nos bastidores. -I look forward to seeing what people build with Wails in this next exciting phase of the project! +Estou ansioso para ver o que as pessoas construirão com Wails nesta próxima fase emocionante do projeto! Lea. -PS: Linux users, you're next! +PS: usuários do Linux, você será o próximo! -PPS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! +PPS: Se você ou sua empresa consideram o Wails útil, considere [patrocinar o projeto](https://github.com/sponsors/leaanthony). Obrigado! diff --git a/website/i18n/pt/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx b/website/i18n/pt/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx index b405953cfbc..e0a2c87c5dd 100644 --- a/website/i18n/pt/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx @@ -1,6 +1,6 @@ --- slug: wails-v2-beta-for-linux -title: Wails v2 Beta for Linux +title: Wails v2 Beta para Linux authors: - leaanthony tags: @@ -19,9 +19,9 @@ tags:
``` -I'm pleased to finally announce that Wails v2 is now in beta for Linux! It is somewhat ironic that the very first experiments with v2 was on Linux and yet it has ended up as the last release. That being said, the v2 we have today is very different from those first experiments. So without further ado, let's go over the new features: +Tenho o prazer de finalmente anunciar que as Wails v2 agora estão em beta para Linux! É um pouco irônico que os primeiros experimentos com v2 tenha sido no Linux e, no entanto, ele acabou como o último lançamento. Dito isso, o v2 que temos hoje é muito diferente desses primeiros experimentos. Então, sem mais demora, vamos analisar os novos recursos: -### New Features +### Novos Recursos ```mdx-code-block
@@ -34,38 +34,38 @@ I'm pleased to finally announce that Wails v2 is now in beta for Linux! It is so
``` -There were a lot of requests for native menu support. Wails has finally got you covered. Application menus are now available and include support for most native menu features. This includes standard menu items, checkboxes, radio groups, submenus and separators. +Houve muitas solicitações para o suporte ao menu nativo. Wails finalmente ajudou você. Os menus de aplicativo agora estão disponíveis e incluem suporte para a maioria dos recursos do menu nativo. Isto inclui itens de menu padrão, caixas de seleção, grupos de rádio, submenus e separadores. -There were a huge number of requests in v1 for the ability to have greater control of the window itself. I'm happy to announce that there's new runtime APIs specifically for this. It's feature-rich and supports multi-monitor configurations. There is also an improved dialogs API: Now, you can have modern, native dialogs with rich configuration to cater for all your dialog needs. +Houve um grande número de pedidos na v1 no sentido de se conseguir um maior controlo da própria janela. Estou feliz em anunciar que há novas APIs de tempo de execução especificamente para isso. Ele é rico em recursos e suporta configurações multi-monitores. Há também uma API de diálogos aprimorada: agora você pode ter diálogos com configuração avançada para atender a todas as suas necessidades de diálogo. -### No requirement to bundle assets +### Nenhum requisito para empacotar assets -A huge pain-point of v1 was the need to condense your entire application down to single JS & CSS files. I'm happy to announce that for v2, there is no requirement to bundle assets, in any way, shape or form. Want to load a local image? Use an `` tag with a local src path. Want to use a cool font? Copy it in and add the path to it in your CSS. +Um grande problema da v1 foi a necessidade de condensar todo o seu aplicativo em um único JS & Arquivos CSS. Estou feliz em anunciar que para v2, não há nenhum requisito de agrupar assets, de qualquer forma. Quer carregar uma imagem local? Use uma tag `` com um caminho de src local. Quer usar uma fonte legal? Copie ele e adicione o caminho para ele em seu CSS. -> Wow, that sounds like a webserver... +> Uau, isso soa como um servidor web... -Yes, it works just like a webserver, except it isn't. +Sim, funciona como um servidor web, mas não é. -> So how do I include my assets? +> Então, como incluo meus assets? -You just pass a single `embed.FS` that contains all your assets into your application configuration. They don't even need to be in the top directory - Wails will just work it out for you. +Você apenas passa um único `embed.FS` que contém todos os seus assets na configuração da sua aplicação. Eles nem precisam estar no diretório superior - o Wails resolverá isso para você. -### New Development Experience +### Nova Experiência de Desenvolvimento -Now that assets don't need to be bundled, it's enabled a whole new development experience. The new `wails dev` command will build and run your application, but instead of using the assets in the `embed.FS`, it loads them directly from disk. +Agora que os ativos não precisam ser agrupados, foi possível uma experiência de desenvolvimento totalmente nova. O novo comando `wail dev` irá construir e executar seu aplicativo, mas em vez de usar os ativos do `incorporados. S`, carrega-os diretamente do disco. -It also provides the additional features: +Ele também fornece os recursos adicionais: -- Hot reload - Any changes to frontend assets will trigger and auto reload of the application frontend -- Auto rebuild - Any changes to your Go code will rebuild and relaunch your application +- Carregamento automatico - Quaisquer mudanças nos recursos do frontend irão ativar e recarregar automaticamente o frontend do aplicativo +- Reconstrução automática - Qualquer alteração ao seu código Go irá reconstruir e reiniciar seu aplicativo -In addition to this, a webserver will start on port 34115. This will serve your application to any browser that connects to it. All connected web browsers will respond to system events like hot reload on asset change. +Além disso, um servidor web iniciará na porta 34115. Isso servirá seu aplicativo para qualquer navegador que conecta a ele. Todos os navegadores web conectados responderão a eventos do sistema como recarregar rapidamente na alteração de ativos. -In Go, we are used to dealing with structs in our applications. It's often useful to send structs to our frontend and use them as state in our application. In v1, this was a very manual process and a bit of a burden on the developer. I'm happy to announce that in v2, any application run in dev mode will automatically generate TypeScript models for all structs that are input or output parameters to bound methods. This enables seamless interchange of data models between the two worlds. +Em Go, estamos acostumados a lidar com estruturas em nossas aplicações. Muitas vezes é útil enviar estruturas para nosso frontend e use-os como estado em nosso aplicativo. Na v1, este foi um processo muito manual e um pouco de sobrecarga para o desenvolvedor. Tenho prazer em anunciar isso em v2, qualquer aplicativo executado no modo de desenvolvimento irá gerar automaticamente modelos TypeScript para todas as construções que são entradas ou parâmetros de saída para métodos vinculados. Isso permite uma troca perfeita de dados modelos entre os dois mundos. -In addition to this, another JS module is dynamically generated wrapping all your bound methods. This provides JSDoc for your methods, providing code completion and hinting in your IDE. It's really cool when you get data models auto-imported when hitting tab in an auto-generated module wrapping your Go code! +Além disso, outro módulo JS é gerado dinamicamente todos os seus métodos de vinculação. Isso fornece JSDoc para seus métodos, fornecendo a conclusão de código e dicas em seu IDE. É muito legal quando você obtém modelos de dados auto-importado quando atinge a aba em um módulo gerado automaticamente embrulhando o seu código Go! -### Remote Templates +### Modelos Remotos ```mdx-code-block
@@ -78,13 +78,13 @@ In addition to this, another JS module is dynamically generated wrapping all you
``` -Getting an application up and running quickly was always a key goal for the Wails project. When we launched, we tried to cover a lot of the modern frameworks at the time: react, vue and angular. The world of frontend development is very opinionated, fast moving and hard to keep on top of! As a result, we found our base templates getting out of date pretty quickly and this caused a maintenance headache. It also meant that we didn't have cool modern templates for the latest and greatest tech stacks. +Colocar um aplicativo em funcionamento rapidamente sempre foi um objetivo importante do projeto Wails. Quando lançamos, tentamos cobrir muitos dos frameworks modernos no momento: react, vue e angular. O mundo do desenvolvimento frontend é muito teimoso, rápido e difícil de manter no controle! Como resultado, descobrimos que nossos modelos básicos estavam bastante desatualizados rapidamente e isso causou uma dor de cabeça de manutenção. Também significava que não tínhamos modelos modernos legais para os mais recentes e os maiores cumes tecnológicos. -With v2, I wanted to empower the community by giving you the ability to create and host templates yourselves, rather than rely on the Wails project. So now you can create projects using community supported templates! I hope this will inspire developers to create a vibrant ecosystem of project templates. I'm really quite excited about what our developer community can create! +Com a v2, eu queria capacitar a comunidade, dando a vocês a capacidade de criar e hospedar modelos por conta própria, em vez de do que confiar no projeto Wails. Então agora você pode criar projetos usando templates suportados pela comunidade! Espero que isto vá inspirar os desenvolvedores a criar um ecossistema vibrante de modelos de projeto. Estou realmente animado com o que a comunidade de desenvolvedores pode criar! -### Cross Compilation to Windows +### Compilação para Windows -Because Wails v2 for Windows is pure Go, you can target Windows builds without docker. +Como o Wails v2 para Windows é um Go, você pode direcionar versões do Windows sem o docker. ```mdx-code-block
@@ -97,18 +97,18 @@ Because Wails v2 for Windows is pure Go, you can target Windows builds without d
``` -### In Conclusion +### Em conclusão -As I'd said in the Windows release notes, Wails v2 represents a new foundation for the project. The aim of this release is to get feedback on the new approach, and to iron out any bugs before a full release. Your input would be most welcome! Please direct any feedback to the [v2 Beta](https://github.com/wailsapp/wails/discussions/828) discussion board. +Como eu disse nas notas de lançamento do Windows, Wails v2 representa uma nova fundação para o projeto. O objetivo desta versão é obter feedback sobre a nova abordagem e aperfeiçoar quaisquer erros antes de uma versão completa. Sua opinião seria muito bem-vinda! Por favor, direcione qualquer feedback para o fórum de discussão [Beta](https://github.com/wailsapp/wails/discussions/828). -Linux is **hard** to support. We expect there to be a number of quirks with the beta. Please help us to help you by filing detailed bug reports! +Linux é **Difícil** de suporte. Esperamos que haja uma série de peculiaridades com o beta. Por favor, ajude-nos a lhe ajudar por a visualizar relatórios detalhados de erros! -Finally, I'd like to give a special thank you to all the [project sponsors](/credits#sponsors) whose support drive the project in many ways behind the scenes. +Por fim, gostaria de agradecer especialmente a todos os [patrocinadores do projeto](/credits#sponsors) cujo apoio conduzir o projeto de várias maneiras nos bastidores. -I look forward to seeing what people build with Wails in this next exciting phase of the project! +Estou ansioso para ver o que as pessoas construirão com Wails nesta próxima fase emocionante do projeto! Lea. -PS: The v2 release isn't far off now! +PS: A versão v2 não está muito distante agora! -PPS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! +PPS: Se você ou sua empresa consideram o Wails útil, considere [patrocinar o projeto](https://github.com/sponsors/leaanthony). Obrigado! diff --git a/website/i18n/pt/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx b/website/i18n/pt/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx index c321f5042d7..064c1ad85d3 100644 --- a/website/i18n/pt/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx @@ -1,6 +1,6 @@ --- slug: wails-v2-released -title: Wails v2 Released +title: Wails v2 liberado authors: - leaanthony tags: @@ -18,81 +18,81 @@ tags:
``` -# It's here! +# Já disponível! -Today marks the release of [Wails](https://wails.io) v2. It's been about 18 months since the first v2 alpha and about a year from the first beta release. I'm truly grateful to everyone involved in the evolution of the project. +Hoje marca a liberação de [Wails](https://wails.io) v2. Já faz cerca de 18 meses desde o primeiro alfa v2 e cerca de um ano desde a primeira versão beta. Eu sou realmente grato a todos que participaram da evolução do projeto. -Part of the reason it took that long was due to wanting to get to some definition of completeness before officially calling it v2. The truth is, there's never a perfect time to tag a release - there's always outstanding issues or "just one more" feature to squeeze in. What tagging an imperfect major release does do, however, is to provide a bit of stability for users of the project, as well as a bit of a reset for the developers. +Uma parte da razão por que demorou tanto tempo a querer chegar a uma definição de completude antes de a designar oficialmente como v2. A verdade é que nunca há um momento perfeito para marcar uma versão - há sempre questões pendentes ou "apenas mais um" recurso para entrar. No entanto, o que marcar uma versão principal imperfeita faz. é fornecer um pouco de estabilidade para os usuários do projeto, bem como um pouco de redefinição para os desenvolvedores. -This release is more than I'd ever expected it to be. I hope it gives you as much pleasure as it has given us to develop it. +Esta versão é mais do que eu já esperava que fosse. Espero que lhe dê tanto prazer como nos deu em desenvolvê-lo. -# What _is_ Wails? +# O que _é_ Wails? -If you are unfamiliar with Wails, it is a project that enables Go programmers to provide rich frontends for their Go programs using familiar web technologies. It's a lightweight, Go alternative to Electron. Much more information can be found on the [official site](https://wails.io/docs/introduction). +Se você não estiver familiarizado com as Wails, é um projeto que permite aos programadores Go fornecerem frontend ricos para seus programas usando tecnologias web familiares. É leve, uma alternativa ao Electron. Muito mais informações podem ser encontradas no [site oficial](https://wails.io/docs/introduction). -# What's new? +# O que há de novo? -The v2 release is a huge leap forward for the project, addressing many of the pain points of v1. If you have not read any of the blog posts on the Beta releases for [macOS](/blog/wails-v2-beta-for-mac), [Windows](/blog/wails-v2-beta-for-windows) or [Linux](/blog/wails-v2-beta-for-linux), then I encourage you to do so as it covers all the major changes in more detail. In summary: +A liberação v2 é um enorme salto em frente para o projeto, abordando muitos dos pontos de dor da v1. Se você não leu nenhum dos posts no blog em versões beta para [macOS](/blog/wails-v2-beta-for-mac), [Windows](/blog/wails-v2-beta-for-windows) ou [Linux](/blog/wails-v2-beta-for-linux), Então eu encorajo-os a fazê-lo porque cobre todas as principais mudanças em mais detalhes. Resumo: -- Webview2 component for Windows that supports modern web standards and debugging capabilities. -- [Dark / Light theme](/docs/reference/options#theme) + [custom theming](/docs/reference/options#customtheme) on Windows. -- Windows now has no CGO requirements. -- Out-of-the-box support for Svelte, Vue, React, Preact, Lit & Vanilla project templates. -- [Vite](https://vitejs.dev/) integration providing a hot-reload development environment for your application. -- Native application [menus](/docs/guides/application-development#application-menu) and [dialogs](/docs/reference/runtime/dialog). -- Native window translucency effects for [Windows](/docs/reference/options#windowistranslucent) and [macOS](/docs/reference/options#windowistranslucent-1). Support for Mica & Acrylic backdrops. -- Easily generate an [NSIS installer](/docs/guides/windows-installer) for Windows deployments. -- A rich [runtime library](/docs/reference/runtime/intro) providing utility methods for window manipulation, eventing, dialogs, menus and logging. -- Support for [obfuscating](/docs/guides/obfuscated) your application using [garble](https://github.com/burrowers/garble). -- Support for compressing your application using [UPX](https://upx.github.io/). -- Automatic TypeScript generation of Go structs. More info [here](/docs/howdoesitwork#calling-bound-go-methods). -- No extra libraries or DLLs are required to be shipped with your application. For any platform. -- No requirement to bundle frontend assets. Just develop your application like any other web application. +- Componente Webview2 para Windows que oferece suporte a modernos padrões da web e capacidades de depuração. +- [Tema Dark / Light](/docs/reference/options#theme) + [tema personalizado](/docs/reference/options#customtheme) no Windows. +- Windows não tem os requisitos de CGO. +- Suporte fora da caixa para modelos de projeto Svelte, Vue, React, Preact, Lit & Vanilla. +- [Vite](https://vitejs.dev/) integração proporcionando um ambiente de desenvolvimento de atualização quente para seu aplicativo. +- Aplicação nativa [menus](/docs/guides/application-development#application-menu) e [dialogs](/docs/reference/runtime/dialog). +- Efeitos de translucência nativos da janela para [Windows](/docs/reference/options#windowistranslucent) e [macOS](/docs/reference/options#windowistranslucent-1). Support para Mica & Acrylic backdrops. +- Gera facilmente um instalador [NSIS](/docs/guides/windows-installer) para deploys com Windows. +- Uma biblioteca [em tempo de execução](/docs/reference/runtime/intro) que fornece métodos utilitários para manipulação de janelas, eventos, diálogos, menus e logs. +- Suporte para [ofuscar](/docs/guides/obfuscated) seu aplicativo usando [garble](https://github.com/burrowers/garble). +- Suporte para compactar sua aplicação usando [UPX](https://upx.github.io/). +- Geração automática de estruturas de Go para TypeScript. Mais informações [aqui](/docs/howdoesitwork#calling-bound-go-methods). +- Não são necessárias bibliotecas extras ou DLLs para enviar com sua aplicação. Para qualquer plataforma. +- Nenhum requisito para empacotar assets. Apenas desenvolva seu aplicativo como qualquer outro aplicativo da web. -# Credit & Thanks +# Crédito & Obrigado -Getting to v2 has been a huge effort. There have been ~2.2K commits by 89 contributors between the initial alpha and the release today, and many, many more that have provided translations, testing, feedback and help on the discussion forums as well as the issue tracker. I'm so unbelievably grateful to each one of you. I'd also like to give an extra special thank you to all the project sponsors who have provided guidance, advice and feedback. Everything you do is hugely appreciated. +Chegar à v2 tem sido um enorme esforço. Houve cerca de 2,2 mil commits de 89 colaboradores entre o alfa inicial e o lançamento de hoje, e muitos, muitos mais que forneceram traduções, testes, feedback e ajuda nos fóruns de discussão, bem como no rastreador de problemas. Sou tão incrivelmente grato a cada um de vocês. Eu também gostaria de dar um agradecimento especial extra para todos os patrocinadores do projeto que forneceram orientação, conselho e feedback. Tudo o que você faz é extremamente apreciado. -There are a few people I'd like to give special mention to: +Há algumas pessoas que eu gostaria de mencionar especialmente: -Firstly, a **huge** thank you to [@stffabi](https://github.com/stffabi) who has provided so many contributions which we all benefit from, as well as providing a lot of support on many issues. He has provided some key features such as the external dev server support which transformed our dev mode offering by allowing us to hook into [Vite](https://vitejs.dev/)'s superpowers. It's fair to say that Wails v2 would be a far less exciting release without his [incredible contributions](https://github.com/wailsapp/wails/commits?author=stffabi&since=2020-01-04). Thank you so much @stffabi! +Primeiro, a **enorme** obrigado a [@stffabi](https://github.com/stffabi) que forneceu tantas contribuições das quais todos nos beneficiamos, Para além de prestar muito apoio em muitas questões. Ele forneceu algumas principais características, como o suporte ao servidor de desenvolvimento externo que transformou a oferta de desenvolvimento ao nos permitir conectar aos superpoderes do [Vite](https://vitejs.dev/). É justo dizer que o Wails v2 seria um lançamento muito menos emocionante sem sua [incrível contribuição](https://github.com/wailsapp/wails/commits?author=stffabi&since=2020-01-04). Muito obrigado @stffabi! -I'd also like to give a huge shout-out to [@misitebao](https://github.com/misitebao) who has tirelessly been maintaining the website, as well as providing Chinese translations, managing Crowdin and helping new translators get up to speed. This is a hugely important task, and I'm extremely grateful for all the time and effort put into this! You rock! +Eu também gostaria de dar um grande grito para [@misitebao](https://github.com/misitebao) que tem mantido o site incansavelmente. Além de fornecer traduções chinesas, gerenciar Crowdin e ajudar novos tradutores a se atualizarem para a velocidade. Esta é uma tarefa extremamente importante, e estou extremamente grato por todo o tempo e esforço investido nisso! Você é espetacular! -Last, but not least, a huge thank you to Mat Ryer who has provided advice and support during the development of v2. Writing xBar together using an early Alpha of v2 was helpful in shaping the direction of v2, as well as give me an understanding of some design flaws in the early releases. I'm happy to announce that as of today, we will start to port xBar to Wails v2, and it will become the flagship application for the project. Cheers Mat! +Por último, mas não menos importante, um enorme agradecimento ao senhor Mat Ryer, que prestou aconselhamento e apoio durante o desenvolvimento da v2. Escrever xBar junto usando um alfa inicial da v2 foi útil para moldar a direção da v2, Além de me dar uma ideia de algumas falhas de design nas versões mais cedo. Tenho prazer em anunciar que, a partir de hoje, começaremos a abrir o porto xBar para Wails v2, e tornar-se-á o principal aplicativo para o projecto. Felicidades Mat! -# Lessons Learnt +# Lições Aprendidas -There are a number of lessons learnt in getting to v2 that will shape development moving forward. +Há uma série de lições aprendidas ao chegar à v2 que irão moldar o desenvolvimento avançado. -## Smaller, Quicker, Focused Releases +## Menor, Rápido e Focado Versões -In the course of developing v2, there were many features and bug fixes that were developed on an ad-hoc basis. This led to longer release cycles and were harder to debug. Moving forward, we are going to create releases more often that will include a reduced number of features. A release will involve updates to documentation as well as thorough testing. Hopefully, these smaller, quicker, focussed releases will lead to fewer regressions and better quality documentation. +No desenvolvimento da v2, havia muitas características e correções de erros que foram desenvolvidas numa base ad-hoc. Isso levou a ciclos de lançamento mais longos e era mais difícil de depurar. Seguindo, vamos criar lançamentos com mais frequência que incluirá um número reduzido de recursos. Uma versão envolverá atualizações da documentação, bem como testes completos. Esperemos que estes lançamentos mais pequenos, mais rápidos e focados conduzam a menos regressões e a uma melhor documentação de qualidade. -## Encourage Engagement +## Incentive o envolvimento -When starting this project, I wanted to immediately help everyone who had a problem. Issues were "personal" and I wanted them resolved as quickly as possible. This is unsustainable and ultimately works against the longevity of the project. Moving forward, I will be giving more space for people to get involved in answering questions and triaging issues. It would be good to get some tooling to help with this so if you have any suggestions, please join in the discussion [here](https://github.com/wailsapp/wails/discussions/1855). +Ao iniciar este projeto, eu queria ajudar imediatamente todos os que tiveram um problema. Os problemas eram "pessoais" e eu queria que eles fossem resolvidos o mais rapidamente possível. Isto é insustentável e acaba por contrariar a longevidade do projecto. Avançando, darei mais espaço para que as pessoas se envolvam na resposta a perguntas e na triagem de questões. Seria bom conseguir algumas ferramentas para ajudar com isso, portanto, se você tiver alguma sugestão, por favor, participe da discussão [aqui](https://github.com/wailsapp/wails/discussions/1855). -## Learning to say No +## Aprendendo a dizer Não -The more people that engage with an Open Source project, the more requests there will be for additional features that may or may not be useful to the majority of people. These features will take an initial amount of time to develop and debug, and incur an ongoing maintenance cost from that point on. I myself am the most guilty of this, often wanting to "boil the sea" rather than provide the minimum viable feature. Moving forward, we will need to say "No" a bit more to adding core features and focus our energies on a way to empower developers to provide that functionality themselves. We are looking seriously into plugins for this scenario. This will allow anyone to extend the project as they see fit, as well as providing an easy way to contribute towards the project. +Quanto mais pessoas se envolverem com um projeto de Código Aberto, quanto mais pedidos forem para recursos adicionais que possam ou não ser úteis para a maioria das pessoas. Estas características levarão um tempo inicial para desenvolver e debugar e incorrer num custo de manutenção contínuo a partir desse ponto. Eu próprio sou o mais culpado por isso, querendo muitas vezes "ferver o mar" em vez de fornecer o mínimo viável. Seguindo, precisaremos dizer "Não" um pouco mais para adicionar recursos principais e concentrar as nossas energias em uma forma de capacitar os desenvolvedores para fornecer essas funcionalidades por conta própria. Estamos analisando seriamente plugins para este cenário. Isso permitirá a qualquer pessoa alargar o projecto como considerar conveniente, bem como proporcionar uma forma fácil de contribuir para o projecto. -# Looking to the Future +# Olhando para o futuro -There are so many core features we are looking at to add to Wails in the next major development cycle already. The [roadmap](https://github.com/wailsapp/wails/discussions/1484) is full of interesting ideas, and I'm keen to start work on them. One of the big asks has been for multiple window support. It's a tricky one and to do it right, and we may need to look at providing an alternative API, as the current one was not designed with this in mind. Based on some preliminary ideas and feedback, I think you'll like where we're looking to go with it. +Há tantas características centrais que estamos a analisar para adicionar às Wails no próximo grande ciclo de desenvolvimento. O [planejamento](https://github.com/wailsapp/wails/discussions/1484) está cheio de ideias interessantes, e estou ansioso para começar a trabalhar nelas. Uma das grandes perguntas tem sido a de múltiplas janelas. É um pouco complicado e para fazer isso certo, e pode ser que precisemos analisar a prover uma API alternativa. Como o actual não foi concebido tendo isso em mente. Com base em algumas idéias preliminares e feedback, eu acho que você vai gostar de onde estamos procurando acompanhá-lo. -I'm personally very excited at the prospect of getting Wails apps running on mobile. We already have a demo project showing that it is possible to run a Wails app on Android, so I'm really keen to explore where we can go with this! +Eu pessoalmente estou muito animado com a perspectiva de fazer aplicativos Wails funcionarem em dispositivos móveis. Já temos um projeto de demonstração mostrando que é possível executar um aplicativo Wails no Android, então eu estou realmente ansioso para explorar onde podemos ir com isso! -A final point I'd like to raise is that of feature parity. It has long been a core principle that we wouldn't add anything to the project without there being full cross-platform support for it. Whilst this has proven to be (mainly) achievable so far, it has really held the project back in releasing new features. Moving forward, we will be adopting a slightly different approach: any new feature that cannot be immediately released for all platforms will be released under an experimental configuration or API. This allows early adopters on certain platforms to try the feature and provide feedback that will feed into the final design of the feature. This, of course, means that there are no guarantees of API stability until it is fully supported by all the platforms it can be supported on, but at least it will unblock development. +Um último ponto que gostaria de levantar é o da paridade de recursos. Há muito tempo é um princípio central que não adicionamos nada ao projeto sem que haja suporte completo entre plataformas para ele. Embora isto tenha provado ser (principalmente) exequível até agora, conseguiu realmente reter o projeto na libertação de novas características. Movendo para frente, estaremos adotando uma abordagem ligeiramente diferente: qualquer nova funcionalidade que não possa ser imediatamente lançada para todas as plataformas será lançada sob uma configuração experimental ou API. Isso permite que usuários antecipados de certas plataformas experimentem o recurso e forneçam feedback que serão incorporados no design final da funcionalidade. Isto, claro, significa que não há garantias de estabilidade da API até que ela seja totalmente suportada por todas as plataformas em que pode ser suportada, mas pelo menos vai desbloquear o desenvolvimento. -# Final Words +# Palavras Finais -I'm really proud of what we've been able to achieve with the V2 release. It's amazing to see what people have already been able to build using the beta releases so far. Quality applications like [Varly](https://varly.app/), [Surge](https://getsurge.io/) and [October](https://october.utf9k.net/). I encourage you to check them out. +Estou muito orgulhoso do que conseguimos alcançar com a versão V2. É incrível ver o que as pessoas já foram capazes de construir usando os lançamentos beta. Aplicativos de qualidade como [Varly](https://varly.app/), [Surge](https://getsurge.io/) e [outubro](https://october.utf9k.net/). Eu encorajo você a conferi-los. -This release was achieved through the hard work of many contributors. Whilst it is free to download and use, it has not come about through zero cost. Make no mistakes, this project has come at considerable cost. It has not only been my time and the time of each and every contributor, but also the cost of absence from friends and families of each of those people too. That's why I'm extremely grateful for every second that has been dedicated to making this project happen. The more contributors we have, the more this effort can be spread out and the more we can achieve together. I'd like to encourage you all to pick one thing that you can contribute, whether it is confirming someone's bug, suggesting a fix, making a documentation change or helping out someone who needs it. All of these small things have such a huge impact! It would be so awesome if you too were part of the story in getting to v3. +Esta libertação foi conseguida através do trabalho árduo de muitos contribuintes. Embora seja livre de descarregar e utilizar, não surgiu através de custo zero. Não cometamos erros, este projeto tem um custo considerável. Não foi só o meu tempo e o tempo de cada um dos intervenientes, mas também o de cada um. mas também o custo da ausência de amigos e famílias de cada uma dessas pessoas. É por isso que eu estou extremamente grato por cada segundo que foi dedicado a fazer este projeto acontecer. Quanto mais contribuidores tivermos, mais este esforço poderá ser repartido e mais conseguiremos alcançar em conjunto. Eu gostaria de encorajar todos vocês a escolherem uma coisa que você possa contribuir, seja confirmar o bug de alguém, Sugira uma correção, fazendo uma documentação mudar ou ajudando alguém que precise. Todas estas pequenas coisas têm um impacto tão grande! Seria fantástico se o senhor também fizesse parte da história para chegar à v3. -Enjoy! +Aproveite! ‐ Lea -PS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! +PS: Se você ou sua empresa consideram o Wails útil, considere [patrocinar o projeto](https://github.com/sponsors/leaanthony). Obrigado! diff --git a/website/i18n/pt/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx b/website/i18n/pt/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx index 9a137d09a8c..7ec5accb578 100644 --- a/website/i18n/pt/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx @@ -1,6 +1,6 @@ --- slug: the-road-to-wails-v3 -title: The Road to Wails v3 +title: A Estrada para Wails v3 authors: - leaanthony tags: @@ -18,61 +18,61 @@ tags:
``` -# Introduction +# Introdução -Wails is a project that simplifies the ability to write cross-platform desktop applications using Go. It uses native webview components for the frontend (not embedded browsers), bringing the power of the world's most popular UI system to Go, whilst remaining lightweight. +Wails é um projeto que simplifica a capacidade de escrever aplicativos de desktop multiplataforma usando Ir. Ele usa componentes nativos do webview para o frontend (não os navegadores incorporados), trazendo o poder do sistema de interface de usuário mais popular do mundo, enquanto permanece leve. -Version 2 was released on the 22nd of September 2022 and brought with it a lot of enhancements including: +A versão 2 foi lançada no dia 22 de setembro de 2022 e trouxe uma série de aprimoramentos incluindo: -- Live development, leveraging the popular Vite project -- Rich functionality for managing windows and creating menus +- Desenvolvimento ao vivo, aproveitando o projeto popular Vite +- Funcionalidade rica para gerenciar janelas e criar menus - Microsoft's WebView2 component -- Generation of Typescript models that mirror your Go structs -- Creating of NSIS Installer -- Obfuscated builds +- Geração de modelos de Typescript que espelham suas construções em Go +- Criação do NSIS Installer +- Compilações ofuscadas -Right now, Wails v2 provides powerful tooling for creating rich, cross-platform desktop applications. +No momento, o Wails v2 fornece ferramentas poderosas para a criação de ‘desktops’ ricos e multiplataforma. Formulários. Ir. -This blog post aims to look at where the project is at right now and what we can improve on moving forward. +Este post de blog visa ver onde o projeto está no momento e o que podemos melhorar ao avançar. -# Where are we now? +# Em que ponto estamos agora? -It's been incredible to see the popularity of Wails rising since the v2 release. I'm constantly amazed by the creativity of the community and the wonderful things that are being built with it. With more popularity, comes more eyes on the project. And with that, more feature requests and bug reports. +Tem sido incrível ver a popularidade das Wails aumentar desde o lançamento v2. Estou constantemente maravilhados com a criatividade da comunidade e com as coisas maravilhosas que estão sendo construídas com ela. Com mais popularidade, vem mais olhos para o projeto. E com isso, mais solicitações de recursos e relatórios de bugs. -Over time, I've been able to identify some of the most pressing issues facing the project. I've also been able to identify some of the things that are holding the project back. +Ao longo do tempo, consegui identificar alguns dos problemas mais prementes enfrentados pelo projeto. Eu também fui capaz de identificar algumas das coisas que estão segurando o projeto de volta. -## Current issues +## Problemas atuais -I've identified the following areas that I feel are holding the project back: +Eu identifiquei as seguintes áreas que sinto que estão segurando o projeto de volta: -- The API -- Bindings generation -- The Build System +- A API +- Geração de Atalhos +- O Sistema de Compilação -### The API +### A API -The API to build a Wails application currently consists of 2 parts: +A API para criar um aplicativo Wails atualmente consiste de 2 partes: -- The Application API -- The Runtime API +- A API do Aplicativo +- A API Runtime -The Application API famously has only 1 function: `Run()` which takes a heap of options which govern how the application will work. Whilst this is very simple to use, it is also very limiting. It is a "declarative" approach which hides a lot of the underlying complexity. For instance, there is no handle to the main window, so you can't interact with it directly. For that, you need to use the Runtime API. This is a problem when you start to want to do more complex things like create multiple windows. +A aplicação API famosamente tem apenas 1 função: `Run()` que dá um monte de opções que governam como a aplicação vai funcionar. Embora isso seja muito simples de usar, é também muito limitado. É uma abordagem "declarativa" que esconde muita da complexidade subjacente. Por exemplo, não há nenhum identificador para a janela principal, então você não pode interagir diretamente com ele. Para isso, você precisa usar a API Runtime. Este é um problema quando você começar a fazer coisas mais complexas como criar múltiplas janelas. -The Runtime API provides a lot of utility functions for the developer. This includes: +A API do Runtime fornece um monte de funções de utilidade para o desenvolvedor. Isso inclui: -- Window management -- Dialogs +- Gestão de janelas +- Caixa de diálogo - Menus -- Events +- Eventos - Logs -There are a number of things I am not happy with the Runtime API. The first is that it requires a "context" to be passed around. This is both frustrating and confusing for new developers who pass in a context and then get a runtime error. +Há várias coisas que eu não estou feliz com a API de Runtime. O primeiro é que ele requer um "contexto" para ser passado. Isto é frustrante e confuso para novos desenvolvedores que passam em um contexto e depois recebem um erro de tempo de execução. -The biggest issue with the Runtime API is that it was designed for applications that only use a single window. Over time, the demand for multiple windows has grown and the API is not well suited to this. +O maior problema com a API de Runtime é que ela foi projetada para aplicativos que só usam uma única janela. Ao longo do tempo, a demanda por várias janelas cresceu e a API é não adequada a isso. -### Thoughts on the v3 API +### Pensamentos na API v3 -Wouldn't it be great if we could do something like this? +Não seria ótimo se pudéssemos fazer algo assim? ```go func main() { @@ -86,7 +86,7 @@ func main() { } ``` -This programmatic approach is far more intuitive and allows the developer to interact with the application elements directly. All current runtime methods for windows would simply be methods on the window object. For the other runtime methods, we could move them to the application object like so: +Esta abordagem programática é muito mais intuitiva e permite que o desenvolvedor interaja com os elementos do aplicativo diretamente. Todos os métodos atuais de tempo de execução para o windows seriam simplesmente métodos no objeto da janela. Para os outros métodos de execução, poderíamos mover eles para o objeto do aplicativo assim: ```go app := wails.NewApplication(options.App{}) @@ -94,7 +94,7 @@ app.NewInfoDialog(options.InfoDialog{}) app.Log.Info("Hello World") ``` -This is a much more powerful API which will allow for more complex applications to be built. It also allows for the creation of multiple windows, [the most up-voted feature on GitHub](https://github.com/wailsapp/wails/issues/1480): +Esta é uma API muito mais poderosa que permitirá a criação de aplicações mais complexas. Ele também permite a criação de múltiplas janelas, [o recurso mais votado para cima no GitHub](https://github.com/wailsapp/wails/issues/1480): ```go func main() { @@ -113,72 +113,72 @@ func main() { } ``` -### Bindings generation +### Geração de Atalhos -One of the key features of Wails is generating bindings for your Go methods so they may be called from Javascript. The current method for doing this is a bit of a hack. It involves building the application with a special flag and then running the resultant binary which uses reflection to determine what has been bound. This leads to a bit of a chicken and egg situation: You can't build the application without the bindings and you can't generate the bindings without building the application. There are many ways around this but the best one would be not to use this approach at all. +Uma das principais características das Wails é gerar ligações para seus métodos Go para que possam ser chamadas a partir de Javascript. O método atual para fazer isso é um pouco hackeado. Ele envolve construir o aplicativo com uma bandeira especial e, em seguida, executar o binário resultante que usa reflexão para determinar o que foi vinculado. Isso leva a um pouco de galinha e ovo situação: você não pode construir o aplicativo sem as ligações e não pode gerar o vinculações sem compilar o aplicativo. Há muitas maneiras em torno disso, mas a melhor seria não usar essa abordagem de todo. -There was a number of attempts at writing a static analyser for Wails projects but they didn't get very far. In more recent times, it has become slightly easier to do this with more material available on the subject. +Houve uma série de tentativas de escrever um analisador estático para projetos Wails, mas eles não chegaram muito longe. Em tempos mais recentes, tornou-se um pouco mais fácil fazer isso com mais material disponível sobre o assunto. -Compared to reflection, the AST approach is much faster however it is significantly more complicated. To start with, we may need to impose certain constraints on how to specify bindings in the code. The goal is to support the most common use cases and then expand it later on. +Comparada à reflexão, a abordagem AST é muito mais rápida, porém é significativamente mais complicado. Para começar, podemos precisar impor certas restrições de como especificar ligações no código. O objetivo é suportar os casos de uso mais comuns e, em seguida, expandir mais tarde. -### The Build System +### O Sistema de Compilação -Like the declarative approach to the API, the build system was created to hide the complexities of building a desktop application. When you run `wails build`, it does a lot of things behind the scenes: -- Builds the backend binary for bindings and generates the bindings -- Installs the frontend dependencies -- Builds the frontend assets -- Determines if the application icon is present and if so, embeds it -- Builds the final binary -- If the build is for `darwin/universal` it builds 2 binaries, one for `darwin/amd64` and one for `darwin/arm64` and then creates a fat binary using `lipo` -- If compression is required, it compresses the binary with UPX -- Determines if this binary is to be packaged and if so: - - Ensures the icon and application manifest are compiled into the binary (Windows) - - Builds out the application bundle, generates the icon bundle and copies it, the binary and Info.plist to the application bundle (Mac) -- If an NSIS installer is required, it builds it +Como a abordagem declarativa para a API, o sistema de compilação foi criado para ocultar as complexidades da construção de um aplicativo desktop. Quando você executa `wails build`, ele faz um muitas coisas nos bastidores: +- Constrói o binário de back-end para ligações e gera as ligações +- Instala as dependências de frontend +- Constrói os assets do frontend +- Determina se o ícone da aplicação está presente e, se sim, incorpora-o +- Constrói o binário final +- Se a compilação for para `darwin/universal` compila 2 binários, um para `darwin/amd64` e um para `darwin/arm64` e então cria um binário gordo usando `lipo` +- Se a compressão é necessária, ela compacta o binário com UPX +- Determina se este binário deve ser empacotado e, se isso acontecer: + - Garante que o ícone e manifesto da aplicação são compilados em binário (Windows) + - Constrói o pacote de aplicativos, gera o pacote de ícones e copia-lo, o binário e o Info.plist para o pacote de aplicações (Mac) +- Se um instalador do NSIS for necessário, ele será compilado -This entire process, whilst very powerful, is also very opaque. It is very difficult to customise it and it is very difficult to debug. +Todo este processo, apesar de muito poderoso, é também muito opaco. É muito difícil personalize-o e é muito difícil de depurar. -To address this in v3, I would like to move to a build system that exists outside of Wails. After using [Task](https://taskfile.dev/) for a while, I am a big fan of it. It is a great tool for configuring build systems and should be reasonably familiar to anyone who has used Makefiles. +Para abordar esta questão na v3, gostaria de passar para um sistema de construção que existe fora das Wails. Depois de usar [Tarefa](https://taskfile.dev/) por um tempo, eu sou um grande fã dela. É uma ótima ferramenta para configurar sistemas de construção e deve ser razoavelmente familiar para qualquer um que tenha usado Makefiles. -The build system would be configured using a `Taskfile.yml` file which would be generated by default with any of the supported templates. This would have all of the steps required to do all the current tasks, such as building or packaging the application, allowing for easy customisation. +O sistema de compilação seria configurado usando um arquivo `Taskfile.yml` que seria gerado por padrão com qualquer um dos modelos suportados. Isto teria todas as etapas necessárias para realizar todas as tarefas atuais, como construir ou embalar a aplicação, permitindo fácil personalização. -There will be no external requirement for this tooling as it would form part of the Wails CLI. This means that you can still use `wails build` and it will do all the things it does today. However, if you want to customise the build process, you can do so by editing the `Taskfile.yml` file. It also means you can easily understand the build steps and use your own build system if you wish. +É muito difícil personalizar e é muito difícil de purificar. Isso significa que você ainda pode usar `wails build` e ele fará tudo o que faz hoje. No entanto, se você deseja personalizar o processo de construção, você pode fazê-lo editando o arquivo `Taskfile.yml`. Isso também significa que você pode facilmente entender as etapas de compilação e usar seu próprio sistema de compilação se desejar. -The missing piece in the build puzzle is the atomic operations in the build process, such as icon generation, compression and packaging. To require a bunch of external tooling would not be a great experience for the developer. To address this, the Wails CLI will provide all these capabilities as part of the CLI. This means that the builds still work as expected, with no extra external tooling, however you can replace any step of the build with any tool you like. +A peça que falta no quebra-cabeça da construção são as operações atómicas no processo de construção, como geração, compactação e empacotamento de ícones. Para exigir um monte de ferramentas externas não seria uma ótima experiência para o desenvolvedor. Para resolver isso, a CLI do Wails fornecerá todos esses recursos como parte da CLI. Isso significa que as compilações ainda funcionam como esperado, sem ferramenta externa extra, no entanto você pode substituir qualquer passo da compilação por qualquer ferramenta que você goste. -This will be a much more transparent build system which will allow for easier customisation and address a lot of the issues that have been raised around it. +Este será um sistema de compilação muito mais transparente que permitirá uma personalização mais fácil e resolverá muitas das questões que foram levantadas em torno dele. -## The Payoff +## A Recompensa -These positive changes will be a huge benefit to the project: -- The new API will be much more intuitive and will allow for more complex applications to be built. -- Using static analysis for bindings generation will be much faster and reduce a lot of the complexity around the current process. -- Using an established, external build system will make the build process completely transparent, allowing for powerful customisation. +Estas mudanças positivas serão um grande benefício para o projeto: +- A nova API será muito mais intuitiva e permitirá a criação de aplicativos mais complexos. +- Usar análise estática para geração de ligações será muito mais rápido e reduzirá muito da complexidade em torno do processo atual. +- Usando um sistema de compilação externa estabelecido tornará o processo de construção completamente transparente, permitindo uma personalização poderosa. -Benefits to the project maintainers are: +Os benefícios para os mantenedores do projeto são: -- The new API will be much easier to maintain and adapt to new features and platforms. -- The new build system will be much easier to maintain and extend. I hope this will lead to a new ecosystem of community driven build pipelines. -- Better separation of concerns within the project. This will make it easier to add new features and platforms. +- A nova API será muito mais fácil de manter e adaptar a novos recursos e plataformas. +- O novo sistema de construção será muito mais fácil de manter e alargar. Espero que isto conduza a um novo ecossistema de gasodutos de construção orientados pela comunidade. +- Melhor separação das preocupações no âmbito do projecto. Isso facilitará a adição de novos recursos e plataformas. -## The Plan +## O Plano -A lot of the experimentation for this has already been done and it's looking good. There is no current timeline for this work but I'm hoping by the end of Q1 2023, there will be an alpha release for Mac to allow the community to test, experiment with and provide feedback. +Muitos dos experimentos já foram feitos e estão bons. Não há um cronograma atual para este trabalho, mas espero que até o final do primeiro trimestre de 2023, haja será uma versão alfa para Mac para permitir que a comunidade teste, experimente e dar uma resposta. -## Summary +## Resumo -- The v2 API is declarative, hides a lot from the developer and not suitable for features such as multiple windows. A new API will be created which will be simpler, intuitive and more powerful. -- The build system is opaque and difficult to customise so we will move to an external build system which will open it all up. -- The bindings generation is slow and complex so we will move to static analysis which will remove a lot of the complexity the current method has. +- A API v2 é declarativa, esconde muito do desenvolvedor e não é adequada para recursos, como múltiplas janelas. Será criada uma nova API que será mais simples, intuitiva e mais poderosa. +- O sistema de construção é opaco e difícil de personalizar, pelo que passaremos a um sistema de construção externa que o irá abrir. +- A geração de bindings é lenta e complexa, pelo que avançaremos para uma análise estática, que removerá grande parte da complexidade do método atual. -There has been a lot of work put into the guts of v2 and it's solid. It's now time to address the layer on top of it and make it a much better experience for the developer. +Tem havido muito trabalho colocado nas entranhas da v2 e é sólido. Agora é hora de abordar a camada acima dela e torná-la uma experiência muito melhor para o desenvolvedor. -I hope you are as excited about this as I am. I'm looking forward to hearing your thoughts and feedback. +Espero que você esteja tão empolgado com isso quanto eu. Estou ansioso em ouvir os seus pensamentos e sugestões. -Regards, +Cumprimentos, ‐ Lea -PS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! +PS: Se você ou sua empresa consideram o Wails útil, considere [patrocinar o projeto](https://github.com/sponsors/leaanthony). Obrigado! -PPS: Yes, that's a genuine screenshot of a multi-window application built with Wails. It's not a mockup. It's real. It's awesome. It's coming soon. \ No newline at end of file +PPS: Sim, esse é um genuíno screenshot de um aplicativo com várias janelas construído com Wails. Não é uma simulação. É real. É impressionante. Está chegando. \ No newline at end of file diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/links.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/links.mdx index 58bc9761afd..006ff61ed06 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/links.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/links.mdx @@ -12,7 +12,7 @@ A [lista definitiva](https://github.com/wailsapp/awesome-wails) dos links relaci ## Canais de Suporte -- [Wails Discord Server](https://discord.gg/JDdSxwjhGf) +- [Servidor do Discord Wails](https://discord.gg/JDdSxwjhGf) - [Github Issues](https://github.com/wailsapp/wails/issues) - [v2 Beta Discussion Board](https://github.com/wailsapp/wails/discussions/828) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx index 37be75135ed..6373a7ff16e 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx @@ -7,4 +7,4 @@

``` -The [BulletinBoard](https://github.com/raguay/BulletinBoard) application is a versital message board for static messages or dialogs to get information from the user for a script. It has a TUI for creating new dialogs that can latter be used to get information from the user. It's design is to stay running on your system and show the information as needed and then hide away. I have a process for watching a file on my system and sending the contents to BulletinBoard when changed. It works great with my workflows. There is also an [Alfred workflow](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) for sending information to the program. The workflow is also for working with [EmailIt](https://github.com/raguay/EmailIt). +O aplicativo [BulletinBoard](https://github.com/raguay/BulletinBoard) é um quadro de mensagens versitais para mensagens estáticas ou diálogos para obter informações do usuário para um script. Ele tem uma TUI para criar novas caixas de diálogo que podem ser usadas por último para obter informações do usuário. É um design que fica em execução no seu sistema e mostra as informações conforme necessário e depois se esconde. Tenho um processo para assistir um arquivo no meu sistema e enviar o conteúdo para o BulletinBoard quando alterado. Funciona bem com meus fluxos de trabalho. Há também um [fluxo de trabalho Alfred](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) para enviar informações ao programa. O fluxo de trabalho também é para trabalhar com [EmailIt](https://github.com/raguay/EmailIt). diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx index c1817b70fff..862c304e973 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx @@ -7,4 +7,4 @@

``` -[EmailIt](https://github.com/raguay/EmailIt/) is a Wails 2 program that is a markdown based email sender only with nine notepads, scripts to manipulate the text, and templates. It also has a scripts terminal to run scripts in EmailIt on files in your system. The scripts and templates can be used from the commandline itself or with the Alfred, Keyboard Maestro, Dropzone, or PopClip extensions. It also supports scripts and themes downloaded form GitHub. Documentation is not complete, but the programs works. It’s built using Wails2 and Svelte, and the download is a universal macOS application. +[EmailIt](https://github.com/raguay/EmailIt/) é um programa Wails 2 que é um remetente de e-mail baseado em markdown com apenas nove notepads, scripts para manipular o texto e templates. Ele também tem um terminal para rodar scripts em EmailIt em arquivos do seu sistema. Os scripts e modelos podem ser usados na própria linha de comando ou com as extensões Alfred, Keyboard Maestro, Dropzone ou PopClip. Ele também suporta scripts e temas baixados no formulário GitHub. A documentação não está completa, mas os programas funcionam. É construído usando Wails2 e Svelte, e o download é um aplicativo macOS universal. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx index b8ee7fdf38f..0b5d7fc36ba 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx @@ -13,4 +13,4 @@ O Utilitário de Exportação FileHound permite os admnistradores FileHound a ca Backend construído com: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 -Frontend with: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 +Frontend com: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx index 87e5837d32f..13ed9dc7c53 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx @@ -7,4 +7,4 @@

``` -[hiposter](https://github.com/obity/hiposter) is a simple and efficient http API testing client tool. Based on Wails, Go and sveltejs. +[hiposter](https://github.com/obity/hiposter) é uma ferramenta simples e eficiente de teste de cliente http API. Baseado em Wails, Go and sveltejs. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx index 8a075d19c8a..9633a7d4a84 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx @@ -9,6 +9,6 @@

``` -[Gerenciador de Arquivos Modal](https://github.com/raguay/ModalFileManager) é um gerenciador de arquivos dual pane usando tecnologias da web. Meu design original foi baseado em NW.js e pode ser encontrado [aqui](https://github.com/raguay/ModalFileManager-NWjs). Esta versão usa o mesmo código de frontend baseado em Svelte (mas foi muito modificado desde a partida do NW.), mas o backend implementado com [Wails 2](https://wails.io/). By using this implementation, I no longer use command line `rm`, `cp`, etc. commands, but a git install has to be on the system to download themes and extensions. Está totalmente codificado usando Go e roda muito mais rápido do que as versões anteriores. +[Gerenciador de Arquivos Modal](https://github.com/raguay/ModalFileManager) é um gerenciador de arquivos dual pane usando tecnologias da web. Meu design original foi baseado em NW.js e pode ser encontrado [aqui](https://github.com/raguay/ModalFileManager-NWjs). Esta versão usa o mesmo código de frontend baseado em Svelte (mas foi muito modificado desde a partida do NW.), mas o backend implementado com [Wails 2](https://wails.io/). Ao usar esta implementação, não uso mais comandos de linha de comando `rm`, `cp`, etc., mas um git install deve estar no sistema para baixar temas e extensões. Está totalmente codificado usando Go e roda muito mais rápido do que as versões anteriores. -Este gerenciador de arquivos é projetado em torno do mesmo princípio do Vim: uma ação controlada pelo teclado. O número de estados não é fixo, mas muito programável. Portanto, um número infinito de configurações de teclado pode ser criado e usado. Esta é a principal diferença em relação a outros gerenciadores de arquivos. There are themes and extensions available to download from GitHub. +Este gerenciador de arquivos é projetado em torno do mesmo princípio do Vim: uma ação controlada pelo teclado. O número de estados não é fixo, mas muito programável. Portanto, um número infinito de configurações de teclado pode ser criado e usado. Esta é a principal diferença em relação a outros gerenciadores de arquivos. Existem temas e extensões disponíveis para download do GitHub. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx index 066086458b0..335e360e37a 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx @@ -7,4 +7,4 @@

``` -[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) is a program to show the output of scripts or [Node-Red](https://nodered.org) server. It runs scripts defined in EmailIt program and shows the output. Scripts from xBar or TextBar can be used, but currently on the TextBar scripts work well. Também exibe a saída de scripts no seu sistema. ScriptBar não os coloca na barra de menus, mas os tenha todos em uma janela convidada para fácil visualização. Você pode ter várias abas para ter muitas coisas diferentes mostradas. Você também pode manter os links para os sites mais visitados. +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) é um programa que mostra a saída dos scripts ou o servidor [Node-Red](https://nodered.org). Ele executa scripts definidos no programa EmailIt e mostra a saída. Scripts de xBar ou TextBar podem ser usados, mas atualmente em scripts da TextBar funcionam bem. Também exibe a saída de scripts no seu sistema. ScriptBar não os coloca na barra de menus, mas os tenha todos em uma janela convidada para fácil visualização. Você pode ter várias abas para ter muitas coisas diferentes mostradas. Você também pode manter os links para os sites mais visitados. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx index 950dc3f3db1..c56880f0e19 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx @@ -1,4 +1,4 @@ -# Minecraft launcher for WarMine +# Launcher Minecraft para WarMine ```mdx-code-block

@@ -12,8 +12,8 @@

``` -[Minecraft launcher for WarMine](https://warmine.ru/) is a Wails application, that allows you to easily join modded game servers and manage your game accounts. +[Minecraft Launcher para WarMine](https://warmine.ru/) é uma aplicação Wails que permite que você entre nos servidores de jogos modificados facilmente e gerencie suas contas de jogo. -The Launcher downloads the game files, checks their integrity and launches the game with a wide range of customization options for the launch arguments from the backend. +O Launcher baixa os arquivos do jogo, verifica sua integridade e lança o jogo com uma grande variedade de opções de personalização para os argumentos de lançamento pelo backend. -Frontend is written in Svelte, whole launcher fits in 9MB and supports Windows 7-11. +Frontend está escrito em Svelte, todo o launcher se enquadra em 9MB e suporta Windows 7-11. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/templates.mdx index a019ec078dc..5fab64fd42a 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -25,13 +25,13 @@ Se você não tiver certeza sobre um template, inspecione `package.json` e `wail - [wails-template-vue](https://github.com/misitebao/wails-template-vue) - Wails template based on Vue ecology (Integrated TypeScript, Dark theme, Internationalization, Single page routing, TailwindCSS) - [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Vue 3 TypeScript with Vite (and instructions to add features) - [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vue 3 TypeScript with Vite, Vuex, Vue Router, Sass, and ESLint + Prettier -- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - A template using JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) -- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - A template using TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, Composition API with <script setup>) -- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Wails template based on Naive UI (A Vue 3 Component Library) +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - Um template usando JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - Um modelo usando TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, API de composição com <configuração de script>) +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Modelo de ervas baseado em Naive UI (biblioteca de componentes Vue 3) ## Angular -- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - Angular 15+ action packed & ready to roll to production. +- [A-modelo de wails-angular](https://github.com/mateothegreat/wails-template-angular) - ação Angular 15 + compactada & pronta para ser rolada para produção. - [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - Angular with TypeScript, Sass, Hot-Reload, Code-Splitting and i18n ## React @@ -40,26 +40,30 @@ Se você não tiver certeza sobre um template, inspecione `package.json` e `wail - [wails-react-template](https://github.com/flin7/wails-react-template) - A minimal template for React that supports live development - [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - A template using Next.js and TypeScript - [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - A template for React + TypeScript + Vite + TailwindCSS -- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - Um modelo com Vite, React, TypeScript, TailwindCSS, e shadcn/ui ## Svelte - [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - A template using Svelte - [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - A template using Svelte and Vite - [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - A template using Svelte and Vite with TailwindCSS v3 -- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - Um modelo atualizado usando Svelte v4.2.0 e Vite com TailwindCSS v3.3.3 - [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - A template using SvelteKit ## Solid -- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - A template using Solid + Ts + Vite -- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - A template using Solid + Js + Vite +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - Um modelo usando Sólido + Ts + Vite +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - Um modelo usando Sólid + Js + Vite ## Elm - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + ## Pure JavaScript (Vanilla) - [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx index de8bebe77b3..be641d3b98d 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx @@ -7,7 +7,7 @@ sidebar_position: 5 Você pode executar sua aplicação no modo de desenvolvimento executando `wails dev` no diretório do seu projeto. Isso fará o seguinte: - Construa seu aplicativo e o execute -- Bind your Go code to the frontend so it can be called from JavaScript +- Vincule seu código Go para o frontend para que ele possa ser chamado a partir de JavaScript - Usando o poder do [Vite](https://vitejs.dev/), observará modificações em seus arquivos Go e reconstruir/re-executar na alteração - Configure um [servidor web](http://localhost:34115) que irá servir seu aplicativo em um navegador. Isso permite usar suas extensões de navegador favoritas. Você pode até mesmo chamar seu código Go do console diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/angular.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/angular.mdx index 2b6c5a84571..cae0d7f95a5 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/angular.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/angular.mdx @@ -1,10 +1,10 @@ # Angular -Whilst Wails does not have an Angular template, it is possible to use Angular with Wails. +Embora as Wails não tenham um modelo angular, é possível usar Angular com Wails. -## Dev Mode +## Modo Desenvolvedor -To get dev mode working with Angular, you need to add the following to your `wails.json`: +Para que o modo de desenvolvimento funcione com Angular, você precisa adicionar o seguinte ao seu `wails.json`: ```json "frontend:build": "npx ng build", diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/application-development.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/application-development.mdx index 0d78b9d9a47..f81eae7fa8d 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/application-development.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/application-development.mdx @@ -189,16 +189,16 @@ O servidor de desenvolvimento utiliza uma técnica chamada "debwaring" que signi Alguns frameworks vêm com seu próprio servidor ao vivo, no entanto, eles não serão capazes de tirar proveito das ligações Go/Wails. Neste cenário, é melhor executar um script de observador que reconstrui o projeto no diretório build, que Wails estará assistindo. Por exemplo, veja o modelo padrão do svelte que usa [rollup](https://rollupjs.org/guide/en/). -### Create React App +### Criar Aplicativo React -The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: +O processo para um projeto Create-React-App é um pouco mais complicado. Para ajudar o frontend a recarregar ao vivo a seguinte configuração precisa ser adicionado ao seu `wails.json`: ```json "frontend:dev:watcher": "yarn start", "frontend:dev:serverUrl": "http://localhost:3000", ``` -The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: +O comando `frontend: dev:watcher` iniciará o servidor de desenvolvimento Create-React-App (hospedado na porta `3000` normalmente). O comando `frontend: dev:serverUrl` e instrui a Wails a servir assets do servidor de desenvolvimento ao carregar o frontend em vez de a partir da pasta de compilação. Além do acima acima o `index.html` precisa ser atualizado com o seguinte: ```html @@ -208,7 +208,7 @@ The `frontend:dev:watcher` command will start the Create-React-App development s ``` -This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. +Isso é necessário pois o comando do observador que reconstrui o frontend impede que o Wails injete os scripts necessários. Isso contorna esse problema garantindo os scripts são sempre injetados. Com esta configuração, `ondas de desenvolvimento` pode ser executado, o que irá construir adequadamente o frontend e o backend com o carregamento de quente ativado. Além disso, ao acessar o aplicativo a partir de um navegador, as ferramentas de desenvolvedor do React agora podem ser usadas em uma versão não minificada do aplicativo para simplificar depuração. Finalmente, para compilações mais rápidas, `wail dev -s` pode ser executado para ignorar o edifício padrão do frontend pelas Wails, pois este é um passo desnecessário. ## Go Module diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx index 989202e27a6..627bbeb5e36 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx @@ -1,12 +1,12 @@ -# Dynamic Assets +# Recursos Dinâmicos -If you want to load or generate assets for your frontend dynamically, you can achieve that using the [AssetsHandler](../reference/options#assetshandler) option. The AssetsHandler is a generic `http.Handler` which will be called for any non GET request on the assets server and for GET requests which can not be served from the bundled assets because the file is not found. +Se você deseja carregar ou gerar assets para seu frontend de forma dinâmica, você pode conseguir isso usando a opção [AssetsHandler](../reference/options#assetshandler). O AssetsHandler é um `http.Handler` genérico que irá ser chamado para qualquer solicitação não GET no servidor de ativos e para solicitações GET que não podem ser atendidas pelo ativos agrupados porque o arquivo não foi encontrado. -By installing a custom AssetsHandler, you can serve your own assets using a custom asset server. +Ao instalar um AssetsHandler personalizado, você pode servir seus próprios assets usando um servidor de arquivos personalizado. -## Example +## Exemplo -In our example project, we will create a simple assets handler which will load files off disk: +Em nosso projeto de exemplo, vamos criar um gerenciador de arquivos simples que irá carregar arquivos fora do disco: ```go title=main.go {17-36,49} package main @@ -72,7 +72,7 @@ func main() { } ``` -When we run the application in dev mode using `wails dev`, we will see the following output: +Quando executarmos o aplicativo no modo de desenvolvimento usando `wail dev`, veremos a seguinte saída: ``` DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' @@ -80,15 +80,15 @@ DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, Requesting file: favicon.ico ``` -As you can see, the assets handler is called when the default assets server is unable to serve the `favicon.ico` file. +Como você pode ver, o manipulador de assets é chamado quando o servidor de assets padrão não consegue servir o arquivo `favicon.ico`. -If you right click the main application and select "inspect" to bring up the devtools, you can test this feature out by typing the following into the console: +Se você clicar com o botão direito no aplicativo principal e selecionar "inspeção" para abrir as devtools, você pode testar esta função digitando o seguinte no console: ``` let response = await fetch('does-not-exist.txt'); ``` -This will generate an error in the devtools. We can see that the error is what we expect, returned by our custom assets handler: +Isto irá gerar um erro no devtools. Podemos ver que o erro é o que esperamos, retornado por nosso manipulador de ativos personalizados: ```mdx-code-block

@@ -98,7 +98,7 @@ This will generate an error in the devtools. We can see that the error is what w

``` -However, if we request `go.mod`, we will see the following output: +No entanto, se requisitarmos `go.mod`, veremos a seguinte saída: ```mdx-code-block

@@ -106,19 +106,19 @@ However, if we request `go.mod`, we will see the following output:

``` -This technique can be used to load images directly into the page. If we updated our default vanilla template and replaced the logo image: +Esta técnica pode ser usada para carregar imagens diretamente para a página. Se atualizarmos nosso modelo padrão do vanilla e substituirmos a imagem do logotipo: ```html ``` -with: +com: ```html ``` -Then we would see the following: +Veremos então o seguinte: ```mdx-code-block

@@ -131,6 +131,6 @@ Then we would see the following: :::warning -Exposing your filesystem in this way is a security risk. It is recommended that you properly manage access to your filesystem. +Expor seu sistema de arquivos desta forma é um risco à segurança. É recomendável que você gerencie corretamente o acesso ao seu sistema de arquivos. ::: diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/frameless.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/frameless.mdx index 3845736f427..430f53a6afc 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/frameless.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/frameless.mdx @@ -1,8 +1,8 @@ -# Frameless Applications +# Aplicações sem frames -Wails supports application that have no frames. This can be achieved by using the [frameless](../reference/options.mdx#frameless) field in [Application Options](../reference/options.mdx#application-options). +O Wails suporta aplicativos que não possuem frames. Isso pode ser conseguido usando o campo [sem frameless](../reference/options.mdx#frameless) no [Application Options](../reference/options.mdx#application-options). -Wails offers a simple solution for dragging the window: Any HTML element that has the CSS style `--wails-draggable:drag` will act as a "drag handle". This property applies to all child elements. If you need to indicate that a nested element should not drag, then use the attribute '--wails-draggable:no-drag' on that element. +Wails oferece uma solução simples para arrastar a janela: qualquer elemento HTML que tenha o estilo CSS `--wails-draggable:drag` irá atuar como uma "alça de arrastar". Esta propriedade se aplica a todos os elementos filhos. Se você precisar indicar que um elemento aninhado não deve arrastar, então use o atributo '--wails-draggable:no-drag' nesse elemento. ```html @@ -23,7 +23,7 @@ Wails offers a simple solution for dragging the window: Any HTML element that ha ``` -For some projects, using a CSS variable may not be possible due to dynamic styling. In this case, you can use the `CSSDragProperty` and `CSSDragValue` application options to define a property and value that will be used to indicate draggable regions: +Para alguns projetos, usar uma variável CSS pode não ser possível devido a um estilo dinâmico. Neste caso, você pode usar o aplicativo `CSSDragProperty` e `CSSDragValue` opções para definir uma propriedade e valor que serão usados para indicar regiões arrastáveis: ```go title=main.go package main @@ -80,8 +80,8 @@ func main() { ``` -:::info Fullscreen +:::info Tela Cheia -If you allow your application to go fullscreen, this drag functionality will be disabled. +Se você permitir que seu aplicativo vá para tela cheia, esta funcionalidade de arrastar será desativada. ::: diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/local-development.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/local-development.mdx index 4ba1f34c37d..4ba06231463 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/local-development.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/local-development.mdx @@ -1,36 +1,36 @@ -# Local Development +# Desenvolvimento Local -## Overview +## Visão geral -Wails is in constant development and new releases are regularly "tagged". This usually happens when all the newer code on `master` has been tested and confirmed working. If you need a bugfix or feature that has not yet made it to a release, it's possible to use the latest "bleeding edge" version using the following steps: +Os dispositivos estão em constante desenvolvimento e novos lançamentos são regularmente "marcados". Isso geralmente acontece quando todo o código no `master` foi testado e confirmado funcionando. Se você precisar de uma correção de bug ou recurso que ainda não foi lançado, é possível usar a versão mais recente "bleeding edge" seguindo as seguintes etapas: - `git clone https://github.com/wailsapp/wails` - `cd wails/v2/cmd/wails` - `go install` -NOTE: The directory that you cloned the project into will now be called "clonedir". +NOTA: O diretório para o qual você clonou o projeto será agora chamado de "clonedir". -The Wails CLI will now be at the very latest version. +A CLI do Wails estará na versão mais recente. -### Updating your project +### Atualizando seu projeto -To update projects to use the latest version of the Wails library, update the project's `go.mod` and ensure the following line is at the bottom of the file: +Para atualizar projetos para usar a versão mais recente da biblioteca Wails, atualize o `do projeto. od` e certifique-se que a seguinte linha está no final do arquivo: `replace github.com/wailsapp/wails/v2 => ` -Example: +Exemplo: -On Windows: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` +Windows: `substitua github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` -On 'nix: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` +Em 'nix: `substitui github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` -To revert to a stable version, run: +Para reverter para uma versão estável, execute: `go install github.com/wailsapp/wails/v2/cmd/wails@latest` -## Testing a Branch +## Testando uma Branch -If you want to test a branch, follow the instructions above, but ensure you switch the branch you want to test before installing: +Se você deseja testar um branch, siga as instruções acima, mas certifique-se de alternar o branch que você deseja testar antes de instalar: - `git clone https://github.com/wailsapp/wails` - `cd wails` @@ -38,11 +38,11 @@ If you want to test a branch, follow the instructions above, but ensure you swit - `cd v2/cmd/wails` - `go install` -Make sure you [update your project](#updating-your-project) as described above. +Certifique-se de [atualizar seu projeto](#updating-your-project) conforme descrito acima. -## Testing a PR +## Testando uma PR -If you want to test a PR, follow the instructions above, but ensure you fetch the PR and switch the branch before installing. Please replace `[IDofThePR]` with the ID of the PR shown on github.com: +Se você deseja testar um branch, siga as instruções acima, mas certifique-se de alternar o branch que você deseja testar antes de instalar. Por favor, substitua `[IDofThePR]` pelo ID do PR mostrado no github.com: - `git clone https://github.com/wailsapp/wails` - `cd wails` @@ -52,4 +52,4 @@ If you want to test a PR, follow the instructions above, but ensure you fetch th - `cd v2/cmd/wails` - `go install` -Make sure you [update your project](#updating-your-project) as described above. +Certifique-se de [atualizar seu projeto](#updating-your-project) conforme descrito acima. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx index 961595711c7..4b7b589c1a4 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx @@ -1,42 +1,42 @@ -# Mac App Store Guide +# Guia para Mac App Store -This page gives a brief overview of how to submit your Wails App to the Mac App Store. +Esta página dá uma breve visão geral de como enviar seu App Wails para a Mac App Store. -## Prerequisites +## Pré-requisitos -- You will need to have an Apple Developer account. Please find more information on the [Apple Developer Program](https://developer.apple.com/support/compare-memberships/) site -- You will need to have your Certificates, Identifiers, and App created on the developer portal. More on this below -- Xcode command line tools will need to be installed on your local machine +- Você precisará ter uma conta para o Apple Develop. Encontre mais informações no site [do Programa de Desenvolvedor Apple](https://developer.apple.com/support/compare-memberships/) +- Você precisará ter seus Certificados, Identificadores e App criados no portal de desenvolvimento. Mais sobre isto abaixo +- Ferramentas de linha de comando Xcode precisarão ser instaladas na sua máquina local -#### Create Certificates and Identifiers +#### Criar certificados e identificadores -1. Go to your [Apple Developer Account](https://developer.apple.com/account/) -2. Under `Certificates, Identifiers & Profiles`, click `Identifiers` and Register a New App ID. Use the format (com.example.app) -3. Under the same page click `Certificates` and generate new Certificates for Mac App Store Distribution. Download them and import the certificates into Keychain on your local machine. +1. Vá para sua [Conta de desenvolvedor Apple](https://developer.apple.com/account/) +2. Sob `Certificados, Identificadores & Perfis`, clique em `Identificadores` e Registrar um Novo App ID. Use o formato (com.exemplo.app) +3. Sob a mesma página, clique `Certificados` e gere novos Certificados para a Distribuição da Loja de Aplicativos Mac. Baixe-os e importe os certificados para o Keychain em sua máquina local. -#### Create App Submission +#### Criar Envio de App -1. Go to the [App Store Connect Site](https://appstoreconnect.apple.com/apps) -2. Register a new application and link the bundle ID that you created in the previous step -3. Populate your app with the correct screen shots, descriptions, etc. as required by Apple -4. Create a new version of your app +1. Ir para [App Store Connect Site](https://appstoreconnect.apple.com/apps) +2. Registre um novo aplicativo e vincule o ID do pacote que você criou no passo anterior +3. Preencher seu aplicativo com as capturas de tela corretas, descrições, etc. conforme exigido pela Apple +4. Criar uma nova versão do seu aplicativo -#### Create Provisioning Profile -1. Go to the [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) page -2. Add a new provisioning profile for Mac App Store Distribution -3. Set the Profile Type as Mac and select the App ID for the application created above -4. Select the Mac App Distribution certificate -5. Name the Provisioning Profile embedded and download the created profile. +#### Criar perfil de provisionamento +1. Vá para a página [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) +2. Adicionar um novo perfil de provisionamento para Mac App Store de distribuição +3. Defina o tipo de perfil como Mac e selecione o ID do aplicativo criado acima +4. Selecione o certificado de Distribuição de Aplicativos Mac +5. Nomeie o Perfil de Provisão incorporado e baixe o perfil criado. -## Mac App Store Process +## Guia para Mac App Store -#### Enable Apple's App Sandbox +#### Ativar a App Sandbox da Apple -Apps submitted to the Mac App Store must run under Apple's [App Sandbox](https://developer.apple.com/app-sandboxing/). You must create an `entitlements.plist` file for this to work. The recommendation is to create this file under this path `{PROJECT_DIR}/build/darwin/entitlements.plist`. +Os aplicativos enviados para Mac App Store devem ser executados na [App Sandbox](https://developer.apple.com/app-sandboxing/) da Apple. Você deve criar um arquivo `entitlements.plist` para que isso funcione. A recomendação é criar este arquivo sob este caminho `{PROJECT_DIR}/build/darwin/entitlements.plist`. -**Example Entitlements File** +**Exemplo de arquivo de direitos** -This is an example entitlements file from the [RiftShare](https://github.com/achhabra2/riftshare) app. For reference please put in the entitlements your app requires. Refer to [this site](https://developer.apple.com/documentation/bundleresources/entitlements) for more information. You will need to replace the Team ID and Application Name with the ones you registered above. +Este é um exemplo de titularidade de arquivo do aplicativo [RiftShare](https://github.com/achhabra2/riftshare). Para referência, por favor coloque os direitos que seu aplicativo exigir. Consulte [este site](https://developer.apple.com/documentation/bundleresources/entitlements) para obter mais informações. Você precisará substituir a ID da Equipe e o Nome da Aplicação pelos que você se registrou acima. ```xml title="entitlements.plist" @@ -61,13 +61,13 @@ This is an example entitlements file from the [RiftShare](https://github.com/ach ``` -**Add the Embedded Provisioning Profile** The Provisioning Profile created above needs to be added to the root of the applicaton. It needs to be named embedded.provisionprofile. +**Adicionar Perfil de Provisão Embutido** O Perfil de Provisionamento criado acima precisa ser adicionado à raiz da aplicação. Precisa ser nomeado como embedded.provisionprofile. -#### Build and Sign the App Package +#### Construa e assine o Pacote de Aplicativos -The following is an example script for building and signing your app for Mac App Store submission. It assumes you are running the script from your root project directory. +O seguinte é um exemplo de script para construir e assinar seu aplicativo para o envio da Mac App Store. Ele presume que você está executando o script do diretório do seu projeto raiz. -Note the certificates for signing the app and signing the installer are different. Please make sure both are imported into Keychain. Find the strings in Keychain and insert them below. Populate your certificate names, and app name below. Running the following script will generate a signed `app.pkg` file in the root directory of your app. +Observe que os certificados para a assinatura do aplicativo e a assinatura do instalador são diferentes. Certifique-se de que ambos são importados para o Keychain. Encontre as sequências de caracteres no Keychain e insira-as abaixo. Preencha os nomes do seu certificado e o nome do app abaixo. Executar o seguinte script irá gerar um arquivo `assinado app.pkg` no diretório raiz do seu aplicativo. ```bash title="macappstore-build.sh" #!/bin/bash @@ -85,13 +85,13 @@ codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements . productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg ``` -#### Upload App Bundle +#### Enviar pacote de aplicativos -You will need to upload the generated package file and associate it to your Application before you will be able to submit it for review. +Você precisará enviar o arquivo do pacote gerado e associá-lo à sua Aplicação antes de poder enviá-lo para revisão. -1. Download the [Transporter App](https://apps.apple.com/us/app/transporter/id1450874784) from the Mac App Store -2. Open it and sign in with your Apple ID -3. Click the + sign and select the `APP_NAME.pkg` file that you generated in the previous step. Upload it -4. Go back to the [App Store Connect](https://appstoreconnect.apple.com/apps) site and navigate back into your app submission. Select the version that you are ready to make available on the App Store. Under `Build` select the package that you uploaded via Transporter. +1. Baixe o [aplicativo Transporter](https://apps.apple.com/us/app/transporter/id1450874784) na Mac App Store +2. Abra e inicie sessão com a sua Apple ID +3. Clique no sinal + e selecione o arquivo `APP_NAME.pkg` que você gerou na etapa anterior. Carregar isto +4. Volte para [Loja de Apps Conectar](https://appstoreconnect.apple.com/apps) e navegue de volta para a submissão de seu aplicativo. Selecione a versão que você está pronto para disponibilizar na App Store. Em `Build` selecione o pacote que você enviou via Transporter. -That's it! You can now use the site to submit your App for review. After a few business days if all goes well you should see your App live on the Mac App Store. +É isso! Agora você pode usar o site para enviar seu Aplicativo para análise. Após alguns dias úteis, se tudo correr bem, você verá seu App ao vivo na Mac App Store. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/migrating.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/migrating.mdx index 7123cbe6b60..a874b03f4f5 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/migrating.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/migrating.mdx @@ -1,14 +1,14 @@ -# Migrating from v1 +# Migrando da v1 -## Overview +## Visão geral -Wails v2 is a significant change from v1. This document aims to highlight the changes and the steps in migrating an existing project. +Wails v2 é uma mudança significativa em relação à v1. Este documento visa destacar as mudanças e as etapas na migração de um projeto existente. -### Creating the Application +### Criando uma aplicação Cli -In v1, the main application is created using `wails.CreateApp`, bindings are added with `app.Bind`, then the application is run using `app.Run()`. +Na v1, o aplicativo principal é criado usando `wails.CreateApp`, as ligações são adicionadas com `app.Bind` e, em seguida, o o aplicativo é executado usando `app.Run()`. -Example: +Exemplo: ```go title="v1" app := wails.CreateApp(&wails.AppConfig{ @@ -23,7 +23,7 @@ Example: app.Run() ``` -In v2, there is just a single method, `wails.Run()`, that accepts [application options](../reference/options.mdx#application-options). +Na v2, há apenas um único método, `wails.Run()`, que aceita as opções de aplicação [](../reference/options.mdx#application-options). ```go title="v2" err := wails.Run(&options.App{ @@ -39,9 +39,9 @@ In v2, there is just a single method, `wails.Run()`, that accepts [application o }) ``` -### Binding +### Mapeamento -In v1, it was possible to bind both arbitrary functions and structs. In v2, this has been simplified to only binding structs. The struct instances that were previously passed to the `Bind()` method in v1, are now specified in the `Bind` field of the [application options](../reference/options.mdx#application-options): +Na v1, foi possível vincular funções e estruturas arbitrárias. Em v2, foi simplificado apenas para estruturas vinculativas. As instâncias de construção que foram passadas anteriormente para o método `Bind()` na v1, estão agora especificados no campo `Vincular` do as opções de aplicação [](../reference/options.mdx#application-options): ```go title="v1" app := wails.CreateApp(/* options */) @@ -57,19 +57,19 @@ In v1, it was possible to bind both arbitrary functions and structs. In v2, this }) ``` -In v1, bound methods were available to the frontend at `window.backend`. This has changed to `window.go`.`` +Na v1, métodos vinculados estavam disponíveis para o frontend em `window.backend`. Isto mudou para `window.go` -### Application Lifecycle +### Ciclo de vida da Aplicação -In v1, there were 2 special methods in a bound struct: `WailsInit()` and `WailsShutdown()`. These have been replaced with 3 lifecycle hooks as part of the [application options](../reference/options.mdx#application-options): +Na v1, havia 2 métodos especiais em uma estrutura vinculada: `WailsInit()` e `WailsShutdown()`. Foi substituído por três ganchos de ciclo de vida como parte das [opções do aplicativo](../reference/options.mdx#application-options): - [OnStartup](../reference/options.mdx#onstartup) - [OnShutdown](../reference/options.mdx#onshutdown) - [OnDomReady](../reference/options.mdx#ondomready) -Note: [OnDomReady](../reference/options.mdx#ondomready) replaces the `wails:ready` system event in v1. +Nota: [OnDomReady](../reference/options.mdx#ondomready) substitui o evento do sistema `wails:ready` na v1. -These methods can be standard functions, but a common practice is to have them part of a struct: +Estes métodos podem ser funções padrão, mas uma prática comum é tê-los incluído numa estrutura: ```go title="v2" basic := NewBasicApp() @@ -89,11 +89,11 @@ func (b *Basic) startup(ctx context.Context) { ... ``` -### Runtime +### Tempo de execução -The runtime in v2 is much richer than v1 with support for menus, window manipulation and better dialogs. The signature of the methods has changed slightly - please refer the the [Runtime Reference](../reference/runtime/intro.mdx). +O tempo de execução na v2 é muito mais rico que a v1 com suporte para menus, manipulação de janelas e melhores diálogos. A assinatura dos métodos mudou ligeiramente - consulte a [Referência de tempo de execução](../reference/runtime/intro.mdx). -In v1, the [runtime](../reference/runtime/intro.mdx) was available via a struct passed to `WailsInit()`. In v2, the runtime has been moved out to its own package. Each method in the runtime takes the `context.Context` that is passed to the [OnStartup](../reference/options.mdx#onstartup) method. +Na v1, o [runtime](../reference/runtime/intro.mdx) estava disponível através de uma struct passada para `WailsInit()`. Em v2, o tempo de execução foi movido para o seu próprio pacote. Cada método no tempo de execução leva o `context.Context` que é passado para o método [OnStartup](../reference/options.mdx#onstartup). ```go title="Runtime Example" package main @@ -114,22 +114,22 @@ func (a *App) startup(ctx context.Context) { ### Assets -The _biggest_ change in v2 is how assets are handled. +A _maior mudança_ na v2 é como os ativos são geridos. -In v1, assets were passed via 2 application options: +Na v1, os ativos foram passados via 2 opções de aplicativo: -- `JS` - The application's JavaScript -- `CSS` - The application's CSS +- `JS` - O JavaScript do aplicativo +- `CSS` - O CSS da aplicação -This meant that the responsibility of generating a single JS and CSS file was on the developer. This essentially required the use of complicated packers such as webpack. +Isso significava que a responsabilidade de gerar um único arquivo JS e CSS era do desenvolvedor. Isto exigia essencialmente a utilização de embalagens complicadas como o webpack. -In v2, Wails makes no assumptions about your frontend assets, just like a webserver. All of your application assets are passed to the application options as an `embed.FS`. +Na v2, Wails não faz suposições sobre seus ativos no frontend, como um servidor web. Todos os seus ativos de aplicação são passados para as opções de aplicação como um `embed.FS`. -**This means there is no requirement to bundle your assets, encode images as Base64 or attempt the dark art of bundler configuration to use custom fonts**. +**Isso significa que não há necessidade de agrupar seus ativos, codificar imagens como Base64 ou experimente a arte obscura da configuração do bundler para usar fontes personalizadas**. -At startup, Wails will scan the given `embed.FS` for `index.html` and use its location as the root path for all the other application assets - just like a webserver would. +Na inicialização, Wails verificará o `embed.FS` fornecido em busca de `index.html` e usará sua localização como caminho raiz para todos os outros ativos do aplicativo - assim como faria um servidor web. -Example: An application has the following project layout. All final assets are placed in the `frontend/dist` directory: +Exemplo: Uma aplicação tem o seguinte layout do projeto. Todos os arquivos finais são colocados no diretório `frontend/dist`: ```shell . @@ -144,7 +144,7 @@ Example: An application has the following project layout. All final assets are p └── wails.json ``` -Those assets may be used by the application by simply creating an `embed.FS`: +Esses ativos podem ser usados pelo aplicativo simplesmente criando um `embed.FS`: ```go title="Assets Example" //go:embed all:frontend/dist @@ -160,13 +160,13 @@ func main() { } ``` -Of course, bundlers can be used if you wish to. The only requirement is to pass the final application assets directory to Wails using an `embed.FS` in the `Assets` key of the [application options](../reference/options.mdx#application-options). +Claro, empacotadores podem ser usados se você quiser. O único requisito é passar o diretório final de ativos do aplicativo para Wails usando um `embed.FS` no `Assets` chave das [opções do aplicativo](../reference/options.mdx#application-options). -### Project Configuration +### Configuração do Projeto -In v1, the project configuration was stored in the `project.json` file in the project root. In v2, the project configuration is stored in the `wails.json` file in the project root. +Na v1, a configuração do projeto foi armazenada no arquivo `project.json` na raiz do projeto. Na v2, a configuração do projeto é armazenada no arquivo `wails.json` na raiz do projeto. -The format of the file is slightly different. Here is a comparison: +O formato do arquivo é ligeiramente diferente. Aqui está uma comparação:

@@ -185,7 +185,7 @@ The format of the file is slightly different. Here is a comparison: | frontend / serve | | Removed | | tags | | Removed | | | wailsjsdir | The directory to generate wailsjs modules | -| | assetdir | The directory of the compiled frontend assets for `dev` mode. This is normally inferred and could be left empty. | -| | reloaddirs | Comma separated list of additional directories to watch for changes and to trigger reloads in `dev` mode. This is only needed for some more advanced asset configurations. | +| | assetdir | The directory of the compiled frontend assets for `dev` mode. Normalmente, isto é inferido e pode ser deixado vazio. | +| | reloaddirs | Lista separada por vírgulas de diretórios adicionais para observar alterações e acionar recarregamentos no modo `dev`. Isso só é necessário para algumas configurações de ativos mais avançadas. |

diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx index 4a3de2a61b5..054ac263bd1 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx @@ -1,6 +1,6 @@ -# Mouse Buttons +# Botões do Mouse -The Wails runtime intercepts mouse clicks to determine whether a frameless window needs resizing or a window needs to be moved. It has been asked how to detect when a mouse click has occurred, because `window.onclick` doesn't report the mouse buttons correctly. The following code shows how to detect mouse clicks: +As Wails runtime interceptam cliques do mouse para determinar se uma janela precisa ser redimensionada ou uma janela precisa ser movida. Foi perguntado como detectar quando um clique do mouse ocorreu, porque `window.onclick` não relata os botões do mouse corretamente. O código a seguir mostra como detectar cliques do mouse: ```javascript window.addEventListener("mousedown", handleMouseButtonDown); diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx index 21f7875e389..d921a6a5fc2 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx @@ -1,32 +1,32 @@ -# Obfuscated Builds +# Builds ofuscadas -Wails includes support for obfuscating your application using [garble](https://github.com/burrowers/garble). +Wails inclui suporte para ofuscar a sua aplicação usando [garble](https://github.com/burrowers/garble). -To produce an obfuscated build, you can use the `-obfuscate` flag with the `wails build` command: +Para produzir uma compilação ofuscada, você pode usar o sinalizador `-obfuscate` com o comando `wails build`: ```bash wails build -obfuscated ``` -To customise the obfuscation settings, you can use the `-garbleargs` flag: +Para personalizar a configuração de ofuscação, pode-se utilizar a flag: `-garbleargs`: ```bash wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" ``` -These settings may be persisted in your [project config](../reference/project-config). +Essas configurações podem estar persistentes na configuração do seu [projeto](../reference/project-config). -## How it works +## Como funciona -In a standard build, all bound methods are available in the frontend under the `window.go` variable. When these methods are called, the corresponding backend method is called using the fully qualified function name. When using an obfuscated build, methods are bound using an ID instead of a name. The bindings generated in the `wailsjs` directory use these IDs to call the backend functions. +Em uma compilação padrão, todos os métodos vinculados estão disponíveis no frontend sob a variável `window.go`. Quando esses métodos são chamados, o método backend correspondente é chamado usando o nome da função totalmente qualificada. Quando usando uma compilação ofuscada, os métodos são vinculados usando um ID em vez de um nome. Os bindings gerados no diretório `wailsjs` usam esses IDs para chamar as funções de backend. :::note -To ensure that your application will work in obfuscated mode, you must use the generated bindings under the `wailsjs` directory in your application. +Para garantir que o seu aplicativo irá funcionar no modo ofuscado, você deve usar os bindings geradas sob o diretório `wailsjs` no seu aplicativo. ::: -## Example +## Exemplo Importing the "Greet" method from the bindings like this: @@ -37,4 +37,4 @@ import { Greet } from "../../wailsjs/go/main/App"; Greet("World"); ``` -will ensure that the method will work correctly in obfuscated mode, as the bindings will be regenerated with IDs and the call mechanism updated. +irá garantir que o método funcionará corretamente no modo ofuscado, como os bindings serão regenerados com IDs e o mecanismo de chamada atualizado. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/overscroll.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/overscroll.mdx index 9d1d772d0fb..dcd13f147c3 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/overscroll.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/overscroll.mdx @@ -1,6 +1,6 @@ -# Overscroll +# Rolar demais -[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) is the "bounce effect" you sometimes get when you scroll beyond a page's content boundaries. This is common in mobile apps. This can be disabled using CSS: +[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) é o "efeito de salto" que você às vezes obtém quando você rola além dos limites de conteúdo de uma página. This is common in mobile apps. Isso pode ser desativado usando CSS: ```css html { diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/routing.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/routing.mdx index 90d06d2872d..550e73a57f4 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/routing.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/routing.mdx @@ -27,7 +27,7 @@ RouterModule.forRoot(routes, { useHash: true }); ## React -The recommended approach for routing in React is [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): +A abordagem recomendada para roteamento em React é [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): ```jsx import { HashRouter } from "react-router-dom"; diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx index 4651c422ed1..8281281f325 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx @@ -1,66 +1,66 @@ # SvelteKit -This guide will go into: +Este guia será para: -1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. -2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. -3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. +1. Passos de instalação do Miminal - Os passos necessários para obter uma instalação mínima de Wails funcionando para o SvelteKit. +2. Script de Instalação - Bash script para cumprir os passos de Instalação Mínima com a marca opcional de Wails. +3. Notas importantes - Problemas que podem ser encontrados ao usar o SvelteKit + Wails e correções. -## 1. Minimal Installation Steps +## 1. Passos mínimos de instalação -##### Install Wails for Svelte. +##### Instalar Wails para Svelte. - `wails init -n myapp -t svelte` -##### Delete the svelte frontend. +##### Exclua o front-end do Svelte. -- Navigate into your newly created myapp folder. -- Delete the folder named "frontend" +- Navegue até a pasta myapp recém-criada. +- Excluir a pasta chamada "frontend" -##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. +##### Enquanto estiver na raiz do projeto Wails. Use o seu gerenciador de pacotes favorito e instale o SvelteKit como o novo frontend. Siga as instruções. - `npm create svelte@latest frontend` -##### Modify wails.json. +##### Modificar o wails.json. -- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. -- Change your package manager frontend here if not using npm. +- Add `"wailsjsdir": "./frontend/src/lib",` Observe que é aqui que suas funções Go e runtime aparecerão. +- Mude o frontend do seu gerenciador de pacotes se não estiver usando npm. -##### Modify main.go. +##### Modificar o main.go. -- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` +- O primeiro comentário `//go:embed all:frontend/dist` precisa ser alterado para `//go:embed all:frontend/build` -##### Install/remove dependencies using your favorite package manager. +##### Instalar/remover dependências usando seu gerenciador de pacote favorito. -- Navigate into your "frontend" folder. +- Entre na sua pasta "frontend". - `npm i` - `npm uninstall @sveltejs/adapter-auto` - `npm i -D @sveltejs/adapter-static` -##### Change adapter in svelte.config.js +##### Alterar adaptador em svelte.config.js -- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` +- A primeira linha de arquivo altera `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` -##### Put SvelteKit into SPA mode with prerendering. +##### Coloque o SvelteKit no modo SPA com pré-renderização. -- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. -- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` +- Crie um arquivo sob myapp/frontend/src/routes/ chamado +layout.ts/+layout.js. +- Adicione duas linhas ao recém-criado arquivo `export const prerender = true` e `export const ssr = false` -##### Test installation. +##### Testar instalação. -- Navigate back into the Wails project root (one directory up). -- run `wails dev` -- If the application doesn't run please check through the previous steps. +- Navegue de volta à raiz do projeto Wails (um diretório para cima). +- execute `wails dev` +- Se a aplicação não executar, por favor verifique os passos anteriores. -## 2. Install Script +## 2. Script de Instalação -##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. +##### Este Script do Bash faz as etapas listadas acima. Certifique-se de ler o script e de entender o que o script está fazendo no seu computador. -- Create a file sveltekit-wails.sh -- Copy the below code into the new file then save it. -- Make it executable with `chmod +x sveltekit-wails.sh` -- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. -- Example usage: `./sveltekit-wails.sh pnpm newapp brand` +- Criar um arquivo sveltekit-wails.sh +- Copie o código abaixo para o novo arquivo e o salve. +- Torná-lo executável com `chmod +x sveltekit-wails.sh` +- A marca é um parâmetro opcional abaixo que adiciona de volta na marca de fregueses. Deixe o terceiro parâmetro em branco para não inserir a marca das Wails. +- Exemplo de uso: `./sveltekit-wails.sh pnpm newapp brand` ##### sveltekit-wails.sh: @@ -96,17 +96,17 @@ cd .. wails dev ``` -## 3. Important Notes +## 3. Notas importantes -##### Server files will cause build failures. +##### Os arquivos do servidor causarão falhas de construção. -- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. +- \+layout.server.ts, +page.server.ts, +server.ts ou qualquer arquivo com "server" em nome falhará na construção enquanto todas as rotas forem pré-renderizadas. -##### The Wails runtime unloads with full page navigations! +##### O tempo de execução do Wails descarrega com navegações de página inteira! -- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. -- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. -- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html +- Tudo que causa navegações de página completa: `window.location.href = '//'` ou recarga do menu de contexto ao usar wails dev. Isso significa que você pode acabar perdendo a capacidade de chamar qualquer falha no aplicativo em tempo de execução. Há duas formas de trabalhar em torno desta questão. +- Use `import { goto } from '$app/navigation'` e então chame `goto('//')` em sua + page.svelte. Isso impedirá uma navegação de página completa. +- Se a navegação por página inteira não puder ser impedido que o tempo de execução do Wails seja adicionado a todas as páginas, adicionando o abaixo ao `` de myapp/frontend/src/app. Mt ``` @@ -118,17 +118,17 @@ wails dev ``` -See https://wails.io/docs/guides/frontend for more information. +Veja https://wails.io/docs/guides/frontend para mais informações. -##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. +##### Os dados de e-mail podem ser carregados e atualizados a partir de +page.ts/+page.js para +page.svelte. -- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data -- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. +- \+page.ts/+page.js funciona bem com o load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() em +page.svelte irá chamar load() de +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. -##### Error Handling +##### Tratamento de erros -- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors -- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. +- Erros esperados usando o erro Throw funcionam em +page.ts/+page.js com uma página +error.svelte. https://kit.svelte.dev/docs/errors#expected-errors +- Erros inesperados farão com que o aplicativo se torne inutilizável. Somente a opção de recuperação (conhecida até agora) de erros inesperados é recarregar o aplicativo. Para fazer isso, crie um arquivo myapp/frontend/src/hooks.client.ts e adicione o código abaixo ao arquivo. ``` import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' @@ -137,10 +137,10 @@ export async function handleError() { } ``` -##### Using Forms and handling functions +##### Usando formas e funções de manipulação -- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `
` -- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: +- A maneira mais simples é chamar uma função do formulário é o padrão, vincular:valor suas variáveis e evitar submissão `` +- A maneira mais avançada é use:enhance (aprimoramento progressivo) que permitirá acesso conveniente a formData, formElemento, emissor. A nota importante é sempre cancel() o formulário que impede o comportamento do lado do servidor. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Exemplo: ``` { diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/templates.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/templates.mdx index 790e3107f04..2ae61796b73 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/templates.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/templates.mdx @@ -1,17 +1,17 @@ # Templates -Wails generates projects from pre-created templates. In v1, this was a difficult to maintain set of projects that were subject to going out of date. In v2, to empower the community, a couple of new features have been added for templates: +Wails gera projetos a partir de modelos pré-criados. Na v1, este era um conjunto de projetos que estavam sujeitos a sair de moda. No v2, para capacitar a comunidade, alguns novos recursos foram adicionados para os templates: -- Ability to generate projects from [Remote Templates](../reference/cli.mdx#remote-templates) -- Tooling to help create your own templates +- Capacidade de gerar projetos a partir de [Modelos Remotos](../reference/cli.mdx#remote-templates) +- Ferramentas para ajudar a criar seus próprios modelos -## Creating Templates +## Criando Templates -To create a template, you can use the `wails generate template` command. To generate a default template, run: +Para criar um template, você pode usar o comando `wails generate template`. Para gerar um modelo padrão, execute: `wails generate template -name mytemplate` -This creates the directory "mytemplate" with default files: +Isso cria o diretório "mytemplate" com os arquivos padrão: ```shell title=mytemplate/ . @@ -35,31 +35,31 @@ This creates the directory "mytemplate" with default files: `-- wails.tmpl.json ``` -### Template Overview +### Visão Geral do Modelo -The default template consists of the following files and directories: +O modelo padrão consiste nos seguintes arquivos e diretórios: -| Filename / Dir | Description | -| --------------- | -------------------------------------------- | -| NEXTSTEPS.md | Instructions on how to complete the template | -| README.md | The README published with the template | -| app.tmpl.go | `app.go` template file | -| frontend/ | The directory containing frontend assets | -| go.mod.tmpl | `go.mod` template file | -| main.tmpl.go | `main.go` template file | -| template.json | The template metadata | -| wails.tmpl.json | `wails.json` template file | +| Nome do arquivo / diretório | Descrição | +| --------------------------- | -------------------------------------------- | +| NEXTSTEPS.md | Instruções sobre como completar o modelo | +| README.md | O README publicado com o modelo | +| app.tmpl.go | Arquivo de modelo `app.go` | +| frontend/ | O diretório que contém os assets do frontend | +| go.mod.tmpl | Arquivo de modelo `go.mod` | +| main.tmpl.go | Arquivo de modelo `main.go` | +| template.json | Os metadados do modelo | +| wails.tmpl.json | Arquivo de modelo `wails.json` | -At this point it is advisable to follow the steps in `NEXTSTEPS.md`. +Neste ponto é aconselhável seguir os passos em `NEXTSTEPS.md`. -## Creating a Template from an Existing Project +## Criando um Template de um Projeto Existente -It's possible to create a template from an existing frontend project by passing the path to the project when generating the template. We will now walk through how to create a Vue 3 template: +É possível criar um modelo a partir de um projeto de frontend existente, passando o caminho para o projeto ao gerar o template. Vamos agora andar sobre como criar um modelo do Vue 3: -- Install the vue cli: `npm install -g @vue/cli` -- Create the default project: `vue create vue3-base` - - Select `Default (Vue 3) ([Vue 3] babel, eslint)` -- After the project has been generated, run: +- Instale o vue cli: `npm install -g @vue/cli` +- Crie o projeto padrão: `vue create vue3-base` + - Selecione `Padrão (Vue 3) ([Vue 3] babel, eslint)` +- Depois que o projeto for gerado, execute: ```shell > wails generate template -name wails-vue3-template -frontend .\vue3-base\ @@ -71,11 +71,11 @@ Updating package-lock.json data... Renaming package-lock.json -> package-lock.tmpl.json... ``` -- The template may now be customised as specified in the `NEXTSTEPS.md` file -- Once the files are ready, it can be tested by running: `wails init -n my-vue3-project -t .\wails-vue3-template\` -- To test the new project, run: `cd my-vue3-project` then `wails build` -- Once the project has compiled, run it: `.\build\bin\my-vue3-project.exe` -- You should have a fully functioning Vue3 application: +- O template agora pode ser personalizado conforme especificado no arquivo `NEXTSTEPS.md` +- Uma vez que os arquivos estão prontos, eles podem ser testados executando: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- Para testar o novo projeto, execute: `cd meu-vue3-projeto` e `wails constroem` +- Uma vez que o projeto tenha compilado, execute-o: `.\build\bin\my-vue3-project.exe` +- Você deve ter um aplicativo Vue3 que funcione plenamente: ```mdx-code-block
@@ -86,12 +86,12 @@ Renaming package-lock.json -> package-lock.tmpl.json...
``` -## Publishing Templates +## Publicando Templates -Publishing a template is simply pushing the files to GitHub. The following best practice is encouraged: +A publicação de um template está simplesmente enviando os arquivos para o GitHub. São encorajadas as seguintes melhores práticas: -- Remove any unwanted files and directories (such as `.git`) from your frontend directory -- Ensure that `template.json` is complete, especially `helpurl` -- Push the files to GitHub -- Create a PR on the [Community Templates](../community/templates.mdx) page -- Announce the template on the [Template Announcement](https://github.com/wailsapp/wails/discussions/825) discussion board +- Remova todos os arquivos e diretórios indesejados (como `.git`) do seu diretório no frontend +- Certifique-se de que `template.json` esteja completo, especialmente `helpurl` +- Faça push dos arquivos para o GitHub +- Crie um PR na página [Templates de Comunidade](../community/templates.mdx) +- Anuncie o modelo no fórum de discussão [de Anúncio de Modelo](https://github.com/wailsapp/wails/discussions/825) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx index 5b319f9a447..b4980716836 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -1,25 +1,25 @@ -# Troubleshooting +# Resolução de Problemas -An assortment of troubleshooting tips. +Uma variedade de dicas de solução de problemas. -## The `wails` command appears to be missing? +## O comando `wail` parece estar faltando? -If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment variable. You will also normally need to close and reopen any open command prompts so that changes to the environment made by the installer are reflected at the command prompt. +Se o sistema está relatando que o comando `wails` está faltando, verifique se você seguiu o guia de instalação do Go corretamente. Normalmente, isso significa que o diretório `go/bin` no diretório inicial do seu usuário não está na variável `PATH` ambiente. Você normalmente também precisará fechar e reabrir qualquer prompt de comando aberto para que as alterações no ambiente feitas pelo instalador sejam refletidas no prompt de comando. -## My application is displaying a white/blank screen +## Meu aplicativo está exibindo uma tela branca/em branco -Check that your application includes the assets from the correct directory. In your `main.go` file, you will have something similar to the following code: +Verifique se sua aplicação inclui os conteúdos do diretório correto. No seu arquivo `main.go`, você terá algo semelhante ao seguinte código: ```go //go:embed all:frontend/dist var assets embed.FS ``` -Check that `frontend/dist` contains your application assets. +Verifique que `frontend/dist` contém os ativos da aplicação. ### Mac -If this happens on Mac, try adding the following to your `Info.plist`: +Se isso acontecer no Mac, tente adicionar o seguinte ao seu `Info.plist`: ```xml NSAppTransportSecurity @@ -31,9 +31,9 @@ If this happens on Mac, try adding the following to your `Info.plist`: Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 -## Mac application not valid +## Aplicativo Mac inválido -If your built application looks like this in finder: +Se a sua aplicação construída se parece com isso no buscador: ```mdx-code-block

@@ -45,17 +45,17 @@ If your built application looks like this in finder:

``` -it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to the `build/darwin` directory. +é provável que o `info.plist` do seu aplicativo seja inválido. Atualize o arquivo em `build/.app/Contents/info.plist` e verifique se os dados são válidos, verifique se o nome binário está correto. Para persistir nas alterações, copie o arquivo de volta para o diretório `build/darwin`. -## My application is not displaying the correct icon in Windows Explorer +## Meu aplicativo não está exibindo o ícone correto no Windows Explorer -If your application is not displaying the correct icon, try deleting the hidden `IconCache.db` file located in the `C:\Users\<your username>\AppData\Local` directory. This will force Windows to rebuild the icon cache. +Se seu aplicativo não estiver exibindo o ícone correto, tente excluir o arquivo `IconCache.db` oculto localizado na pasta Diretório `C:\Users\<seu nome de usuário>\AppData\Local`. Isto irá forçar o Windows a reconstruir o cache de ícones. -Source: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 +Reference: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 -## Cannot call backend method from frontend with variadic arguments +## Não é possível chamar o método backend no frontend com argumentos variados -If you have a backend method defined with variadic parameters, eg: +Se você tem um método de backend definido com parâmetros variadicos, por exemplo: ```go func (a *App) TestFunc(msg string, args ...interface{}) error { @@ -63,7 +63,7 @@ func (a *App) TestFunc(msg string, args ...interface{}) error { } ``` -calling this method from the frontend like this will fail: +chamar esse método a partir do frontend como isso irá falhar: ```js var msg = "Hello: "; @@ -77,7 +77,7 @@ window.go.main.App.TestFunc(msg, ...args) }); ``` -Workaround: +Gambiarra: ```js var msg = "Hello "; @@ -94,30 +94,30 @@ window.go.main.App.TestFunc(msg, args) Credit: https://github.com/wailsapp/wails/issues/1186 -## I'm having getting proxy errors when trying to install Wails +## Estou recebendo erros de proxy ao tentar instalar o Wails -If you are getting errors like this: +Se você estiver recebendo erros como este: ``` "https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. ``` -it's probably because the official Go Proxy is being blocked (Users in China have reported this). The solution is to set up the proxy manually, eg: +é provavelmente porque o Proxy oficial Go está sendo bloqueado (usuários na China relataram isto). A solução é configurar o proxy manualmente, por exemplo: ``` go env -w GO111MODULE=on go env -w GOPROXY=https://goproxy.cn,direct ``` -Source: https://github.com/wailsapp/wails/issues/1233 +Reference: https://github.com/wailsapp/wails/issues/1233 -## The generated TypeScript doesn't have the correct types +## O TypeScript gerado não tem os tipos corretos -Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, it is possible to specify what types should be generated using the `ts_type` struct tag. For more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). +Às vezes, o TypeScript gerado não tem os tipos corretos. Para mitigar isso, é possível especificar quais tipos devem ser gerados usando a tag de struct `ts_type`. Para mais detalhes do, leia [isto](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). -## When I navigate away from `index.html`, I am unable to call methods on the frontend +## Quando navego longe do `index.html`, não consigo chamar métodos no frontend -If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding the following imports to the `` section of any new page you navigate to: +Se você navegar do `index.html` para um novo arquivo html, o contexto será perdido. Isso pode ser corrigido adicionando as seguintes importações para a seção `` de qualquer nova página que você navegar: ```html @@ -126,17 +126,17 @@ If you navigate away from `index.html` to a new html file, the context will be l ``` -Source: https://github.com/wailsapp/wails/discussions/1512 +Reference: https://github.com/wailsapp/wails/discussions/1512 -## I get `too many open files` errors on my Mac when I run `wails dev` +## Eu recebo `muitos arquivos abertos` erros no meu Mac quando eu rodo `wails` -By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. This limit can be increased by running: `ulimit -n 1024` in the terminal. +Por padrão, o macOS só permitirá que você abra um máximo de 256 arquivos. Isso pode afetar o comando `wails dev`. Este limite pode ser aumentado em execução: `ulimit -n 1024` no terminal. -FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). +FSNotify é [procurando mudar para os fsevents](https://github.com/fsnotify/fsnotify/issues/11) da Apple para Mac. Se isso não estiver concluído em breve, criaremos nossa própria implementação, monitorada [aqui](https://github.com/wailsapp/wails/issues/1733). -## My Mac app gives me weird compilation errors +## Meu aplicativo para Mac me dá erros estranhos de compilação -A few users have reported seeing compilation errors such as the following: +Alguns usuários relataram ver erros de compilação como os seguintes: ```shell # github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin @@ -149,31 +149,53 @@ In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/Sy #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) ``` -This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. +Isto é _normalmente_ devido a uma incompatibilidade com a versão do sistema operacional que você está executando e a versão das Ferramentas de Comando XCode instalada. Se você vir um erro como este, tente atualizar suas Ferramentas de Linha de Comando XCode para a versão mais recente. -If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: +Se reinstalar as Ferramentas de Comando Xcode ainda falhar, você pode verificar o caminho onde o kit de ferramentas está usando: `xcode-select -p` -If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` +Se `/Applications/Xcode.app/Contents/Developer` for exibido, rode `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Fontes: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +Se você estiver recebendo erros como este: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 -Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 -- -## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" +## Não foi possível iniciar o serviço: A versão do host "x.x.x não coincide com a versão binária "x.x.x" -It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine that may have different versions of Node installed, you may not be able to run your application. +É preferível adicionar `frontend/node_modules` e `frontend/package-lock.json` ao seu `.gitignore`. Caso contrário, ao abrir o repositório em outra máquina que pode ter diferentes versões do Node instaladas, talvez você não seja capaz de executar seu aplicativo. -If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. +Se isso acontecer, simplesmente exclua `frontend/node_modules` e `frontend/pacote-lock. soa` e corra os seus wails `constroem` ou `wails dev` comando. -## Build process stuck on "Generating bindings" +## Processo de compilação travado em "Gerando vinculações" -Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. +Processo de geração de Bindings executa sua aplicação em um modo especial. Se o aplicativo, intencionalmente ou não intencionalmente, contém um laço infinito (ou seja, não sair após `wails.Run()` terminado), isto pode levar a construção do processo travado na geração do palco de amarras. Por favor, certifique-se de que seu código sai corretamente. -## Mac application flashes white at startup +## Aplicação Mac pisca branco na inicialização -This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: +Isto é devido ao plano de fundo padrão do webview ser branco. Se você quiser usar a cor de fundo da janela, você pode tornar o plano de fundo do webview transparente usando a seguinte configuração: ```go err := wails.Run(&options.App{ @@ -185,4 +207,162 @@ This is due to the default background of the webview being white. If you want to WebviewIsTransparent: true, }, }) -``` \ No newline at end of file +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/cli.mdx index 6d22b657122..c2a5d706bac 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/cli.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -49,35 +49,35 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for `wails build` é usado para compilar seu projeto para um binário pronto para produção. -| Flag | Descrição | Padrão | -|:-------------------- |:---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:----------------------------------------------------------------------------------------------------------------------------------------------------- | -| -clean | Limpa o diretório `compilação/bin` | | -| -compiler "compiler" | Use um compilador de ida diferente para realizar build, por exemplo, go1.15beta1 | go | -| -debug | Retains debug information in the application and shows the debug console. Permite o uso das ferramentas devtools na janela do aplicativo | | -| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used) | | -| -dryrun | Prints the build command without executing it | | -| -f | Forçar compilação de aplicação | | -| -garbleargs | Argumentos para passar para o garble | `-literals -tiny -seed=random` | -| -ldflags "flags" | Ldflags adicionais para passar para o compilador | | -| -m | Skip mod tidy before compile | | -| -nopackage | Não empacotar aplicação | | -| -nocolour | Disable colour in output | | -| -nosyncgomod | Do not sync go.mod with the Wails version | | -| -nsis | Generate NSIS installer for Windows | | -| -o nome de arquivo | Nome do Arquivo de Saída | | -| -obfuscated | Ofuscar a aplicação usando [garble](https://github.com/burrowers/garble) | | -| -platform | Compila para as plataformas [(delimitadas por vírgula)](../reference/cli.mdx#platforms) por exemplo. `windows/arm64`. Note, se você não der arquitetura, `runtime.GOARCH` é usado. | platform = `variável de ambiente GOOS` se determinado `runtime.GOOS`.
arch = `GOARCH` variável de envrionment se for dado `runtime.GOARCH`. | -| -race | Realiza build com o Go race detector | | -| -s | Pular build do frontend | | -| -skipbindings | Skip bindings generation | | -| -tags "extra tags" | Compilar tags para passar para o compilador Go. Deve ser citado. Separados por espaço ou vírgula (mas não ambos) | | -| -trimpath | Remove todos os caminhos do sistema de arquivo do executável resultante. | | -| -u | Atualiza o `go.mod` do seu projeto para usar a mesma versão de Wails que o CLI | | -| -upx | Comprimir binário final usando "upx" | | -| -upxflags | Flags para passar para o upx | | -| -v int | Nível de verbosidade(0 - silencioso, 1 - padrão, 2 - verbose) | 1 | -| -webview2 | Estratégia de instalação WebView2: download,embed,browser,error | baixar | -| -windowsconsole | Manter a janela de console para builds do Windows | | +| Flag | Descrição | Padrão | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:----------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Limpa o diretório `compilação/bin` | | +| -compiler "compiler" | Use um compilador de ida diferente para realizar build, por exemplo, go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Permite o uso das ferramentas devtools na janela do aplicativo | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Forçar compilação de aplicação | | +| -garbleargs | Argumentos para passar para o garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Ldflags adicionais para passar para o compilador | | +| -m | Skip mod tidy before compile | | +| -nopackage | Não empacotar aplicação | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o nome de arquivo | Nome do Arquivo de Saída | | +| -obfuscated | Ofuscar a aplicação usando [garble](https://github.com/burrowers/garble) | | +| -platform | Compila para as plataformas [(delimitadas por vírgula)](../reference/cli.mdx#platforms) por exemplo. `windows/arm64`. Note, se você não der arquitetura, `runtime.GOARCH` é usado. | platform = `variável de ambiente GOOS` se determinado `runtime.GOOS`.
arch = `GOARCH` variável de envrionment se for dado `runtime.GOARCH`. | +| -race | Realiza build com o Go race detector | | +| -s | Pular build do frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Compilar tags para passar para o compilador Go. Deve ser citado. Separados por espaço ou vírgula (mas não ambos) | | +| -trimpath | Remove todos os caminhos do sistema de arquivo do executável resultante. | | +| -u | Atualiza o `go.mod` do seu projeto para usar a mesma versão de Wails que o CLI | | +| -upx | Comprimir binário final usando "upx" | | +| -upxflags | Flags para passar para o upx | | +| -v int | Nível de verbosidade(0 - silencioso, 1 - padrão, 2 - verbose) | 1 | +| -webview2 | Estratégia de instalação WebView2: download,embed,browser,error | baixar | +| -windowsconsole | Manter a janela de console para builds do Windows | | Para uma descrição detalhada do sinalizador `webview2`, consulte o guia [Windows](../guides/windows.mdx). @@ -166,8 +166,8 @@ Your system is ready for Wails development! - Um servidor web foi iniciado em `http://localhost:34115` que serve sua aplicação (não apenas frontend) sobre http. Isso permite que você use suas extensões de desenvolvimento de navegador favoritas - Todos os conteúdos do aplicativo são carregados do disco. Se forem alterados, o aplicativo irá recarregar automaticamente (não reconstruir). Todos os navegadores conectados também recarregarão - A JS module is generated that provides the following: - - JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting - - TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods - A second JS module is generated that provides a wrapper + TS declaration for the runtime - On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/menus.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/menus.mdx index ff9a2442281..a60919a04b3 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/menus.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/menus.mdx @@ -4,9 +4,9 @@ sidebar_position: 4 # Menus -It is possible to add an application menu to Wails projects. This is achieved by defining a [Menu](#menu) struct and setting it in the [`Menu`](../reference/options.mdx#menu) application config, or by calling the runtime method [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). +É possível adicionar um menu de aplicação aos projetos do Wails. Isso é conseguido definindo uma struct [Menu](#menu) e configurando-a no configuração do aplicativo [`Menu`](../reference/options.mdx#menu) ou chamando o método [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). -An example of how to create a menu: +Um exemplo de como criar um menu: ```go @@ -36,13 +36,13 @@ An example of how to create a menu: // ... ``` -It is also possible to dynamically update the menu, by updating the menu struct and calling [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). +Também é possível atualizar dinamicamente o menu, atualizando o menu struct e chamando [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). -The example above uses helper methods, however it's possible to build the menu structs manually. +O exemplo acima usa métodos de ajuda, no entanto, é possível construir as construções do menu manualmente. ## Menu -A Menu is a collection of MenuItems: +Um Menu é uma coleção de MenuItems: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" type Menu struct { @@ -50,19 +50,19 @@ type Menu struct { } ``` -For the Application menu, each MenuItem represents a single menu such as "Edit". +Para o menu de Aplicação, cada MenuItem representa um único menu como "Editar". -A simple helper method is provided for building menus: +Um método simples de ajuda é fornecido para menus de construção: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu ``` -This makes the layout of the code more like that of a menu without the need to add the menu items manually after creating them. Alternatively, you can just create the menu items and add them to the menu manually. +Isto torna o layout do código mais parecido com o de um menu sem a necessidade de adicionar os itens de menu manualmente depois de criá-los. Como alternativa, você pode apenas criar os itens de menu e adicioná-los ao menu manualmente. ## MenuItem -A MenuItem represents an item within a Menu. +Um MenuItem representa um item dentro de um Menu. ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" // MenuItem represents a menu item contained in a menu @@ -79,30 +79,30 @@ type MenuItem struct { } ``` -| Field | Type | Notes | -| ----------- | ------------------------------------ | ------------------------------------------------------------- | -| Label | string | The menu text | -| Accelerator | [\*keys.Accelerator](#accelerator) | Key binding for this menu item | -| Type | [Type](#type) | Type of MenuItem | -| Disabled | bool | Disables the menu item | -| Hidden | bool | Hides this menu item | -| Checked | bool | Adds check to item (Checkbox & Radio types) | -| SubMenu | [\*Menu](#menu) | Sets the submenu | -| Click | [Callback](#callback) | Callback function when menu clicked | -| Role | string | Defines a [role](#role) for this menu item. Mac only for now. | +| Campo | Tipo | Notas | +| ----------- | ------------------------------------ | -------------------------------------------------------------------------- | +| Label | string | O texto do menu | +| Accelerator | [\*keys.Accelerator](#accelerator) | Vinculação de teclas para este item de menu | +| Tipo | [Tipo](#type) | Tipo de MenuItem | +| Disabled | bool | Desativa o item de menu | +| Hidden | bool | Oculta este item de menu | +| Checked | bool | Adiciona uma seleção para o item ( & Tipos de Rádio) | +| SubMenu | [\*Menu](#menu) | Define o submenu | +| Click | [Callback](#callback) | Função Callback quando clicado no menu | +| Role | string | Define um papel [](#role) para este item de menu. Mac apenas por enquanto. | ### Accelerator -Accelerators (sometimes called keyboard shortcuts) define a binding between a keystroke and a menu item. Wails defines an Accelerator as a combination or key + [Modifier](#modifier). They are available in the `"github.com/wailsapp/wails/v2/pkg/menu/keys"` package. +Os aceleradores (às vezes chamados atalhos de teclado) definem uma ligação entre um toque de tecla e um item de menu. Lamentos define um Acelerador como uma combinação ou tecla + [Modificador](#modifier). Eles estão disponíveis no pacote `"github.com/wailsapp/wails/v2/pkg/menu/keys"`. -Example: +Exemplo: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" // Defines cmd+o on Mac and ctrl-o on Window/Linux myShortcut := keys.CmdOrCtrl("o") ``` -Keys are any single character on a keyboard with the exception of `+`, which is defined as `plus`. Some keys cannot be represented as characters so there are a set of named characters that may be used: +Teclas são qualquer caractere único em um teclado com exceção de `+`, que é definido como `plus`. Algumas chaves não podem ser representadas como caracteres, portanto há um conjunto de caracteres nomeados que podem ser usados: | | | | | |:-----------:|:-----:|:-----:|:---------:| @@ -122,18 +122,18 @@ Keys are any single character on a keyboard with the exception of `+`, which is | `page up` | `f14` | `f39` | | | `page down` | `f15` | `f30` | | -Wails also supports parsing accelerators using the same syntax as Electron. This is useful for storing accelerators in config files. +Wails também suportam a análise de aceleradores usando a mesma sintaxe que o Electron. Isso é útil para armazenar aceleradores em arquivos de configuração. -Example: +Exemplo: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" // Defines cmd+o on Mac and ctrl-o on Window/Linux myShortcut, err := keys.Parse("Ctrl+Option+A") ``` -#### Modifier +#### Modificador -The following modifiers are keys that may be used in combination with the accelerator key: +Os seguintes modificadores são chaves que podem ser usadas em combinação com a tecla de aceleração: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" const ( @@ -148,7 +148,7 @@ const ( ) ``` -A number of helper methods are available to create Accelerators using modifiers: +Vários métodos de ajuda estão disponíveis para criar aceleradores usando modificadores: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" func CmdOrCtrl(key string) *Accelerator @@ -157,16 +157,16 @@ func Shift(key string) *Accelerator func Control(key string) *Accelerator ``` -Modifiers can be combined using `keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)`: +Modificadores podem ser combinados usando `keys.Combo(string de chaves, modificador 1 modificador, modificador modificador, rest ...Modificador)`: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) ``` -### Type +### Tipo -Each menu item must have a type and there are 5 types available: +Cada item de menu deve ter um tipo e existem 5 tipos disponíveis: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" const ( @@ -178,7 +178,7 @@ const ( ) ``` -For convenience, helper methods are provided to quickly create a menu item: +Para conveniência, métodos auxiliares são fornecidos para criar rapidamente um item de menu: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem @@ -188,7 +188,7 @@ func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click C func SubMenu(label string, menu *Menu) *Menu ``` -You can also create menu items directly on a menu by using the "Add" helpers: +Você também pode criar itens de menu diretamente em um menu, usando os ajudantes "Adicionar": ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem @@ -198,11 +198,11 @@ func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Acceler func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI ``` -A note on radio groups: A radio group is defined as a number of radio menu items that are next to each other in the menu. This means that you do not need to group items together as it is automatic. However, that also means you cannot have 2 radio groups next to each other - there must be a non-radio item between them. +Uma nota nos grupos de rádio: Um grupo de rádio é definido como um número de itens do menu de rádio que estão próximos um ao outro no menu. Isso significa que não é necessário agrupar os itens porque é automático. No entanto, isso também significa que você não pode ter 2 grupos lado a lado - deve haver um item que não seja de rádio entre eles. ### Callback -Each menu item may have a callback that is executed when the item is clicked: +Cada item de menu pode ter um callback que é executado quando o item é clicado: ```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" type Callback func(*CallbackData) @@ -212,19 +212,19 @@ type CallbackData struct { } ``` -The function is given a `CallbackData` struct which indicates which menu item triggered the callback. This is useful when using radio groups that may share a callback. +A função recebe uma instrução `CallbackData` que indica qual item de menu acionou a callback. Isso é útil quando usar grupos de rádio que podem compartilhar um callback. ### Role -:::info Roles +:::info Regras -Roles are currently supported on Mac only. +As regras que são atualmente suportados apenas no Mac. ::: -A menu item may have a role, which is essentially a pre-defined menu item. We currently support the following roles: +Um item de menu pode ter uma função, que é essencialmente um item de menu pré-definido. Atualmente, apoiamos as seguintes funções: -| Role | Description | -| ------------ | ------------------------------------------------------------------------ | -| AppMenuRole | The standard Mac application menu. Can be created using `menu.AppMenu()` | -| EditMenuRole | The standard Mac edit menu. Can be created using `menu.EditMenu()` | +| Role | Descrição | +| ------------ | --------------------------------------------------------------------------- | +| AppMenuRole | Menu padrão do aplicativo para Mac. Pode ser criado usando `menu.AppMenu()` | +| EditMenuRole | O menu de edição padrão para Mac. Pode ser criado usando `menu.EditMenu()` | diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/options.mdx index e1bb970a127..46d1c071b60 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/options.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -366,7 +366,7 @@ Name: CSSDragValue
Type: `string` EnableDefaultContextMenu enables the browser's default context-menu in production. -By default, the browser's default context-menu is only available in development and in a `-debug` or `-devtools` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : @@ -593,6 +593,12 @@ Setting this to `true` will disable GPU hardware acceleration for the webview. Name: WebviewGpuIsDisabled
Type: `bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + ### Mac This defines [Mac specific options](#mac). @@ -688,6 +694,42 @@ Setting this to `true` will make the window background translucent. Often combin Name: WindowIsTranslucent
Type: `bool` +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### About This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.3.1.json b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.3.1.json index c793961dca3..3d8fbe05279 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.3.1.json +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.3.1.json @@ -4,35 +4,35 @@ "description": "The label for version v2.3.1" }, "sidebar.docs.category.Getting Started": { - "message": "Getting Started", + "message": "Guia de Introdução", "description": "The label for category Getting Started in sidebar docs" }, "sidebar.docs.category.Reference": { - "message": "Reference", + "message": "Referência", "description": "The label for category Reference in sidebar docs" }, "sidebar.docs.category.Runtime": { - "message": "Runtime", + "message": "Tempo de execução", "description": "The label for category Runtime in sidebar docs" }, "sidebar.docs.category.Community": { - "message": "Community", + "message": "Comunidade", "description": "The label for category Community in sidebar docs" }, "sidebar.docs.category.Showcase": { - "message": "Showcase", + "message": "Casos de sucesso", "description": "The label for category Showcase in sidebar docs" }, "sidebar.docs.category.Guides": { - "message": "Guides", + "message": "Guias", "description": "The label for category Guides in sidebar docs" }, "sidebar.docs.category.Tutorials": { - "message": "Tutorials", + "message": "Tutoriais", "description": "The label for category Tutorials in sidebar docs" }, "sidebar.docs.link.Contributing": { - "message": "Contributing", + "message": "Contribuir", "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" } } diff --git a/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx index ac2ac18bc68..7f4f545d3fb 100644 --- a/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx @@ -13,6 +13,28 @@ O formato é baseado em [Manter um Log de Alterações](https://keepachangelog.c ## [Unreleased] +### Adicionado + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### Alterado + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### Corrigido + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + ## v2.6.0 - 2023-09-06 ### Grandes Alterações @@ -27,6 +49,7 @@ O formato é baseado em [Manter um Log de Alterações](https://keepachangelog.c - Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) - Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) - Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) - Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) - Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) @@ -44,6 +67,7 @@ O formato é baseado em [Manter um Log de Alterações](https://keepachangelog.c - Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) - Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) - Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) ### Alterado @@ -680,9 +704,6 @@ Experimental: &options.Experimental{ - [v2, nsis] Seems like / as path separator works only for some directives in a cross platform way by [@stffabi](https://github.com/stffabi) in #1227 - import models on binding definition by [@adalessa](https://github.com/adalessa) in #123 - - 1 - - Use local search on website by [@leaanthony](https://github.com/leaanthony) in #1234 - Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 - Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/templates.mdx index 6dd14b917ae..0b86bb8d9bd 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -60,6 +60,10 @@ sidebar_position: 1 - [wails-elm-шаблон](https://github.com/benjamin-thomas/wails-elm-template) - Разработайте ваше GUI приложение с функциональным программированием и **snappy** настройками горячей перезагрузки :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Объединение возможностей :muscle: Elm + Tailwind CSS + Wails! Поддерживается горячая перезагрузка. +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + ## Чистый JavaScript (Vanilla) - [wails-pure-js-шаблон](https://github.com/KiddoV/wails-pure-js-template) - шаблон, содержащий только базовый JavaScript, HTML и CSS diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx index 5b319f9a447..1c3b6a9cc52 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -159,6 +159,28 @@ If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-se Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + -- ## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" @@ -185,4 +207,162 @@ This is due to the default background of the webview being white. If you want to WebviewIsTransparent: true, }, }) -``` \ No newline at end of file +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/cli.mdx index a4817110fc8..c7aea10a64d 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/cli.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -49,35 +49,35 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for `wails build` is used for compiling your project to a production-ready binary. -| Flag | Description | Default | -|:-------------------- |:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:--------------------------------------------------------------------------------------------------------------------------------------------------- | -| -clean | Cleans the `build/bin` directory | | -| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | -| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | -| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used) | | -| -dryrun | Prints the build command without executing it | | -| -f | Force build application | | -| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | -| -ldflags "flags" | Additional ldflags to pass to the compiler | | -| -m | Skip mod tidy before compile | | -| -nopackage | Do not package application | | -| -nocolour | Disable colour in output | | -| -nosyncgomod | Do not sync go.mod with the Wails version | | -| -nsis | Generate NSIS installer for Windows | | -| -o filename | Output filename | | -| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | -| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | -| -race | Build with Go's race detector | | -| -s | Skip building the frontend | | -| -skipbindings | Skip bindings generation | | -| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | -| -trimpath | Remove all file system paths from the resulting executable. | | -| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | -| -upx | Compress final binary using "upx" | | -| -upxflags | Flags to pass to upx | | -| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | -| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | -| -windowsconsole | Keep the console window for Windows builds | | +| Flag | Description | Default | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Cleans the `build/bin` directory | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Force build application | | +| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -m | Skip mod tidy before compile | | +| -nopackage | Do not package application | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o filename | Output filename | | +| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | +| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Build with Go's race detector | | +| -s | Skip building the frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | +| -trimpath | Remove all file system paths from the resulting executable. | | +| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | +| -upx | Compress final binary using "upx" | | +| -upxflags | Flags to pass to upx | | +| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | +| -windowsconsole | Keep the console window for Windows builds | | For a detailed description of the `webview2` flag, please refer to the [Windows](../guides/windows.mdx) Guide. @@ -166,8 +166,8 @@ Your system is ready for Wails development! - A webserver is started on `http://localhost:34115` which serves your application (not just frontend) over http. This allows you to use your favourite browser development extensions - All application assets are loaded from disk. If they are changed, the application will automatically reload (not rebuild). All connected browsers will also reload - A JS module is generated that provides the following: - - JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting - - TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods - A second JS module is generated that provides a wrapper + TS declaration for the runtime - On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/options.mdx index e1bb970a127..46d1c071b60 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/options.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -366,7 +366,7 @@ Name: CSSDragValue
Type: `string` EnableDefaultContextMenu enables the browser's default context-menu in production. -By default, the browser's default context-menu is only available in development and in a `-debug` or `-devtools` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : @@ -593,6 +593,12 @@ Setting this to `true` will disable GPU hardware acceleration for the webview. Name: WebviewGpuIsDisabled
Type: `bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + ### Mac This defines [Mac specific options](#mac). @@ -688,6 +694,42 @@ Setting this to `true` will make the window background translucent. Often combin Name: WindowIsTranslucent
Type: `bool` +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### About This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. diff --git a/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx index 3cda7c1c716..5828a733f3a 100644 --- a/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx @@ -13,6 +13,28 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), ## [Unreleased] +### Added + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### Changed + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### Fixed + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + ## v2.6.0 - 2023-09-06 ### Breaking Changes @@ -27,6 +49,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) - Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) - Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) - Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) - Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) @@ -44,6 +67,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) - Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) - Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) ### Changed @@ -680,9 +704,6 @@ Experimental: &options.Experimental{ - [v2, nsis] Seems like / as path separator works only for some directives in a cross platform way by [@stffabi](https://github.com/stffabi) in #1227 - import models on binding definition by [@adalessa](https://github.com/adalessa) in #123 - - 1 - - Use local search on website by [@leaanthony](https://github.com/leaanthony) in #1234 - Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 - Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 diff --git a/website/i18n/vi/code.json b/website/i18n/vi/code.json new file mode 100644 index 00000000000..c3b0bdfeb4b --- /dev/null +++ b/website/i18n/vi/code.json @@ -0,0 +1,431 @@ +{ + "homepage.Features.Title1": { + "message": "Feature Rich" + }, + "homepage.Features.Description1": { + "message": "Build comprehensive cross-platform applications using native UI elements such as menus and dialogs." + }, + "homepage.Features.Title2": { + "message": "Familiar" + }, + "homepage.Features.Description2": { + "message": "Use the technologies you already know to build amazing applications." + }, + "homepage.Features.Title3": { + "message": "Fast" + }, + "homepage.Features.Description3": { + "message": "Quickly generate, build and package your projects using the Wails CLI." + }, + "homepage.Tagline": { + "message": "Build beautiful cross-platform applications using Go" + }, + "homepage.ButtonText": { + "message": "Get Started" + }, + "homepage.LearnMoreButtonText": { + "message": "Learn More" + }, + "theme.ErrorPageContent.title": { + "message": "This page crashed.", + "description": "The title of the fallback page when the page crashed" + }, + "theme.ErrorPageContent.tryAgain": { + "message": "Try again", + "description": "The label of the button to try again rendering when the React error boundary captures an error" + }, + "theme.NotFound.title": { + "message": "Page Not Found", + "description": "The title of the 404 page" + }, + "theme.NotFound.p1": { + "message": "We could not find what you were looking for.", + "description": "The first paragraph of the 404 page" + }, + "theme.NotFound.p2": { + "message": "Please contact the owner of the site that linked you to the original URL and let them know their link is broken.", + "description": "The 2nd paragraph of the 404 page" + }, + "theme.AnnouncementBar.closeButtonAriaLabel": { + "message": "Close", + "description": "The ARIA label for close button of announcement bar" + }, + "theme.blog.archive.title": { + "message": "Archive", + "description": "The page & hero title of the blog archive page" + }, + "theme.blog.archive.description": { + "message": "Archive", + "description": "The page & hero description of the blog archive page" + }, + "theme.BackToTopButton.buttonAriaLabel": { + "message": "Scroll back to top", + "description": "The ARIA label for the back to top button" + }, + "theme.blog.paginator.navAriaLabel": { + "message": "Blog list page navigation", + "description": "The ARIA label for the blog pagination" + }, + "theme.blog.paginator.newerEntries": { + "message": "Newer Entries", + "description": "The label used to navigate to the newer blog posts page (previous page)" + }, + "theme.blog.paginator.olderEntries": { + "message": "Older Entries", + "description": "The label used to navigate to the older blog posts page (next page)" + }, + "theme.blog.post.readingTime.plurals": { + "message": "One min read|{readingTime} min read", + "description": "Pluralized label for \"{readingTime} min read\". Use as much plural forms (separated by \"|\") as your language support (see https://www.unicode.org/cldr/cldr-aux/charts/34/supplemental/language_plural_rules.html)" + }, + "theme.blog.post.readMoreLabel": { + "message": "Read more about {title}", + "description": "The ARIA label for the link to full blog posts from excerpts" + }, + "theme.blog.post.readMore": { + "message": "Read More", + "description": "The label used in blog post item excerpts to link to full blog posts" + }, + "theme.blog.post.paginator.navAriaLabel": { + "message": "Blog post page navigation", + "description": "The ARIA label for the blog posts pagination" + }, + "theme.blog.post.paginator.newerPost": { + "message": "Newer Post", + "description": "The blog post button label to navigate to the newer/previous post" + }, + "theme.blog.post.paginator.olderPost": { + "message": "Older Post", + "description": "The blog post button label to navigate to the older/next post" + }, + "theme.blog.sidebar.navAriaLabel": { + "message": "Blog recent posts navigation", + "description": "The ARIA label for recent posts in the blog sidebar" + }, + "theme.blog.post.plurals": { + "message": "One post|{count} posts", + "description": "Pluralized label for \"{count} posts\". Use as much plural forms (separated by \"|\") as your language support (see https://www.unicode.org/cldr/cldr-aux/charts/34/supplemental/language_plural_rules.html)" + }, + "theme.blog.tagTitle": { + "message": "{nPosts} tagged with \"{tagName}\"", + "description": "The title of the page for a blog tag" + }, + "theme.tags.tagsPageLink": { + "message": "View All Tags", + "description": "The label of the link targeting the tag list page" + }, + "theme.CodeBlock.copyButtonAriaLabel": { + "message": "Copy code to clipboard", + "description": "The ARIA label for copy code blocks button" + }, + "theme.CodeBlock.copied": { + "message": "Copied", + "description": "The copied button label on code blocks" + }, + "theme.CodeBlock.copy": { + "message": "Copy", + "description": "The copy button label on code blocks" + }, + "theme.colorToggle.ariaLabel": { + "message": "Switch between dark and light mode (currently {mode})", + "description": "The ARIA label for the navbar color mode toggle" + }, + "theme.colorToggle.ariaLabel.mode.dark": { + "message": "dark mode", + "description": "The name for the dark color mode" + }, + "theme.colorToggle.ariaLabel.mode.light": { + "message": "light mode", + "description": "The name for the light color mode" + }, + "theme.docs.DocCard.categoryDescription": { + "message": "{count} items", + "description": "The default description for a category card in the generated index about how many items this category includes" + }, + "theme.docs.sidebar.expandButtonTitle": { + "message": "Expand sidebar", + "description": "The ARIA label and title attribute for expand button of doc sidebar" + }, + "theme.docs.sidebar.expandButtonAriaLabel": { + "message": "Expand sidebar", + "description": "The ARIA label and title attribute for expand button of doc sidebar" + }, + "theme.docs.paginator.navAriaLabel": { + "message": "Docs pages navigation", + "description": "The ARIA label for the docs pagination" + }, + "theme.docs.paginator.previous": { + "message": "Previous", + "description": "The label used to navigate to the previous doc" + }, + "theme.docs.paginator.next": { + "message": "Next", + "description": "The label used to navigate to the next doc" + }, + "theme.docs.sidebar.collapseButtonTitle": { + "message": "Collapse sidebar", + "description": "The title attribute for collapse button of doc sidebar" + }, + "theme.docs.sidebar.collapseButtonAriaLabel": { + "message": "Collapse sidebar", + "description": "The title attribute for collapse button of doc sidebar" + }, + "theme.DocSidebarItem.toggleCollapsedCategoryAriaLabel": { + "message": "Toggle the collapsible sidebar category '{label}'", + "description": "The ARIA label to toggle the collapsible sidebar category" + }, + "theme.docs.tagDocListPageTitle.nDocsTagged": { + "message": "One doc tagged|{count} docs tagged", + "description": "Pluralized label for \"{count} docs tagged\". Use as much plural forms (separated by \"|\") as your language support (see https://www.unicode.org/cldr/cldr-aux/charts/34/supplemental/language_plural_rules.html)" + }, + "theme.docs.tagDocListPageTitle": { + "message": "{nDocsTagged} with \"{tagName}\"", + "description": "The title of the page for a docs tag" + }, + "theme.docs.versionBadge.label": { + "message": "Version: {versionLabel}" + }, + "theme.docs.versions.unreleasedVersionLabel": { + "message": "This is unreleased documentation for {siteTitle} {versionLabel} version.", + "description": "The label used to tell the user that he's browsing an unreleased doc version" + }, + "theme.docs.versions.unmaintainedVersionLabel": { + "message": "This is documentation for {siteTitle} {versionLabel}, which is no longer actively maintained.", + "description": "The label used to tell the user that he's browsing an unmaintained doc version" + }, + "theme.docs.versions.latestVersionSuggestionLabel": { + "message": "For up-to-date documentation, see the {latestVersionLink} ({versionLabel}).", + "description": "The label used to tell the user to check the latest version" + }, + "theme.docs.versions.latestVersionLinkLabel": { + "message": "latest version", + "description": "The label used for the latest version suggestion link label" + }, + "theme.common.editThisPage": { + "message": "Edit this page", + "description": "The link label to edit the current page" + }, + "theme.common.headingLinkTitle": { + "message": "Direct link to heading", + "description": "Title for link to heading" + }, + "theme.lastUpdated.atDate": { + "message": " on {date}", + "description": "The words used to describe on which date a page has been last updated" + }, + "theme.lastUpdated.byUser": { + "message": " by {user}", + "description": "The words used to describe by who the page has been last updated" + }, + "theme.lastUpdated.lastUpdatedAtBy": { + "message": "Last updated{atDate}{byUser}", + "description": "The sentence used to display when a page has been last updated, and by who" + }, + "theme.navbar.mobileSidebarSecondaryMenu.backButtonLabel": { + "message": "← Back to main menu", + "description": "The label of the back button to return to main menu, inside the mobile navbar sidebar secondary menu (notably used to display the docs sidebar)" + }, + "theme.navbar.mobileVersionsDropdown.label": { + "message": "Versions", + "description": "The label for the navbar versions dropdown on mobile view" + }, + "theme.common.skipToMainContent": { + "message": "Skip to main content", + "description": "The skip to content label used for accessibility, allowing to rapidly navigate to main content with keyboard tab/enter navigation" + }, + "theme.tags.tagsListLabel": { + "message": "Tags:", + "description": "The label alongside a tag list" + }, + "theme.TOCCollapsible.toggleButtonLabel": { + "message": "On this page", + "description": "The label used by the button on the collapsible TOC component" + }, + "theme.navbar.mobileLanguageDropdown.label": { + "message": "Languages", + "description": "The label for the mobile language switcher dropdown" + }, + "theme.SearchBar.seeAll": { + "message": "See all {count} results" + }, + "theme.SearchBar.label": { + "message": "Search", + "description": "The ARIA label and placeholder for search button" + }, + "theme.SearchPage.documentsFound.plurals": { + "message": "One document found|{count} documents found", + "description": "Pluralized label for \"{count} documents found\". Use as much plural forms (separated by \"|\") as your language support (see https://www.unicode.org/cldr/cldr-aux/charts/34/supplemental/language_plural_rules.html)" + }, + "theme.SearchPage.existingResultsTitle": { + "message": "Search results for \"{query}\"", + "description": "The search page title for non-empty query" + }, + "theme.SearchPage.emptyResultsTitle": { + "message": "Search the documentation", + "description": "The search page title for empty query" + }, + "theme.SearchPage.inputPlaceholder": { + "message": "Type your search here", + "description": "The placeholder for search page input" + }, + "theme.SearchPage.inputLabel": { + "message": "Search", + "description": "The ARIA label for search page input" + }, + "theme.SearchPage.algoliaLabel": { + "message": "Search by Algolia", + "description": "The ARIA label for Algolia mention" + }, + "theme.SearchPage.noResultsText": { + "message": "No results were found", + "description": "The paragraph for empty search result" + }, + "theme.SearchPage.fetchingNewResults": { + "message": "Fetching new results...", + "description": "The paragraph for fetching new search results" + }, + "theme.tags.tagsPageTitle": { + "message": "Tags", + "description": "The title of the tag list page" + }, + "theme.docs.breadcrumbs.home": { + "message": "Home page", + "description": "The ARIA label for the home page in the breadcrumbs" + }, + "theme.docs.breadcrumbs.navAriaLabel": { + "message": "Breadcrumbs", + "description": "The ARIA label for the breadcrumbs" + }, + "theme.CodeBlock.wordWrapToggle": { + "message": "Toggle word wrap", + "description": "The title attribute for toggle word wrapping button of code block lines" + }, + "theme.admonition.note": { + "message": "note", + "description": "The default label used for the Note admonition (:::note)" + }, + "theme.admonition.tip": { + "message": "tip", + "description": "The default label used for the Tip admonition (:::tip)" + }, + "theme.admonition.danger": { + "message": "danger", + "description": "The default label used for the Danger admonition (:::danger)" + }, + "theme.admonition.info": { + "message": "info", + "description": "The default label used for the Info admonition (:::info)" + }, + "theme.admonition.caution": { + "message": "caution", + "description": "The default label used for the Caution admonition (:::caution)" + }, + "theme.SearchModal.searchBox.resetButtonTitle": { + "message": "Clear the query", + "description": "The label and ARIA label for search box reset button" + }, + "theme.SearchModal.searchBox.cancelButtonText": { + "message": "Cancel", + "description": "The label and ARIA label for search box cancel button" + }, + "theme.SearchModal.startScreen.recentSearchesTitle": { + "message": "Recent", + "description": "The title for recent searches" + }, + "theme.SearchModal.startScreen.noRecentSearchesText": { + "message": "No recent searches", + "description": "The text when no recent searches" + }, + "theme.SearchModal.startScreen.saveRecentSearchButtonTitle": { + "message": "Save this search", + "description": "The label for save recent search button" + }, + "theme.SearchModal.startScreen.removeRecentSearchButtonTitle": { + "message": "Remove this search from history", + "description": "The label for remove recent search button" + }, + "theme.SearchModal.startScreen.favoriteSearchesTitle": { + "message": "Favorite", + "description": "The title for favorite searches" + }, + "theme.SearchModal.startScreen.removeFavoriteSearchButtonTitle": { + "message": "Remove this search from favorites", + "description": "The label for remove favorite search button" + }, + "theme.SearchModal.errorScreen.titleText": { + "message": "Unable to fetch results", + "description": "The title for error screen of search modal" + }, + "theme.SearchModal.errorScreen.helpText": { + "message": "You might want to check your network connection.", + "description": "The help text for error screen of search modal" + }, + "theme.SearchModal.footer.selectText": { + "message": "to select", + "description": "The explanatory text of the action for the enter key" + }, + "theme.SearchModal.footer.selectKeyAriaLabel": { + "message": "Enter key", + "description": "The ARIA label for the Enter key button that makes the selection" + }, + "theme.SearchModal.footer.navigateText": { + "message": "to navigate", + "description": "The explanatory text of the action for the Arrow up and Arrow down key" + }, + "theme.SearchModal.footer.navigateUpKeyAriaLabel": { + "message": "Arrow up", + "description": "The ARIA label for the Arrow up key button that makes the navigation" + }, + "theme.SearchModal.footer.navigateDownKeyAriaLabel": { + "message": "Arrow down", + "description": "The ARIA label for the Arrow down key button that makes the navigation" + }, + "theme.SearchModal.footer.closeText": { + "message": "to close", + "description": "The explanatory text of the action for Escape key" + }, + "theme.SearchModal.footer.closeKeyAriaLabel": { + "message": "Escape key", + "description": "The ARIA label for the Escape key button that close the modal" + }, + "theme.SearchModal.footer.searchByText": { + "message": "Search by", + "description": "The text explain that the search is making by Algolia" + }, + "theme.SearchModal.noResultsScreen.noResultsText": { + "message": "No results for", + "description": "The text explains that there are no results for the following search" + }, + "theme.SearchModal.noResultsScreen.suggestedQueryText": { + "message": "Try searching for", + "description": "The text for the suggested query when no results are found for the following search" + }, + "theme.SearchModal.noResultsScreen.reportMissingResultsText": { + "message": "Believe this query should return results?", + "description": "The text for the question where the user thinks there are missing results" + }, + "theme.SearchModal.noResultsScreen.reportMissingResultsLinkText": { + "message": "Let us know.", + "description": "The text for the link to report missing results" + }, + "theme.SearchModal.placeholder": { + "message": "Search docs", + "description": "The placeholder of the input of the DocSearch pop-up modal" + }, + "theme.docs.sidebar.closeSidebarButtonAriaLabel": { + "message": "Close navigation bar", + "description": "The ARIA label for close button of mobile sidebar" + }, + "theme.docs.sidebar.toggleSidebarButtonAriaLabel": { + "message": "Toggle navigation bar", + "description": "The ARIA label for hamburger menu button of mobile navigation" + }, + "theme.NavBar.navAriaLabel": { + "message": "Main", + "description": "The ARIA label for the main navigation" + }, + "theme.docs.sidebar.navAriaLabel": { + "message": "Docs sidebar", + "description": "The ARIA label for the sidebar navigation" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx b/website/i18n/vi/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx new file mode 100644 index 00000000000..d01b542b153 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/2021-09-27-v2-beta1-release-notes.mdx @@ -0,0 +1,161 @@ +--- +slug: wails-v2-beta-for-windows +title: Wails v2 Beta for Windows +authors: + - leaanthony +tags: + - wails + - v2 +--- + +```mdx-code-block +
+ +
+
+``` + +When I first announced Wails on Reddit, just over 2 years ago from a train in Sydney, I did not expect it to get much attention. A few days later, a prolific tech vlogger released a tutorial video, gave it a positive review and from that point on, interest in the project has skyrocketed. + +It was clear that people were excited about adding web frontends to their Go projects, and almost immediately pushed the project beyond the proof of concept that I had created. At the time, Wails used the [webview](https://github.com/webview/webview) project to handle the frontend, and the only option for Windows was the IE11 renderer. Many bug reports were rooted in this limitation: poor JavaScript/CSS support and no dev tools to debug it. This was a frustrating development experience but there wasn't much that could have been done to rectify it. + +For a long time, I'd firmly believed that Microsoft would eventually have to sort out their browser situation. The world was moving on, frontend development was booming and IE wasn't cutting it. When Microsoft announced the move to using Chromium as the basis for their new browser direction, I knew it was only a matter of time until Wails could use it, and move the Windows developer experience to the next level. + +Today, I am pleased to announce: **Wails v2 Beta for Windows**! There's a huge amount to unpack in this release, so grab a drink, take a seat and we'll begin... + +### No CGO Dependency! + +No, I'm not joking: _No_ _CGO_ _dependency_ 🤯! The thing about Windows is that, unlike MacOS and Linux, it doesn't come with a default compiler. In addition, CGO requires a mingw compiler and there's a ton of different installation options. Removing the CGO requirement has massively simplified setup, as well as making debugging an awful lot easier. Whilst I have put a fair bit of effort in getting this working, the majority of the credit should go to [John Chadwick](https://github.com/jchv) for not only starting a couple of projects to make this possible, but also being open to someone taking those projects and building on them. Credit also to [Tad Vizbaras](https://github.com/tadvi) whose [winc](https://github.com/tadvi/winc) project started me down this path. + +### WebView2 Chromium Renderer + +```mdx-code-block +
+ +
+
+``` + +Finally, Windows developers get a first class rendering engine for their applications! Gone are the days of contorting your frontend code to work on Windows. On top of that, you get a first-class developer tools experience! + +The WebView2 component does, however, have a requirement to have the `WebView2Loader.dll` sitting alongside the binary. This makes distribution just that little bit more painful than we gophers are used to. All solutions and libraries (that I know of) that use WebView2 have this dependency. + +However, I'm really excited to announce that Wails applications _have no such requirement_! Thanks to the wizardry of [John Chadwick](https://github.com/jchv), we are able to bundle this dll inside the binary and get Windows to load it as if it were present on disk. + +Gophers rejoice! The single binary dream lives on! + +### New Features + +```mdx-code-block +
+ +
+
+``` + +There were a lot of requests for native menu support. Wails has finally got you covered. Application menus are now available and include support for most native menu features. This includes standard menu items, checkboxes, radio groups, submenus and separators. + +There were a huge number of requests in v1 for the ability to have greater control of the window itself. I'm happy to announce that there's new runtime APIs specifically for this. It's feature-rich and supports multi-monitor configurations. There is also an improved dialogs API: Now, you can have modern, native dialogs with rich configuration to cater for all your dialog needs. + +There is now the option to generate IDE configuration along with your project. This means that if you open your project in a supported IDE, it will already be configured for building and debugging your application. Currently VSCode is supported but we hope to support other IDEs such as Goland soon. + +```mdx-code-block +
+ +
+
+``` + +### No requirement to bundle assets + +A huge pain-point of v1 was the need to condense your entire application down to single JS & CSS files. I'm happy to announce that for v2, there is no requirement to bundle assets, in any way, shape or form. Want to load a local image? Use an `` tag with a local src path. Want to use a cool font? Copy it in and add the path to it in your CSS. + +> Wow, that sounds like a webserver... + +Yes, it works just like a webserver, except it isn't. + +> So how do I include my assets? + +You just pass a single `embed.FS` that contains all your assets into your application configuration. They don't even need to be in the top directory - Wails will just work it out for you. + +### New Development Experience + +```mdx-code-block +
+ +
+
+``` + +Now that assets don't need to be bundled, it's enabled a whole new development experience. The new `wails dev` command will build and run your application, but instead of using the assets in the `embed.FS`, it loads them directly from disk. + +It also provides the additional features: + +- Hot reload - Any changes to frontend assets will trigger and auto reload of the application frontend +- Auto rebuild - Any changes to your Go code will rebuild and relaunch your application + +In addition to this, a webserver will start on port 34115. This will serve your application to any browser that connects to it. All connected web browsers will respond to system events like hot reload on asset change. + +In Go, we are used to dealing with structs in our applications. It's often useful to send structs to our frontend and use them as state in our application. In v1, this was a very manual process and a bit of a burden on the developer. I'm happy to announce that in v2, any application run in dev mode will automatically generate TypeScript models for all structs that are input or output parameters to bound methods. This enables seamless interchange of data models between the two worlds. + +In addition to this, another JS module is dynamically generated wrapping all your bound methods. This provides JSDoc for your methods, providing code completion and hinting in your IDE. It's really cool when you get data models auto-imported when hitting tab in an auto-generated module wrapping your Go code! + +### Remote Templates + +```mdx-code-block +
+ +
+
+``` + +Getting an application up and running quickly was always a key goal for the Wails project. When we launched, we tried to cover a lot of the modern frameworks at the time: react, vue and angular. The world of frontend development is very opinionated, fast moving and hard to keep on top of! As a result, we found our base templates getting out of date pretty quickly and this caused a maintenance headache. It also meant that we didn't have cool modern templates for the latest and greatest tech stacks. + +With v2, I wanted to empower the community by giving you the ability to create and host templates yourselves, rather than rely on the Wails project. So now you can create projects using community supported templates! I hope this will inspire developers to create a vibrant ecosystem of project templates. I'm really quite excited about what our developer community can create! + +### In Conclusion + +Wails v2 represents a new foundation for the project. The aim of this release is to get feedback on the new approach, and to iron out any bugs before a full release. Your input would be most welcome. Please direct any feedback to the [v2 Beta](https://github.com/wailsapp/wails/discussions/828) discussion board. + +There were many twists and turns, pivots and u-turns to get to this point. This was due partly to early technical decisions that needed changing, and partly because some core problems we had spent time building workarounds for were fixed upstream: Go’s embed feature is a good example. Fortunately, everything came together at the right time, and today we have the very best solution that we can have. I believe the wait has been worth it - this would not have been possible even 2 months ago. + +I also need to give a huge thank you :pray: to the following people because without them, this release just wouldn't exist: + +- [Misite Bao](https://github.com/misitebao) - An absolute workhorse on the Chinese translations and an incredible bug finder. +- [John Chadwick](https://github.com/jchv) - His amazing work on [go-webview2](https://github.com/jchv/go-webview2) and [go-winloader](https://github.com/jchv/go-winloader) have made the Windows version we have today possible. +- [Tad Vizbaras](https://github.com/tadvi) - Experimenting with his [winc](https://github.com/tadvi/winc) project was the first step down the path to a pure Go Wails. +- [Mat Ryer](https://github.com/matryer) - His support, encouragement and feedback has really helped drive the project forward. + +And finally, I'd like to give a special thank you to all the [project sponsors](/credits#sponsors), including [JetBrains](https://www.jetbrains.com?from=Wails), whose support drive the project in many ways behind the scenes. + +I look forward to seeing what people build with Wails in this next exciting phase of the project! + +Lea. + +PS: MacOS and Linux users need not feel left out - porting to this new foundation is actively under way and most of the hard work has already been done. Hang in there! + +PPS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx b/website/i18n/vi/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx new file mode 100644 index 00000000000..86d44616f40 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/2021-11-08-v2-beta2-release-notes.mdx @@ -0,0 +1,170 @@ +--- +slug: wails-v2-beta-for-mac +title: Wails v2 Beta for MacOS +authors: + - leaanthony +tags: + - wails + - v2 +--- + +```mdx-code-block +
+ +
+
+``` + +Today marks the first beta release of Wails v2 for Mac! It's taken quite a while to get to this point and I'm hoping that today's release will give you something that's reasonably useful. There have been a number of twists and turns to get to this point and I'm hoping, with your help, to iron out the crinkles and get the Mac port polished for the final v2 release. + +You mean this isn't ready for production? For your use case, it may well be ready, but there are still a number of known issues so keep your eye on [this project board](https://github.com/wailsapp/wails/projects/7) and if you would like to contribute, you'd be very welcome! + +So what's new for Wails v2 for Mac vs v1? Hint: It's pretty similar to the Windows Beta :wink: + +### New Features + +```mdx-code-block +
+ +
+
+``` + +There were a lot of requests for native menu support. Wails has finally got you covered. Application menus are now available and include support for most native menu features. This includes standard menu items, checkboxes, radio groups, submenus and separators. + +There were a huge number of requests in v1 for the ability to have greater control of the window itself. I'm happy to announce that there's new runtime APIs specifically for this. It's feature-rich and supports multi-monitor configurations. There is also an improved dialogs API: Now, you can have modern, native dialogs with rich configuration to cater for all your dialog needs. + +### Mac Specific Options + +In addition to the normal application options, Wails v2 for Mac also brings some Mac extras: + +- Make your window all funky and translucent, like all the pretty swift apps! +- Highly customisable titlebar +- We support the NSAppearance options for the application +- Simple config to auto-create an "About" menu + +### No requirement to bundle assets + +A huge pain-point of v1 was the need to condense your entire application down to single JS & CSS files. I'm happy to announce that for v2, there is no requirement to bundle assets, in any way, shape or form. Want to load a local image? Use an `` tag with a local src path. Want to use a cool font? Copy it in and add the path to it in your CSS. + +> Wow, that sounds like a webserver... + +Yes, it works just like a webserver, except it isn't. + +> So how do I include my assets? + +You just pass a single `embed.FS` that contains all your assets into your application configuration. They don't even need to be in the top directory - Wails will just work it out for you. + +### New Development Experience + +Now that assets don't need to be bundled, it's enabled a whole new development experience. The new `wails dev` command will build and run your application, but instead of using the assets in the `embed.FS`, it loads them directly from disk. + +It also provides the additional features: + +- Hot reload - Any changes to frontend assets will trigger and auto reload of the application frontend +- Auto rebuild - Any changes to your Go code will rebuild and relaunch your application + +In addition to this, a webserver will start on port 34115. This will serve your application to any browser that connects to it. All connected web browsers will respond to system events like hot reload on asset change. + +In Go, we are used to dealing with structs in our applications. It's often useful to send structs to our frontend and use them as state in our application. In v1, this was a very manual process and a bit of a burden on the developer. I'm happy to announce that in v2, any application run in dev mode will automatically generate TypeScript models for all structs that are input or output parameters to bound methods. This enables seamless interchange of data models between the two worlds. + +In addition to this, another JS module is dynamically generated wrapping all your bound methods. This provides JSDoc for your methods, providing code completion and hinting in your IDE. It's really cool when you get data models auto-imported when hitting tab in an auto-generated module wrapping your Go code! + +### Remote Templates + +```mdx-code-block +
+ +
+
+``` + +Getting an application up and running quickly was always a key goal for the Wails project. When we launched, we tried to cover a lot of the modern frameworks at the time: react, vue and angular. The world of frontend development is very opinionated, fast moving and hard to keep on top of! As a result, we found our base templates getting out of date pretty quickly and this caused a maintenance headache. It also meant that we didn't have cool modern templates for the latest and greatest tech stacks. + +With v2, I wanted to empower the community by giving you the ability to create and host templates yourselves, rather than rely on the Wails project. So now you can create projects using community supported templates! I hope this will inspire developers to create a vibrant ecosystem of project templates. I'm really quite excited about what our developer community can create! + +### Native M1 Support + +Thanks to the amazing support of [Mat Ryer](https://github.com/matryer/), the Wails project now supports M1 native builds: + +```mdx-code-block +
+ +
+
+``` + +You can also specify `darwin/amd64` as a target too: + +```mdx-code-block +
+ +
+
+``` + +Oh, I almost forgot.... you can also do `darwin/universal`.... :wink: + +```mdx-code-block +
+ +
+
+``` + +### Cross Compilation to Windows + +Because Wails v2 for Windows is pure Go, you can target Windows builds without docker. + +```mdx-code-block +
+ +
+
+``` + +### WKWebView Renderer + +V1 relied on a (now deprecated) WebView component. V2 uses the most recent WKWebKit component so expect the latest and greatest from Apple. + +### In Conclusion + +As I'd said in the Windows release notes, Wails v2 represents a new foundation for the project. The aim of this release is to get feedback on the new approach, and to iron out any bugs before a full release. Your input would be most welcome! Please direct any feedback to the [v2 Beta](https://github.com/wailsapp/wails/discussions/828) discussion board. + +And finally, I'd like to give a special thank you to all the [project sponsors](/credits#sponsors), including [JetBrains](https://www.jetbrains.com?from=Wails), whose support drive the project in many ways behind the scenes. + +I look forward to seeing what people build with Wails in this next exciting phase of the project! + +Lea. + +PS: Linux users, you're next! + +PPS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx b/website/i18n/vi/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx new file mode 100644 index 00000000000..b405953cfbc --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/2022-02-22-v2-beta3-release-notes.mdx @@ -0,0 +1,114 @@ +--- +slug: wails-v2-beta-for-linux +title: Wails v2 Beta for Linux +authors: + - leaanthony +tags: + - wails + - v2 +--- + +```mdx-code-block +
+ +
+
+``` + +I'm pleased to finally announce that Wails v2 is now in beta for Linux! It is somewhat ironic that the very first experiments with v2 was on Linux and yet it has ended up as the last release. That being said, the v2 we have today is very different from those first experiments. So without further ado, let's go over the new features: + +### New Features + +```mdx-code-block +
+ +
+
+``` + +There were a lot of requests for native menu support. Wails has finally got you covered. Application menus are now available and include support for most native menu features. This includes standard menu items, checkboxes, radio groups, submenus and separators. + +There were a huge number of requests in v1 for the ability to have greater control of the window itself. I'm happy to announce that there's new runtime APIs specifically for this. It's feature-rich and supports multi-monitor configurations. There is also an improved dialogs API: Now, you can have modern, native dialogs with rich configuration to cater for all your dialog needs. + +### No requirement to bundle assets + +A huge pain-point of v1 was the need to condense your entire application down to single JS & CSS files. I'm happy to announce that for v2, there is no requirement to bundle assets, in any way, shape or form. Want to load a local image? Use an `` tag with a local src path. Want to use a cool font? Copy it in and add the path to it in your CSS. + +> Wow, that sounds like a webserver... + +Yes, it works just like a webserver, except it isn't. + +> So how do I include my assets? + +You just pass a single `embed.FS` that contains all your assets into your application configuration. They don't even need to be in the top directory - Wails will just work it out for you. + +### New Development Experience + +Now that assets don't need to be bundled, it's enabled a whole new development experience. The new `wails dev` command will build and run your application, but instead of using the assets in the `embed.FS`, it loads them directly from disk. + +It also provides the additional features: + +- Hot reload - Any changes to frontend assets will trigger and auto reload of the application frontend +- Auto rebuild - Any changes to your Go code will rebuild and relaunch your application + +In addition to this, a webserver will start on port 34115. This will serve your application to any browser that connects to it. All connected web browsers will respond to system events like hot reload on asset change. + +In Go, we are used to dealing with structs in our applications. It's often useful to send structs to our frontend and use them as state in our application. In v1, this was a very manual process and a bit of a burden on the developer. I'm happy to announce that in v2, any application run in dev mode will automatically generate TypeScript models for all structs that are input or output parameters to bound methods. This enables seamless interchange of data models between the two worlds. + +In addition to this, another JS module is dynamically generated wrapping all your bound methods. This provides JSDoc for your methods, providing code completion and hinting in your IDE. It's really cool when you get data models auto-imported when hitting tab in an auto-generated module wrapping your Go code! + +### Remote Templates + +```mdx-code-block +
+ +
+
+``` + +Getting an application up and running quickly was always a key goal for the Wails project. When we launched, we tried to cover a lot of the modern frameworks at the time: react, vue and angular. The world of frontend development is very opinionated, fast moving and hard to keep on top of! As a result, we found our base templates getting out of date pretty quickly and this caused a maintenance headache. It also meant that we didn't have cool modern templates for the latest and greatest tech stacks. + +With v2, I wanted to empower the community by giving you the ability to create and host templates yourselves, rather than rely on the Wails project. So now you can create projects using community supported templates! I hope this will inspire developers to create a vibrant ecosystem of project templates. I'm really quite excited about what our developer community can create! + +### Cross Compilation to Windows + +Because Wails v2 for Windows is pure Go, you can target Windows builds without docker. + +```mdx-code-block +
+ +
+
+``` + +### In Conclusion + +As I'd said in the Windows release notes, Wails v2 represents a new foundation for the project. The aim of this release is to get feedback on the new approach, and to iron out any bugs before a full release. Your input would be most welcome! Please direct any feedback to the [v2 Beta](https://github.com/wailsapp/wails/discussions/828) discussion board. + +Linux is **hard** to support. We expect there to be a number of quirks with the beta. Please help us to help you by filing detailed bug reports! + +Finally, I'd like to give a special thank you to all the [project sponsors](/credits#sponsors) whose support drive the project in many ways behind the scenes. + +I look forward to seeing what people build with Wails in this next exciting phase of the project! + +Lea. + +PS: The v2 release isn't far off now! + +PPS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx b/website/i18n/vi/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx new file mode 100644 index 00000000000..c321f5042d7 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/2022-09-22-v2-release-notes.mdx @@ -0,0 +1,98 @@ +--- +slug: wails-v2-released +title: Wails v2 Released +authors: + - leaanthony +tags: + - wails + - v2 +--- + +```mdx-code-block +
+ +
+
+``` + +# It's here! + +Today marks the release of [Wails](https://wails.io) v2. It's been about 18 months since the first v2 alpha and about a year from the first beta release. I'm truly grateful to everyone involved in the evolution of the project. + +Part of the reason it took that long was due to wanting to get to some definition of completeness before officially calling it v2. The truth is, there's never a perfect time to tag a release - there's always outstanding issues or "just one more" feature to squeeze in. What tagging an imperfect major release does do, however, is to provide a bit of stability for users of the project, as well as a bit of a reset for the developers. + +This release is more than I'd ever expected it to be. I hope it gives you as much pleasure as it has given us to develop it. + +# What _is_ Wails? + +If you are unfamiliar with Wails, it is a project that enables Go programmers to provide rich frontends for their Go programs using familiar web technologies. It's a lightweight, Go alternative to Electron. Much more information can be found on the [official site](https://wails.io/docs/introduction). + +# What's new? + +The v2 release is a huge leap forward for the project, addressing many of the pain points of v1. If you have not read any of the blog posts on the Beta releases for [macOS](/blog/wails-v2-beta-for-mac), [Windows](/blog/wails-v2-beta-for-windows) or [Linux](/blog/wails-v2-beta-for-linux), then I encourage you to do so as it covers all the major changes in more detail. In summary: + +- Webview2 component for Windows that supports modern web standards and debugging capabilities. +- [Dark / Light theme](/docs/reference/options#theme) + [custom theming](/docs/reference/options#customtheme) on Windows. +- Windows now has no CGO requirements. +- Out-of-the-box support for Svelte, Vue, React, Preact, Lit & Vanilla project templates. +- [Vite](https://vitejs.dev/) integration providing a hot-reload development environment for your application. +- Native application [menus](/docs/guides/application-development#application-menu) and [dialogs](/docs/reference/runtime/dialog). +- Native window translucency effects for [Windows](/docs/reference/options#windowistranslucent) and [macOS](/docs/reference/options#windowistranslucent-1). Support for Mica & Acrylic backdrops. +- Easily generate an [NSIS installer](/docs/guides/windows-installer) for Windows deployments. +- A rich [runtime library](/docs/reference/runtime/intro) providing utility methods for window manipulation, eventing, dialogs, menus and logging. +- Support for [obfuscating](/docs/guides/obfuscated) your application using [garble](https://github.com/burrowers/garble). +- Support for compressing your application using [UPX](https://upx.github.io/). +- Automatic TypeScript generation of Go structs. More info [here](/docs/howdoesitwork#calling-bound-go-methods). +- No extra libraries or DLLs are required to be shipped with your application. For any platform. +- No requirement to bundle frontend assets. Just develop your application like any other web application. + +# Credit & Thanks + +Getting to v2 has been a huge effort. There have been ~2.2K commits by 89 contributors between the initial alpha and the release today, and many, many more that have provided translations, testing, feedback and help on the discussion forums as well as the issue tracker. I'm so unbelievably grateful to each one of you. I'd also like to give an extra special thank you to all the project sponsors who have provided guidance, advice and feedback. Everything you do is hugely appreciated. + +There are a few people I'd like to give special mention to: + +Firstly, a **huge** thank you to [@stffabi](https://github.com/stffabi) who has provided so many contributions which we all benefit from, as well as providing a lot of support on many issues. He has provided some key features such as the external dev server support which transformed our dev mode offering by allowing us to hook into [Vite](https://vitejs.dev/)'s superpowers. It's fair to say that Wails v2 would be a far less exciting release without his [incredible contributions](https://github.com/wailsapp/wails/commits?author=stffabi&since=2020-01-04). Thank you so much @stffabi! + +I'd also like to give a huge shout-out to [@misitebao](https://github.com/misitebao) who has tirelessly been maintaining the website, as well as providing Chinese translations, managing Crowdin and helping new translators get up to speed. This is a hugely important task, and I'm extremely grateful for all the time and effort put into this! You rock! + +Last, but not least, a huge thank you to Mat Ryer who has provided advice and support during the development of v2. Writing xBar together using an early Alpha of v2 was helpful in shaping the direction of v2, as well as give me an understanding of some design flaws in the early releases. I'm happy to announce that as of today, we will start to port xBar to Wails v2, and it will become the flagship application for the project. Cheers Mat! + +# Lessons Learnt + +There are a number of lessons learnt in getting to v2 that will shape development moving forward. + +## Smaller, Quicker, Focused Releases + +In the course of developing v2, there were many features and bug fixes that were developed on an ad-hoc basis. This led to longer release cycles and were harder to debug. Moving forward, we are going to create releases more often that will include a reduced number of features. A release will involve updates to documentation as well as thorough testing. Hopefully, these smaller, quicker, focussed releases will lead to fewer regressions and better quality documentation. + +## Encourage Engagement + +When starting this project, I wanted to immediately help everyone who had a problem. Issues were "personal" and I wanted them resolved as quickly as possible. This is unsustainable and ultimately works against the longevity of the project. Moving forward, I will be giving more space for people to get involved in answering questions and triaging issues. It would be good to get some tooling to help with this so if you have any suggestions, please join in the discussion [here](https://github.com/wailsapp/wails/discussions/1855). + +## Learning to say No + +The more people that engage with an Open Source project, the more requests there will be for additional features that may or may not be useful to the majority of people. These features will take an initial amount of time to develop and debug, and incur an ongoing maintenance cost from that point on. I myself am the most guilty of this, often wanting to "boil the sea" rather than provide the minimum viable feature. Moving forward, we will need to say "No" a bit more to adding core features and focus our energies on a way to empower developers to provide that functionality themselves. We are looking seriously into plugins for this scenario. This will allow anyone to extend the project as they see fit, as well as providing an easy way to contribute towards the project. + +# Looking to the Future + +There are so many core features we are looking at to add to Wails in the next major development cycle already. The [roadmap](https://github.com/wailsapp/wails/discussions/1484) is full of interesting ideas, and I'm keen to start work on them. One of the big asks has been for multiple window support. It's a tricky one and to do it right, and we may need to look at providing an alternative API, as the current one was not designed with this in mind. Based on some preliminary ideas and feedback, I think you'll like where we're looking to go with it. + +I'm personally very excited at the prospect of getting Wails apps running on mobile. We already have a demo project showing that it is possible to run a Wails app on Android, so I'm really keen to explore where we can go with this! + +A final point I'd like to raise is that of feature parity. It has long been a core principle that we wouldn't add anything to the project without there being full cross-platform support for it. Whilst this has proven to be (mainly) achievable so far, it has really held the project back in releasing new features. Moving forward, we will be adopting a slightly different approach: any new feature that cannot be immediately released for all platforms will be released under an experimental configuration or API. This allows early adopters on certain platforms to try the feature and provide feedback that will feed into the final design of the feature. This, of course, means that there are no guarantees of API stability until it is fully supported by all the platforms it can be supported on, but at least it will unblock development. + +# Final Words + +I'm really proud of what we've been able to achieve with the V2 release. It's amazing to see what people have already been able to build using the beta releases so far. Quality applications like [Varly](https://varly.app/), [Surge](https://getsurge.io/) and [October](https://october.utf9k.net/). I encourage you to check them out. + +This release was achieved through the hard work of many contributors. Whilst it is free to download and use, it has not come about through zero cost. Make no mistakes, this project has come at considerable cost. It has not only been my time and the time of each and every contributor, but also the cost of absence from friends and families of each of those people too. That's why I'm extremely grateful for every second that has been dedicated to making this project happen. The more contributors we have, the more this effort can be spread out and the more we can achieve together. I'd like to encourage you all to pick one thing that you can contribute, whether it is confirming someone's bug, suggesting a fix, making a documentation change or helping out someone who needs it. All of these small things have such a huge impact! It would be so awesome if you too were part of the story in getting to v3. + +Enjoy! + +‐ Lea + +PS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx b/website/i18n/vi/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx new file mode 100644 index 00000000000..9a137d09a8c --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/2023-01-17-v3-roadmap.mdx @@ -0,0 +1,184 @@ +--- +slug: the-road-to-wails-v3 +title: The Road to Wails v3 +authors: + - leaanthony +tags: + - wails + - v3 +--- + +```mdx-code-block +
+ +
+
+``` + +# Introduction + +Wails is a project that simplifies the ability to write cross-platform desktop applications using Go. It uses native webview components for the frontend (not embedded browsers), bringing the power of the world's most popular UI system to Go, whilst remaining lightweight. + +Version 2 was released on the 22nd of September 2022 and brought with it a lot of enhancements including: + +- Live development, leveraging the popular Vite project +- Rich functionality for managing windows and creating menus +- Microsoft's WebView2 component +- Generation of Typescript models that mirror your Go structs +- Creating of NSIS Installer +- Obfuscated builds + +Right now, Wails v2 provides powerful tooling for creating rich, cross-platform desktop applications. + +This blog post aims to look at where the project is at right now and what we can improve on moving forward. + +# Where are we now? + +It's been incredible to see the popularity of Wails rising since the v2 release. I'm constantly amazed by the creativity of the community and the wonderful things that are being built with it. With more popularity, comes more eyes on the project. And with that, more feature requests and bug reports. + +Over time, I've been able to identify some of the most pressing issues facing the project. I've also been able to identify some of the things that are holding the project back. + +## Current issues + +I've identified the following areas that I feel are holding the project back: + +- The API +- Bindings generation +- The Build System + +### The API + +The API to build a Wails application currently consists of 2 parts: + +- The Application API +- The Runtime API + +The Application API famously has only 1 function: `Run()` which takes a heap of options which govern how the application will work. Whilst this is very simple to use, it is also very limiting. It is a "declarative" approach which hides a lot of the underlying complexity. For instance, there is no handle to the main window, so you can't interact with it directly. For that, you need to use the Runtime API. This is a problem when you start to want to do more complex things like create multiple windows. + +The Runtime API provides a lot of utility functions for the developer. This includes: + +- Window management +- Dialogs +- Menus +- Events +- Logs + +There are a number of things I am not happy with the Runtime API. The first is that it requires a "context" to be passed around. This is both frustrating and confusing for new developers who pass in a context and then get a runtime error. + +The biggest issue with the Runtime API is that it was designed for applications that only use a single window. Over time, the demand for multiple windows has grown and the API is not well suited to this. + +### Thoughts on the v3 API + +Wouldn't it be great if we could do something like this? + +```go +func main() { + app := wails.NewApplication(options.App{}) + myWindow := app.NewWindow(options.Window{}) + myWindow.SetTitle("My Window") + myWindow.On(events.Window.Close, func() { + app.Quit() + }) + app.Run() +} +``` + +This programmatic approach is far more intuitive and allows the developer to interact with the application elements directly. All current runtime methods for windows would simply be methods on the window object. For the other runtime methods, we could move them to the application object like so: + +```go +app := wails.NewApplication(options.App{}) +app.NewInfoDialog(options.InfoDialog{}) +app.Log.Info("Hello World") +``` + +This is a much more powerful API which will allow for more complex applications to be built. It also allows for the creation of multiple windows, [the most up-voted feature on GitHub](https://github.com/wailsapp/wails/issues/1480): + +```go +func main() { + app := wails.NewApplication(options.App{}) + myWindow := app.NewWindow(options.Window{}) + myWindow.SetTitle("My Window") + myWindow.On(events.Window.Close, func() { + app.Quit() + }) + myWindow2 := app.NewWindow(options.Window{}) + myWindow2.SetTitle("My Window 2") + myWindow2.On(events.Window.Close, func() { + app.Quit() + }) + app.Run() +} +``` + +### Bindings generation + +One of the key features of Wails is generating bindings for your Go methods so they may be called from Javascript. The current method for doing this is a bit of a hack. It involves building the application with a special flag and then running the resultant binary which uses reflection to determine what has been bound. This leads to a bit of a chicken and egg situation: You can't build the application without the bindings and you can't generate the bindings without building the application. There are many ways around this but the best one would be not to use this approach at all. + +There was a number of attempts at writing a static analyser for Wails projects but they didn't get very far. In more recent times, it has become slightly easier to do this with more material available on the subject. + +Compared to reflection, the AST approach is much faster however it is significantly more complicated. To start with, we may need to impose certain constraints on how to specify bindings in the code. The goal is to support the most common use cases and then expand it later on. + +### The Build System + +Like the declarative approach to the API, the build system was created to hide the complexities of building a desktop application. When you run `wails build`, it does a lot of things behind the scenes: +- Builds the backend binary for bindings and generates the bindings +- Installs the frontend dependencies +- Builds the frontend assets +- Determines if the application icon is present and if so, embeds it +- Builds the final binary +- If the build is for `darwin/universal` it builds 2 binaries, one for `darwin/amd64` and one for `darwin/arm64` and then creates a fat binary using `lipo` +- If compression is required, it compresses the binary with UPX +- Determines if this binary is to be packaged and if so: + - Ensures the icon and application manifest are compiled into the binary (Windows) + - Builds out the application bundle, generates the icon bundle and copies it, the binary and Info.plist to the application bundle (Mac) +- If an NSIS installer is required, it builds it + +This entire process, whilst very powerful, is also very opaque. It is very difficult to customise it and it is very difficult to debug. + +To address this in v3, I would like to move to a build system that exists outside of Wails. After using [Task](https://taskfile.dev/) for a while, I am a big fan of it. It is a great tool for configuring build systems and should be reasonably familiar to anyone who has used Makefiles. + +The build system would be configured using a `Taskfile.yml` file which would be generated by default with any of the supported templates. This would have all of the steps required to do all the current tasks, such as building or packaging the application, allowing for easy customisation. + +There will be no external requirement for this tooling as it would form part of the Wails CLI. This means that you can still use `wails build` and it will do all the things it does today. However, if you want to customise the build process, you can do so by editing the `Taskfile.yml` file. It also means you can easily understand the build steps and use your own build system if you wish. + +The missing piece in the build puzzle is the atomic operations in the build process, such as icon generation, compression and packaging. To require a bunch of external tooling would not be a great experience for the developer. To address this, the Wails CLI will provide all these capabilities as part of the CLI. This means that the builds still work as expected, with no extra external tooling, however you can replace any step of the build with any tool you like. + +This will be a much more transparent build system which will allow for easier customisation and address a lot of the issues that have been raised around it. + +## The Payoff + +These positive changes will be a huge benefit to the project: +- The new API will be much more intuitive and will allow for more complex applications to be built. +- Using static analysis for bindings generation will be much faster and reduce a lot of the complexity around the current process. +- Using an established, external build system will make the build process completely transparent, allowing for powerful customisation. + +Benefits to the project maintainers are: + +- The new API will be much easier to maintain and adapt to new features and platforms. +- The new build system will be much easier to maintain and extend. I hope this will lead to a new ecosystem of community driven build pipelines. +- Better separation of concerns within the project. This will make it easier to add new features and platforms. + +## The Plan + +A lot of the experimentation for this has already been done and it's looking good. There is no current timeline for this work but I'm hoping by the end of Q1 2023, there will be an alpha release for Mac to allow the community to test, experiment with and provide feedback. + +## Summary + +- The v2 API is declarative, hides a lot from the developer and not suitable for features such as multiple windows. A new API will be created which will be simpler, intuitive and more powerful. +- The build system is opaque and difficult to customise so we will move to an external build system which will open it all up. +- The bindings generation is slow and complex so we will move to static analysis which will remove a lot of the complexity the current method has. + +There has been a lot of work put into the guts of v2 and it's solid. It's now time to address the layer on top of it and make it a much better experience for the developer. + +I hope you are as excited about this as I am. I'm looking forward to hearing your thoughts and feedback. + +Regards, + +‐ Lea + +PS: If you or your company find Wails useful, please consider [sponsoring the project](https://github.com/sponsors/leaanthony). Thanks! + +PPS: Yes, that's a genuine screenshot of a multi-window application built with Wails. It's not a mockup. It's real. It's awesome. It's coming soon. \ No newline at end of file diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/authors.yml b/website/i18n/vi/docusaurus-plugin-content-blog/authors.yml new file mode 100644 index 00000000000..a1621fab6b0 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/authors.yml @@ -0,0 +1,10 @@ +leaanthony: + name: Lea Anthony + title: Maintainer of Wails + url: https://github.com/leaanthony + image_url: https://github.com/leaanthony.png +misitebao: + name: Misite Bao + title: Architect + url: https://github.com/misitebao + image_url: https://github.com/misitebao.png diff --git a/website/i18n/vi/docusaurus-plugin-content-blog/options.json b/website/i18n/vi/docusaurus-plugin-content-blog/options.json new file mode 100644 index 00000000000..9239ff706c2 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-blog/options.json @@ -0,0 +1,14 @@ +{ + "title": { + "message": "Blog", + "description": "The title for the blog used in SEO" + }, + "description": { + "message": "Blog", + "description": "The description for the blog used in SEO" + }, + "sidebar.title": { + "message": "Recent posts", + "description": "The label for the left sidebar" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current.json b/website/i18n/vi/docusaurus-plugin-content-docs/current.json new file mode 100644 index 00000000000..ea817110073 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current.json @@ -0,0 +1,38 @@ +{ + "version.label": { + "message": "Phiên Bản Tiếp Theo 🚧", + "description": "The label for version current" + }, + "sidebar.docs.category.Getting Started": { + "message": "Hướng Dẫn Nhanh", + "description": "The label for category Getting Started in sidebar docs" + }, + "sidebar.docs.category.Reference": { + "message": "Tham Khảo", + "description": "The label for category Reference in sidebar docs" + }, + "sidebar.docs.category.Runtime": { + "message": "Bộ Công Cụ Runtime", + "description": "The label for category Runtime in sidebar docs" + }, + "sidebar.docs.category.Community": { + "message": "Cộng Đồng", + "description": "The label for category Community in sidebar docs" + }, + "sidebar.docs.category.Showcase": { + "message": "Trưng Bày", + "description": "The label for category Showcase in sidebar docs" + }, + "sidebar.docs.category.Guides": { + "message": "Định Hướng", + "description": "The label for category Guides in sidebar docs" + }, + "sidebar.docs.category.Tutorials": { + "message": "Thực Hành", + "description": "The label for category Tutorials in sidebar docs" + }, + "sidebar.docs.link.Contributing": { + "message": "Đóng Góp", + "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/links.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/links.mdx new file mode 100644 index 00000000000..e36a6edfe8c --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# Links + +This page serves as a list for community related links. Please submit a PR (click `Edit this page` at the bottom) to submit links. + +## Awesome Wails + +The [definitive list](https://github.com/wailsapp/awesome-wails) of links related to Wails. + +## Support Channels + +- [Wails Discord Server](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [v2 Beta Discussion Board](https://github.com/wailsapp/wails/discussions/828) + +## Social Media + +- [Twitter](https://twitter.com/wailsapp) +- [Wails Chinese Community QQ Group](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - Group number: 1067173054 + +## Other Tutorials and Articles + +- [Building of Bulletin Board](https://blog.customct.com/building-bulletin-board) diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..37be75135ed --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +

+ +
+

+``` + +The [BulletinBoard](https://github.com/raguay/BulletinBoard) application is a versital message board for static messages or dialogs to get information from the user for a script. It has a TUI for creating new dialogs that can latter be used to get information from the user. It's design is to stay running on your system and show the information as needed and then hide away. I have a process for watching a file on my system and sending the contents to BulletinBoard when changed. It works great with my workflows. There is also an [Alfred workflow](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) for sending information to the program. The workflow is also for working with [EmailIt](https://github.com/raguay/EmailIt). diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx new file mode 100644 index 00000000000..c1817b70fff --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/) is a Wails 2 program that is a markdown based email sender only with nine notepads, scripts to manipulate the text, and templates. It also has a scripts terminal to run scripts in EmailIt on files in your system. The scripts and templates can be used from the commandline itself or with the Alfred, Keyboard Maestro, Dropzone, or PopClip extensions. It also supports scripts and themes downloaded form GitHub. Documentation is not complete, but the programs works. It’s built using Wails2 and Svelte, and the download is a universal macOS application. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/encrypteasy.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..7504950ea76 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app) is a simple and easy to use PGP encryption tool, managing all your and your contacts keys. Encryption should be simple. Developed with Wails.** + +Encrypting messages using PGP is the industry standard. Everyone has a private and a public key. Your private key, well, needs to be kept private so only you can read messages. Your public key is distributed to anyone who wants to send you secret, encrypted messages. Managing keys, encrypting messages and decrypting messages should be a smooth experience. EncryptEasy is all about making it easy. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx new file mode 100644 index 00000000000..bc569c3fec4 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# FileHound Export Utility + +```mdx-code-block +

+ +
+

+``` + +[FileHound Export Utility](https://www.filehound.co.uk/) FileHound is a cloud document management platform made for secure file retention, business process automation and SmartCapture capabilities. + +The FileHound Export Utility allows FileHound Administrators the ability to run a secure document and data extraction tasks for alternative back-up and recovery purposes. This application will download all documents and/or meta data saved in FileHound based on the filters you choose. The metadata will be exported in both JSON and XML formats. + +Backend built with: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +Frontend with: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..87e5837d32f --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter) is a simple and efficient http API testing client tool. Based on Wails, Go and sveltejs. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/minecraftupdater.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..2f6c7c72b57 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Minecraft Updater + +```mdx-code-block +

+ +
+

+``` + +[Minecraft Updater](https://github.com/Gurkengewuerz/MinecraftModUpdater) is a utility tool to update and synchronize Minecraft mods for your userbase. It’s built using Wails2 and React with [antd](https://ant.design/) as frontend framework. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..bcd21239607 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Modal File Manager + +```mdx-code-block +

+ +
+

+``` + +[Modal File Manager](https://github.com/raguay/ModalFileManager) is a dual pane file manager using web technologies. My original design was based on NW.js and can be found [here](https://github.com/raguay/ModalFileManager-NWjs). This version uses the same Svelte based frontend code (but it has be greatly modified since the departure from NW.js), but the backend is a [Wails 2](https://wails.io/) implementation. By using this implementation, I no longer use command line `rm`, `cp`, etc. commands, but a git install has to be on the system to download themes and extensions. It is fully coded using Go and runs much faster than the previous versions. + +This file manager is designed around the same principle as Vim: a state controlled keyboard actions. The number of states isn't fixed, but very programmable. Therefore, an infinite number of keyboard configurations can be created and used. This is the main difference from other file managers. There are themes and extensions available to download from GitHub. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/mollywallet.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..5d846d06d33 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molly Wallet](https://github.com/grvlle/constellation_wallet/) the official $DAG wallet of the Constellation Network. It'll let users interact with the Hypergraph Network in various ways, not limited to producing $DAG transactions. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/october.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/october.mdx new file mode 100644 index 00000000000..66d634dc519 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[October](https://october.utf9k.net) is a small Wails application that makes it really easy to extract highlights from [Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader) and then forward them to [Readwise](https://readwise.io). + +It has a relatively small scope with all platform versions weighing in under 10MB, and that's without enabling [UPX compression](https://upx.github.io/)! + +In contrast, the author's previous attempts with Electron quickly bloated to several hundred megabytes. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/optimus.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/optimus.mdx new file mode 100644 index 00000000000..4f87479d6cf --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# Optimus + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus) is a desktop image optimization application. It supports conversion and compression between WebP, JPEG, and PNG image formats. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/portfall.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/portfall.mdx new file mode 100644 index 00000000000..03e740f4c6d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# Portfall + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - A desktop k8s port-forwarding portal for easy access to all your cluster UIs diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/restic-browser.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..3646384ecad --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - A simple, cross-platform [restic](https://github.com/restic/restic) backup GUI for browsing and restoring restic repositories. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/riftshare.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..9928b478576 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +Easy, Secure, and Free file sharing for everyone. Learn more at [Riftshare.app](https://riftshare.app) + +## Features + +- Easy secure file sharing between computers both in the local network and through the internet +- Supports sending files or directories securely through the [magic wormhole protocol](https://magic-wormhole.readthedocs.io/en/latest/) +- Compatible with all other apps using magic wormhole (magic-wormhole or wormhole-william CLI, wormhole-gui, etc.) +- Automatic zipping of multiple selected files to send at once +- Full animations, progress bar, and cancellation support for sending and receiving +- Native OS File Selection +- Open files in one click once received +- Auto Update - don't worry about having the latest release! diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..3e41eb32abf --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) is a program to show the output of scripts or [Node-Red](https://nodered.org) server. It runs scripts defined in EmailIt program and shows the output. Scripts from xBar or TextBar can be used, but currently on the TextBar scripts work well. It also displays the output of scripts on your system. ScriptBar doesn't put them in the menubar, but has them all in a convient window for easy viewing. You can have multiple tabs to have many different things show. You can also keep the links to your most visited web sites. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/surge.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/surge.mdx new file mode 100644 index 00000000000..c3b3fb4c06d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/) is a p2p filesharing app designed to utilize blockchain technologies to enable 100% anonymous file transfers. Surge is end-to-end encrypted, decentralized and open source. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wally.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wally.mdx new file mode 100644 index 00000000000..7408aa5854f --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally) is the official firmware flasher for [Ergodox](https://ergodox-ez.com/) keyboards. It looks great and is a fantastic example of what you can achieve with Wails: the ability to combine the power of Go and the rich graphical tools of the web development world. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx new file mode 100644 index 00000000000..950dc3f3db1 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# Minecraft launcher for WarMine + +```mdx-code-block +

+ + +
+

+``` + +[Minecraft launcher for WarMine](https://warmine.ru/) is a Wails application, that allows you to easily join modded game servers and manage your game accounts. + +The Launcher downloads the game files, checks their integrity and launches the game with a wide range of customization options for the launch arguments from the backend. + +Frontend is written in Svelte, whole launcher fits in 9MB and supports Windows 7-11. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wombat.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wombat.mdx new file mode 100644 index 00000000000..f100c55e2de --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# Wombat + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat) is a cross platform gRPC client. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/ytd.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/ytd.mdx new file mode 100644 index 00000000000..5db428f722c --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails) is an app for downloading tracks from youtube, creating offline playlists and share them with your friends, your friends will be able to playback your playlists or download them for offline listening, has an built-in player. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/templates.mdx new file mode 100644 index 00000000000..6ecbac37ee8 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# Templates + +This page serves as a list for community supported templates. Please submit a PR (click `Edit this page` at the bottom) to include your templates. To build your own template, please see the [Templates](../guides/templates.mdx) guide. + +To use these templates, run `wails init -n "Your Project Name" -t [the link below[@version]]` + +If there is no version suffix, the main branch code template is used by default. If there is a version suffix, the code template corresponding to the tag of this version is used. + +Example: `wails init -n "Your Project Name" -t https://github.com/misitebao/wails-template-vue` + +:::warning Attention + +**The Wails project does not maintain, is not responsible nor liable for 3rd party templates!** + +If you are unsure about a template, inspect `package.json` and `wails.json` for what scripts are run and what packages are installed. + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - Wails template based on Vue ecology (Integrated TypeScript, Dark theme, Internationalization, Single page routing, TailwindCSS) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Vue 3 TypeScript with Vite (and instructions to add features) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vue 3 TypeScript with Vite, Vuex, Vue Router, Sass, and ESLint + Prettier +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - A template using JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - A template using TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, Composition API with <script setup>) +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Wails template based on Naive UI (A Vue 3 Component Library) + +## Angular + +- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - Angular 15+ action packed & ready to roll to production. +- [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - Angular with TypeScript, Sass, Hot-Reload, Code-Splitting and i18n + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - A template using reactjs +- [wails-react-template](https://github.com/flin7/wails-react-template) - A minimal template for React that supports live development +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - A template using Next.js and TypeScript +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - A template for React + TypeScript + Vite + TailwindCSS +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - A template using Svelte +- [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - A template using Svelte and Vite +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - A template using Svelte and Vite with TailwindCSS v3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - A template using SvelteKit + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - A template using Solid + Ts + Vite +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - A template using Solid + Js + Vite + +## Elm + +- [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + +## Pure JavaScript (Vanilla) + +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/building.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/building.mdx new file mode 100644 index 00000000000..a1f69f03357 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# Compiling your Project + +From the project directory, run `wails build`. This will compile your project and save the production-ready binary in the `build/bin` directory. + +If you run the binary, you should see the default application: + +```mdx-code-block +
+ +
+
+``` + +For more details on compilation options, please refer to the [CLI Reference](../reference/cli.mdx#build). diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx new file mode 100644 index 00000000000..034be31d620 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# Developing your Application + +You can run your application in development mode by running `wails dev` from your project directory. This will do the following things: + +- Build your application and run it +- Bind your Go code to the frontend so it can be called from JavaScript +- Using the power of [Vite](https://vitejs.dev/), will watch for modifications in your Go files and rebuild/re-run on change +- Sets up a [webserver](http://localhost:34115) that will serve your application over a browser. This allows you to use your favourite browser extensions. You can even call your Go code from the console + +To get started, run `wails dev` in the project directory. More information on this can be found [here](../reference/cli.mdx#dev). + +Coming soon: Tutorial diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/firstproject.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..62ee97c8487 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# Creating a Project + +## Project Generation + +Now that the CLI is installed, you can generate a new project by using the `wails init` command. + +Pick your favourite framework: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Generate a Svelte project using JavaScript with:

+ + wails init -n myproject -t svelte + +If you would rather use TypeScript:
+ + wails init -n myproject -t svelte-ts + +
+ + Generate a React project using JavaScript with:

+ + wails init -n myproject -t react + +If you would rather use TypeScript:
+ + wails init -n myproject -t react-ts + +
+ + Generate a Vue project using JavaScript with:

+ + wails init -n myproject -t vue + +If you would rather use TypeScript:
+ + wails init -n myproject -t vue-ts + +
+ + Generate a Preact project using JavaScript with:

+ + wails init -n myproject -t preact + +If you would rather use TypeScript:
+ + wails init -n myproject -t preact-ts + +
+ + Generate a Lit project using JavaScript with:

+ + wails init -n myproject -t lit + +If you would rather use TypeScript:
+ + wails init -n myproject -t lit-ts + +
+ + Generate a Vanilla project using JavaScript with:

+ + wails init -n myproject -t vanilla + +If you would rather use TypeScript:
+ + wails init -n myproject -t vanilla-ts + +
+
+``` + +
+ +There are also [community templates](../community/templates.mdx) available that offer different capabilities and frameworks. + +To see the other options available, you can run `wails init -help`. More details can be found in the [CLI Reference](../reference/cli.mdx#init). + +## Project Layout + +Wails projects have the following layout: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### Project structure rundown + +- `/main.go` - The main application +- `/frontend/` - Frontend project files +- `/build/` - Project build directory +- `/build/appicon.png` - The application icon +- `/build/darwin/` - Mac specific project files +- `/build/windows/` - Windows specific project files +- `/wails.json` - The project configuration +- `/go.mod` - Go module file +- `/go.sum` - Go module checksum file + +The `frontend` directory has nothing specific to Wails and can be any frontend project of your choosing. + +The `build` directory is used during the build process. These files may be updated to customise your builds. If files are removed from the build directory, default versions will be regenerated. + +The default module name in `go.mod` is "changeme". You should change this to something more appropriate. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/installation.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/installation.mdx new file mode 100644 index 00000000000..14ed47373fa --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/gettingstarted/installation.mdx @@ -0,0 +1,90 @@ +--- +sidebar_position: 1 +--- + +# Installation + +## Supported Platforms + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## Dependencies + +Wails has a number of common dependencies that are required before installation: + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +Download Go from the [Go Downloads Page](https://go.dev/dl/). + +Ensure that you follow the official [Go installation instructions](https://go.dev/doc/install). You will also need to ensure that your `PATH` environment variable also includes the path to your `~/go/bin` directory. Restart your terminal and do the following checks: + +- Check Go is installed correctly: `go version` +- Check "~/go/bin" is in your PATH variable: `echo $PATH | grep go/bin` + +### NPM + +Download NPM from the [Node Downloads Page](https://nodejs.org/en/download/). It is best to use the latest release as that is what we generally test against. + +Run `npm --version` to verify. + +## Platform Specific Dependencies + +You will also need to install platform specific dependencies: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wails requires that the xcode command line tools are installed. This can be + done by running xcode-select --install. + + + Wails requires that the WebView2 runtime is installed. Some Windows installations will already have this installed. You can check using the wails doctor command. + + + Linux requires the standard gcc build tools plus libgtk3 and libwebkit. Rather than list a ton of commands for different distros, Wails can try to determine what the installation commands are for your specific distribution. Run wails doctor after installation to be shown how to install the dependencies. If your distro/package manager is not supported, please consult the Add Linux Distro guide. + + +``` + +## Optional Dependencies + +- [UPX](https://upx.github.io/) for compressing your applications. +- [NSIS](https://wails.io/docs/guides/windows-installer/) for generating Windows installers. + +## Installing Wails + +Run `go install github.com/wailsapp/wails/v2/cmd/wails@latest` to install the Wails CLI. + +Note: If you get an error similar to this: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +please check you have Go 1.18+ installed: +```shell +go version +``` + +## System Check + +Running `wails doctor` will check if you have the correct dependencies installed. If not, it will advise on what is missing and help on how to rectify any problems. + +## The `wails` command appears to be missing? + +If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment variable. You will also normally need to close and reopen any open command prompts so that changes to the environment made by the installer are reflected at the command prompt. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/angular.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/angular.mdx new file mode 100644 index 00000000000..2b6c5a84571 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +Whilst Wails does not have an Angular template, it is possible to use Angular with Wails. + +## Dev Mode + +To get dev mode working with Angular, you need to add the following to your `wails.json`: + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/application-development.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/application-development.mdx new file mode 100644 index 00000000000..e3b7cbc668d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/application-development.mdx @@ -0,0 +1,215 @@ +# Application Development + +There are no hard and fast rules for developing applications with Wails, but there are some basic guidelines. + +## Application Setup + +The pattern used by the default templates are that `main.go` is used for configuring and running the application, whilst `app.go` is used for defining the application logic. + +The `app.go` file will define a struct that has 2 methods which act as hooks into the main application: + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- The startup method is called as soon as Wails allocates the resources it needs and is a good place for creating resources, setting up event listeners and anything else the application needs at startup. It is given a `context.Context` which is usually saved in a struct field. This context is needed for calling the [runtime](../reference/runtime/intro.mdx). If this method returns an error, the application will terminate. In dev mode, the error will be output to the console. + +- The shutdown method will be called by Wails right at the end of the shutdown process. This is a good place to deallocate memory and perform any shutdown tasks. + +The `main.go` file generally consists of a single call to `wails.Run()`, which accepts the application configuration. The pattern used by the templates is that before the call to `wails.Run()`, an instance of the struct we defined in `app.go` is created and saved in a variable called `app`. This configuration is where we add our callbacks: + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +More information on application lifecycle hooks can be found [here](../howdoesitwork.mdx#application-lifecycle-callbacks). + +## Binding Methods + +It is likely that you will want to call Go methods from the frontend. This is normally done by adding public methods to the already defined struct in `app.go`: + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +In the main application configuration, the `Bind` key is where we can tell Wails what we want to bind: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +This will bind all public methods in our `App` struct (it will never bind the startup and shutdown methods). + +### Dealing with context when binding multiple structs + +If you want to bind methods for multiple structs but want each struct to keep a reference to the context so that you can use the runtime functions, a good pattern is to pass the context from the `OnStartup` method to your struct instances : + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +More information on Binding can be found [here](../howdoesitwork.mdx#method-binding). + +## Application Menu + +Wails supports adding a menu to your application. This is done by passing a [Menu](../reference/menus.mdx#menu) struct to application config. It's common to use a method that returns a Menu, and even more common for that to be a method on the `App` struct used for the lifecycle hooks. + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## Assets + +The great thing about the way Wails v2 handles assets is that it doesn't! The only thing you need to give Wails is an `embed.FS`. How you get to that is entirely up to you. You can use vanilla html/css/js files like the vanilla template. You could have some complicated build system, it doesn't matter. + +When `wails build` is run, it will check the `wails.json` project file at the project root. There are 2 keys in the project file that are read: + +- "frontend:install" +- "frontend:build" + +The first, if given, will be executed in the `frontend` directory to install the node modules. The second, if given, will be executed in the `frontend` directory to build the frontend project. + +If these 2 keys aren't given, then Wails does absolutely nothing with the frontend. It is only expecting that `embed.FS`. + +### AssetsHandler + +A Wails v2 app can optionally define a `http.Handler` in the `options.App`, which allows hooking into the AssetServer to create files on the fly or process POST/PUT requests. GET requests are always first handled by the `assets` FS. If the FS doesn't find the requested file the request will be forwarded to the `http.Handler` for serving. Any requests other than GET will be directly processed by the `AssetsHandler` if specified. It's also possible to only use the `AssetsHandler` by specifiy `nil` as the `Assets` option. + +## Built in Dev Server + +Running `wails dev` will start the built in dev server which will start a file watcher in your project directory. By default, if any file changes, wails checks if it was an application file (default: `.go`, configurable with `-e` flag). If it was, then it will rebuild your application and relaunch it. If the changed file was in the assets, it will issue a reload after a short amount of time. + +The dev server uses a technique called "debouncing" which means it doesn't reload straight away, as there may be multiple files changed in a short amount of time. When a trigger occurs, it waits for a set amount of time before issuing a reload. If another trigger happens, it resets to the wait time again. By default this value is `100ms`. If this value doesn't work for your project, it can be configured using the `-debounce` flag. If used, this value will be saved to your project config and become the default. + +## External Dev Server + +Some frameworks come with their own live-reloading server, however they will not be able to take advantage of the Wails Go bindings. In this scenario, it is best to run a watcher script that rebuilds the project into the build directory, which Wails will be watching. For an example, see the default svelte template that uses [rollup](https://rollupjs.org/guide/en/). + +### Create React App + +The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: + +```html + + + + + +``` + +This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. + +## Go Module + +The default Wails templates generate a `go.mod` file that contains the module name "changeme". You should change this to something more appropriate after project generation. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..989202e27a6 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# Dynamic Assets + +If you want to load or generate assets for your frontend dynamically, you can achieve that using the [AssetsHandler](../reference/options#assetshandler) option. The AssetsHandler is a generic `http.Handler` which will be called for any non GET request on the assets server and for GET requests which can not be served from the bundled assets because the file is not found. + +By installing a custom AssetsHandler, you can serve your own assets using a custom asset server. + +## Example + +In our example project, we will create a simple assets handler which will load files off disk: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +When we run the application in dev mode using `wails dev`, we will see the following output: + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +As you can see, the assets handler is called when the default assets server is unable to serve the `favicon.ico` file. + +If you right click the main application and select "inspect" to bring up the devtools, you can test this feature out by typing the following into the console: + +``` +let response = await fetch('does-not-exist.txt'); +``` + +This will generate an error in the devtools. We can see that the error is what we expect, returned by our custom assets handler: + +```mdx-code-block +

+ +

+``` + +However, if we request `go.mod`, we will see the following output: + +```mdx-code-block +

+ +

+``` + +This technique can be used to load images directly into the page. If we updated our default vanilla template and replaced the logo image: + +```html + +``` + +with: + +```html + +``` + +Then we would see the following: + +```mdx-code-block +

+ +

+``` + +:::warning + +Exposing your filesystem in this way is a security risk. It is recommended that you properly manage access to your filesystem. + +::: diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frameless.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frameless.mdx new file mode 100644 index 00000000000..3845736f427 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frameless.mdx @@ -0,0 +1,87 @@ +# Frameless Applications + +Wails supports application that have no frames. This can be achieved by using the [frameless](../reference/options.mdx#frameless) field in [Application Options](../reference/options.mdx#application-options). + +Wails offers a simple solution for dragging the window: Any HTML element that has the CSS style `--wails-draggable:drag` will act as a "drag handle". This property applies to all child elements. If you need to indicate that a nested element should not drag, then use the attribute '--wails-draggable:no-drag' on that element. + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +For some projects, using a CSS variable may not be possible due to dynamic styling. In this case, you can use the `CSSDragProperty` and `CSSDragValue` application options to define a property and value that will be used to indicate draggable regions: + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info Fullscreen + +If you allow your application to go fullscreen, this drag functionality will be disabled. + +::: diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frontend.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frontend.mdx new file mode 100644 index 00000000000..ac087ee4514 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/frontend.mdx @@ -0,0 +1,72 @@ +# Frontend + +## Script Injection + +When Wails serves your `index.html`, by default, it will inject 2 script entries into the `` tag to load `/wails/ipc.js` and `/wails/runtime.js`. These files install the bindings and runtime respectively. + +The code below shows where these are injected by default: + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### Overriding Default Script Injection + +To provide more flexibility to developers, there is a meta tag that may be used to customise this behaviour: + +```html + +``` + +The options are as follows: + +| Value | Description | +| ------------------- | ------------------------------------------------ | +| noautoinjectruntime | Disable the autoinjection of `/wails/runtime.js` | +| noautoinjectipc | Disable the autoinjection of `/wails/ipc.js` | +| noautoinject | Disable all autoinjection of scripts | + +Multiple options may be used provided they are comma seperated. + +This code is perfectly valid and operates the same as the autoinjection version: + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/ides.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/ides.mdx new file mode 100644 index 00000000000..f742822832a --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/ides.mdx @@ -0,0 +1,127 @@ +# IDEs + +Wails aims to provide a great development experience. To that aim, we now support generating IDE specific configuration to provide smoother project setup. + +Currently, we support [Visual Studio Code](https://code.visualstudio.com/) but aim to support other IDEs such as Goland. + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +When generating a project using the `-ide vscode` flags, IDE files will be created alongside the other project files. These files are placed into the `.vscode` directory and provide the correct configuration for debugging your application. + +The 2 files generated are `tasks.json` and `launch.json`. Below are the files generated for the default vanilla project: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### Configuring the install and build steps + +The `tasks.json` file is simple for the default project as there is no `npm install` or `npm run build` step needed. For projects that have a frontend build step, such as the svelte template, we would need to edit `tasks.json` to add the install and build steps: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +:::info Future Enhancement + +In the future, we hope to generate a `tasks.json` that includes the install and build steps automatically. + +::: diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux-distro-support.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..8b25c15754a --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Linux Distro Support + +## Overview + +Wails offers Linux support but providing installation instructions for all available distributions is an impossible task. Instead, Wails tries to determine if the packages you need to develop applications are available via your system's package manager. Currently, we support the following package managers: + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## Adding package names + +There may be circumstances where your distro uses one of the supported package managers but the package name is different. For example, you may use an Ubuntu derivative, but the package name for gtk may be different. Wails attempts to find the correct package by iterating through a list of package names. The list of packages are stored in the packagemanager specific file in the `v2/internal/system/packagemanager` directory. In our example, this would be `v2/internal/system/packagemanager/apt.go`. + +In this file, the list of packages are defined by the `Packages()` method: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +Let's assume that in our linux distro, `libgtk-3` is packaged under the name `lib-gtk3-dev`. We could add support for this by adding the following line: + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## Adding new package managers + +To add a new package manager, perform the following steps: + +- Create a new file in `v2/internal/system/packagemanager` called `.go`, where `` is the name of the package manager. +- Define a struct that conforms to the package manager interface defined in `pm.go`: + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()` should return the name of the package manager +- `Packages()` should return a `packagemap`, that provides candidate filenames for dependencies +- `PackageInstalled()` should return `true` if the given package is installed +- `PackageAvailable()` should return `true` if the given package is not installed but available for installation +- `InstallCommand()` should return the exact command to install the given package name + +Take a look at the other package managers code to get an idea how this works. + +:::info Remember + +If you add support for a new package manager, don't forget to also update this page! + +::: diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux.mdx new file mode 100644 index 00000000000..229c282bf55 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +This page has miscellaneous guides related to developing Wails applications for Linux. + +## Video tag doesn't fire "ended" event + +When using a video tag, the "ended" event is not fired when the video is finished playing. This is a bug in WebkitGTK, however you can use the following workaround to fix it: + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +Source: [Lyimmi](https://github.com/Lyimmi) on the [discussions board](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/local-development.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/local-development.mdx new file mode 100644 index 00000000000..4ba1f34c37d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/local-development.mdx @@ -0,0 +1,55 @@ +# Local Development + +## Overview + +Wails is in constant development and new releases are regularly "tagged". This usually happens when all the newer code on `master` has been tested and confirmed working. If you need a bugfix or feature that has not yet made it to a release, it's possible to use the latest "bleeding edge" version using the following steps: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +NOTE: The directory that you cloned the project into will now be called "clonedir". + +The Wails CLI will now be at the very latest version. + +### Updating your project + +To update projects to use the latest version of the Wails library, update the project's `go.mod` and ensure the following line is at the bottom of the file: + +`replace github.com/wailsapp/wails/v2 => ` + +Example: + +On Windows: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +On 'nix: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +To revert to a stable version, run: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## Testing a Branch + +If you want to test a branch, follow the instructions above, but ensure you switch the branch you want to test before installing: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +Make sure you [update your project](#updating-your-project) as described above. + +## Testing a PR + +If you want to test a PR, follow the instructions above, but ensure you fetch the PR and switch the branch before installing. Please replace `[IDofThePR]` with the ID of the PR shown on github.com: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +Make sure you [update your project](#updating-your-project) as described above. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx new file mode 100644 index 00000000000..961595711c7 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Mac App Store Guide + +This page gives a brief overview of how to submit your Wails App to the Mac App Store. + +## Prerequisites + +- You will need to have an Apple Developer account. Please find more information on the [Apple Developer Program](https://developer.apple.com/support/compare-memberships/) site +- You will need to have your Certificates, Identifiers, and App created on the developer portal. More on this below +- Xcode command line tools will need to be installed on your local machine + +#### Create Certificates and Identifiers + +1. Go to your [Apple Developer Account](https://developer.apple.com/account/) +2. Under `Certificates, Identifiers & Profiles`, click `Identifiers` and Register a New App ID. Use the format (com.example.app) +3. Under the same page click `Certificates` and generate new Certificates for Mac App Store Distribution. Download them and import the certificates into Keychain on your local machine. + +#### Create App Submission + +1. Go to the [App Store Connect Site](https://appstoreconnect.apple.com/apps) +2. Register a new application and link the bundle ID that you created in the previous step +3. Populate your app with the correct screen shots, descriptions, etc. as required by Apple +4. Create a new version of your app + +#### Create Provisioning Profile +1. Go to the [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) page +2. Add a new provisioning profile for Mac App Store Distribution +3. Set the Profile Type as Mac and select the App ID for the application created above +4. Select the Mac App Distribution certificate +5. Name the Provisioning Profile embedded and download the created profile. + +## Mac App Store Process + +#### Enable Apple's App Sandbox + +Apps submitted to the Mac App Store must run under Apple's [App Sandbox](https://developer.apple.com/app-sandboxing/). You must create an `entitlements.plist` file for this to work. The recommendation is to create this file under this path `{PROJECT_DIR}/build/darwin/entitlements.plist`. + +**Example Entitlements File** + +This is an example entitlements file from the [RiftShare](https://github.com/achhabra2/riftshare) app. For reference please put in the entitlements your app requires. Refer to [this site](https://developer.apple.com/documentation/bundleresources/entitlements) for more information. You will need to replace the Team ID and Application Name with the ones you registered above. + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +**Add the Embedded Provisioning Profile** The Provisioning Profile created above needs to be added to the root of the applicaton. It needs to be named embedded.provisionprofile. + +#### Build and Sign the App Package + +The following is an example script for building and signing your app for Mac App Store submission. It assumes you are running the script from your root project directory. + +Note the certificates for signing the app and signing the installer are different. Please make sure both are imported into Keychain. Find the strings in Keychain and insert them below. Populate your certificate names, and app name below. Running the following script will generate a signed `app.pkg` file in the root directory of your app. + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### Upload App Bundle + +You will need to upload the generated package file and associate it to your Application before you will be able to submit it for review. + +1. Download the [Transporter App](https://apps.apple.com/us/app/transporter/id1450874784) from the Mac App Store +2. Open it and sign in with your Apple ID +3. Click the + sign and select the `APP_NAME.pkg` file that you generated in the previous step. Upload it +4. Go back to the [App Store Connect](https://appstoreconnect.apple.com/apps) site and navigate back into your app submission. Select the version that you are ready to make available on the App Store. Under `Build` select the package that you uploaded via Transporter. + +That's it! You can now use the site to submit your App for review. After a few business days if all goes well you should see your App live on the Mac App Store. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx new file mode 100644 index 00000000000..190084c52c3 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# Xây Dựng Ứng Dụng Một Cách Thủ Công + +Các giao diện dòng lệnh (CLI) của Wails thực hiện rất nhiều công việc nặng nhọc nhất cho dự án của bạn, nhưng đôi khi bạn muốn xây dựng dự án theo một cách thủ công. Tài liệu tham khảo sau sẽ thảo luận về các cách khác nhau mà CLI có thể đạt được điều nêu trên. + +## Quy Trình Xây Dựng + +Khi dùng một trong hai câu lệnh `wails build` hoặc `wails dev`, CLI của Wails sẽ thực hiện các bước xây dựng sau đây: + + - Cài đặt các tập tin cần thiết cho frontend + - Xây dựng frontend + - Tạo ra các tập tin hệ thống + - Biên dịch ứng dụng cho hệ điều hành + - [không bắt buộc] Nén ứng dụng + +### Cài đặt các tập tin cần thiết cho frontend + +#### Các Bước CLI + +- Nếu lệnh phụ `-s` được đưa ra, bước này được lược bỏ +- Kiểm ra tệp `wails.json` xem nó đã được thêm vào câu lệnh `frontend:install` chưa +- Nếu chưa thì bước này được lược bỏ +- Nếu có thì tiếp tục kiểm tra xem tệp `package.json` có tồn tại trong thư mục frontend không. Nếu không có thì bước này được lược bỏ +- Tệp tin định dạng MD5 được tự tạo từ tệp tin `package.json` trong các tệp tin nội dung +- Nó kiểm tra sự tồn tại của tệp `package.json.md5` và nếu nó tồn tại, thì nó sẽ so sánh nội dung trong tập tin của chính mình (MD5 sum) với nội dung được tự tạo để xem nội dung có bị thay đổi hay không. Nếu chúng giống nhau thì bước này được bỏ qua +- Nếu tệp `package.json.md5` không tồn tại, nó sẽ tự động được tạo ra sử dụng nguồn từ tệp MD5 sum +- If a build is now required, or `node_modules` does not exist, or the `-f` flag is given, the install command is executed in the frontend directory + +#### Manual Steps + +This step could be done from the command line or a script with `npm install`. + +### Xây dựng frontend + +#### Wails CLI + +- Nếu lệnh phụ `-s` được đưa ra, bước này được lược bỏ +- Checks `wails.json` to see if there is a build command in the key `frontend:build` +- Nếu chưa thì bước này được lược bỏ +- If there is, it is executed in the frontend directory + +#### Manual Steps + +This step could be done from the command line or a script with `npm run build` or whatever the frontend build script is. + +### Generate assets + +#### Wails CLI + +- If `-nopackage` flag is set, this stage is skipped +- If the `build/appicon.png` file does not exist, a default one is created +- For Windows, see [Bundling for Windows](#windows) +- If `build/windows/icon.ico` does not exist, it will create it from the `build/appicon.png` image. + +##### Windows + +- If `build/windows/icon.ico` does not exist, it will create it from `build/appicon.png` using icon sizes of 256, 128, 64, 48, 32 and 16. This is done using [winicon](https://github.com/leaanthony/winicon). +- If the `build/windows/.manifest` file does not exist, it creates it from a default version. +- Compiles the application as a production build (above) +- Uses [winres](https://github.com/tc-hib/winres) to bundle the icon and manifest into a `.syso` file ready for linking. + +#### Manual Steps + +- Create `icon.ico` using the [winicon](https://github.com/leaanthony/winicon) CLI tool (or any other tool). +- Create / Update a `.manifest` file for your application +- Use the [winres CLI](https://github.com/tc-hib/go-winres) to generate a `.syso` file. + +### Biên dịch ứng dụng cho hệ điều hành + +#### Wails CLI + +- If the `-clean` flag is provided, the `build` directory is deleted and recreated +- For `wails dev`, the following default Go flags are used: `-tags dev -gcflags "all=-N -l"` +- For `wails build`, the following default Go flags are used: `-tags desktop,production -ldflags "-w -s"` + - On Windows, `-ldflags "-w -h -H windowsgui"` +- Additional tags passed to the CLI using `-tags` are added to the defaults +- Additional ldflags passed to the CLI using `-ldflags` are added to the defaults +- The `-o` flag is passed through +- The Go compiler specified by `-compiler` will be used for compilation + +#### Manual steps + +- For dev build, the minimum command would be: `go build -tags dev -gcflags "all=-N -l"` +- For production build, the minimum command would be: `go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- Ensure that you compile in the same directory as the `.syso` file + +### Compress application + +#### Wails CLI + +- If the `-upx` flag has been given, the `upx` program will be run to compress the application with the default settings +- If `-upxflags` is also passed, these flags are used instead of the default ones + +#### Manual steps + +- Run `upx [flags]` manually to compress the application. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/migrating.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/migrating.mdx new file mode 100644 index 00000000000..7123cbe6b60 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/migrating.mdx @@ -0,0 +1,191 @@ +# Migrating from v1 + +## Overview + +Wails v2 is a significant change from v1. This document aims to highlight the changes and the steps in migrating an existing project. + +### Creating the Application + +In v1, the main application is created using `wails.CreateApp`, bindings are added with `app.Bind`, then the application is run using `app.Run()`. + +Example: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +In v2, there is just a single method, `wails.Run()`, that accepts [application options](../reference/options.mdx#application-options). + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### Binding + +In v1, it was possible to bind both arbitrary functions and structs. In v2, this has been simplified to only binding structs. The struct instances that were previously passed to the `Bind()` method in v1, are now specified in the `Bind` field of the [application options](../reference/options.mdx#application-options): + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +In v1, bound methods were available to the frontend at `window.backend`. This has changed to `window.go`.`` + +### Application Lifecycle + +In v1, there were 2 special methods in a bound struct: `WailsInit()` and `WailsShutdown()`. These have been replaced with 3 lifecycle hooks as part of the [application options](../reference/options.mdx#application-options): + +- [OnStartup](../reference/options.mdx#onstartup) +- [OnShutdown](../reference/options.mdx#onshutdown) +- [OnDomReady](../reference/options.mdx#ondomready) + +Note: [OnDomReady](../reference/options.mdx#ondomready) replaces the `wails:ready` system event in v1. + +These methods can be standard functions, but a common practice is to have them part of a struct: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### Runtime + +The runtime in v2 is much richer than v1 with support for menus, window manipulation and better dialogs. The signature of the methods has changed slightly - please refer the the [Runtime Reference](../reference/runtime/intro.mdx). + +In v1, the [runtime](../reference/runtime/intro.mdx) was available via a struct passed to `WailsInit()`. In v2, the runtime has been moved out to its own package. Each method in the runtime takes the `context.Context` that is passed to the [OnStartup](../reference/options.mdx#onstartup) method. + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startup is called at application startup +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} + +``` + +### Assets + +The _biggest_ change in v2 is how assets are handled. + +In v1, assets were passed via 2 application options: + +- `JS` - The application's JavaScript +- `CSS` - The application's CSS + +This meant that the responsibility of generating a single JS and CSS file was on the developer. This essentially required the use of complicated packers such as webpack. + +In v2, Wails makes no assumptions about your frontend assets, just like a webserver. All of your application assets are passed to the application options as an `embed.FS`. + +**This means there is no requirement to bundle your assets, encode images as Base64 or attempt the dark art of bundler configuration to use custom fonts**. + +At startup, Wails will scan the given `embed.FS` for `index.html` and use its location as the root path for all the other application assets - just like a webserver would. + +Example: An application has the following project layout. All final assets are placed in the `frontend/dist` directory: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +Those assets may be used by the application by simply creating an `embed.FS`: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +Of course, bundlers can be used if you wish to. The only requirement is to pass the final application assets directory to Wails using an `embed.FS` in the `Assets` key of the [application options](../reference/options.mdx#application-options). + +### Project Configuration + +In v1, the project configuration was stored in the `project.json` file in the project root. In v2, the project configuration is stored in the `wails.json` file in the project root. + +The format of the file is slightly different. Here is a comparison: + +

+ +| v1 | v2 | Notes | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | Removed | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | Changed | +| frontend / dir | | Removed | +| frontend / install | frontend:install | Changed | +| frontend / build | frontend:build | Changed | +| frontend / bridge | | Removed | +| frontend / serve | | Removed | +| tags | | Removed | +| | wailsjsdir | The directory to generate wailsjs modules | +| | assetdir | The directory of the compiled frontend assets for `dev` mode. This is normally inferred and could be left empty. | +| | reloaddirs | Comma separated list of additional directories to watch for changes and to trigger reloads in `dev` mode. This is only needed for some more advanced asset configurations. | + +

diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..4a3de2a61b5 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# Mouse Buttons + +The Wails runtime intercepts mouse clicks to determine whether a frameless window needs resizing or a window needs to be moved. It has been asked how to detect when a mouse click has occurred, because `window.onclick` doesn't report the mouse buttons correctly. The following code shows how to detect mouse clicks: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +Reference: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx new file mode 100644 index 00000000000..21f7875e389 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# Obfuscated Builds + +Wails includes support for obfuscating your application using [garble](https://github.com/burrowers/garble). + +To produce an obfuscated build, you can use the `-obfuscate` flag with the `wails build` command: + +```bash +wails build -obfuscated +``` + +To customise the obfuscation settings, you can use the `-garbleargs` flag: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +These settings may be persisted in your [project config](../reference/project-config). + +## How it works + +In a standard build, all bound methods are available in the frontend under the `window.go` variable. When these methods are called, the corresponding backend method is called using the fully qualified function name. When using an obfuscated build, methods are bound using an ID instead of a name. The bindings generated in the `wailsjs` directory use these IDs to call the backend functions. + +:::note + +To ensure that your application will work in obfuscated mode, you must use the generated bindings under the `wailsjs` directory in your application. + +::: + +## Example + +Importing the "Greet" method from the bindings like this: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +will ensure that the method will work correctly in obfuscated mode, as the bindings will be regenerated with IDs and the call mechanism updated. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/overscroll.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/overscroll.mdx new file mode 100644 index 00000000000..9d1d772d0fb --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# Overscroll + +[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) is the "bounce effect" you sometimes get when you scroll beyond a page's content boundaries. This is common in mobile apps. This can be disabled using CSS: + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/routing.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/routing.mdx new file mode 100644 index 00000000000..ce176943ea5 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/routing.mdx @@ -0,0 +1,47 @@ +# Routing + +Routing is a popular way to switch views in an application. This page offers some guidance around how to do that. + +## Vue + +The recommended approach for routing in Vue is [Hash Mode](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode): + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +The recommended approach for routing in Angular is [HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies#_hashlocationstrategy): + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +The recommended approach for routing in React is [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/signing.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/signing.mdx new file mode 100644 index 00000000000..4c7cf45ba99 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/signing.mdx @@ -0,0 +1,387 @@ +# Code Signing + +This is a guide on how you can sign your binaries generated with Wails on MacOS and Windows. The guide will target CI environments, more specifically GitHub Actions. + +## Windows + +First off you need a code signing certificate. If you do not already have one, Microsoft's info page lists some providers [here](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate). Please note that an EV certificate is not required unless you need to write kernel-level software such as device drivers. For signing your Wails app, a standard code signing certificate will do just fine. + +It may be a good idea to check with your certificate provider how to sign your binaries on your local machine before targeting automated build systems, just so you know if there are any special requirements. For instance, [here](https://www.ssl.com/how-to/using-your-code-signing-certificate/) is SSL.com's code signing guide for Windows. If you know how to sign locally, it will be easier to troubleshoot any potential issues in a CI environment. For instance, SSL.com code signing certificates require the `/tr` flag for [SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool) while other providers may only need the `/t` flag for providing the timestamping server. Popular GitHub Actions for signing Windows binaries like [this one](https://github.com/Dana-Prajea/code-sign-action) does not support the `/tr` flag on SignTool.exe. Therefore this guide will focus on signing our app manually with PowerShell commands, but you can use actions like the [code-sign-action](https://github.com/Dana-Prajea/code-sign-action) Action if you prefer. + +First off, let's make sure we are able to build our Wails app in our GitHub CI. Here is a small workflow template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend manually here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +Next we need to give the GitHub workflow access to our signing certificate. This is done by encoding your .pfx or .p12 certificate into a base64 string. To do this in PowerShell, you can use the following command assuming your certificate is called 'my-cert.p12': + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +You should now have your .txt file with the base64 encoded certificate. It should start with _-----BEGIN CERTIFICATE-----_ and end with _-----END CERTIFICATE-----_. Now you need to make two action secrets on GitHub. Navigate to _Settings -> Secrets -> Actions_ and create the two following secrets: + +- **WIN_SIGNING_CERT** with the contents of your base64 encoded certificate text. +- **WIN_SIGNING_CERT_PASSWORD** with the contents of your certificate password. + +Now we're ready to implement the signing in our workflow using one of the two methods: + +### Method 1: signing with commands + +This method uses PowerShell commands to sign our app, and leaves you control over the entire signing process. + +After the `"Build Wails app"` step, we can add the following step to our workflow: + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +This script creates a new directory for your certificate file, creates the certificate file from our base64 secret, converts it to a .pfx file, and finally signs the binary. The following variables needs to be replaced in the last line: + +- **signing algorithm**: usually sha256. +- **timestamping server**: URL to the timestamping server to use with your certificate. +- **path to binary**: path to the binary you want to sign. + +Given that our Wails config has `outputfilename` set to "app.exe" and that we have a certificate from SSL.com, this would be our workflow: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### Method 2: automatically signing with Action + +It is possible to use a Windows code signing Action like [this](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) one, but note it requires a SHA1 hash for the certificate and a certificate name. View an example of how to configure it on the Action's [marketplace](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate). + +--- + +## MacOS + +First off you need your code signing certificate from Apple. If you do not have one, a simple Google search will help you acquire one. Once you have your certificate, you need to export it and encode it to base64. [This tutorial](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions) shows you how to do that in an easy manner. Once you have exported your .p12 certificate file, you can encode it to base64 as seen in the tutorial with the following command: + +```bash +base64 Certificates.p12 | pbcopy +``` + +Now you're ready to create some GitHub project secrets, just as with Windows: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64** with the contents of your newly copied base64 certificate. +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD** with the contents of your certificate password. +- **APPLE_PASSWORD** with the contents of an App-Specific password to your Apple-ID account which you can generate [here](https://appleid.apple.com/account/manage). + +Let's make sure we are able to build our Wails app in our GitHub Action workflow. Here is a small template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +For code signing on macOS, [gon](https://github.com/mitchellh/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and will be used in this guide. + +After the `Build Wails app` step, add the following to the workflow: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +Now we need to configure some gon config files in our `build/darwin` directory: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +Where `source` is your Wails binary, `bundle_id` is your bundle ID, `apple_id` contains your Apple ID username and App-Specific password which you created earlier, and `sign.application_identity` is your identity which you can find by running the following command: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +In this file you configure the entitlements you need for you app, e.g. camera permissions if your app uses the camera. Read more about entitlements [here](https://developer.apple.com/documentation/bundleresources/entitlements). + +Make sure you have updated your `Info.plist` file with the same bundle ID as you entered in `gon-sign.json`. Here's an example `Info.plist` file: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +Now we're ready to add the signing step in our workflow after building the Wails app: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +Please note that signing binaries with Apple could take anywhere from minutes to hours. + +## Combined workflow file: + +Here is our GitHub workflow file with Windows + macOS combined: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# End notes + +This guide inspired by the RiftShare project and its workflow, which is highly recommended to check out [here](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml). diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx new file mode 100644 index 00000000000..4651c422ed1 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +This guide will go into: + +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. + +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. + +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. + +- `npm create svelte@latest frontend` + +##### Modify wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. + +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Install/remove dependencies using your favorite package manager. + +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js + +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. + +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. + +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. + +- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! + +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html + +``` + +... + + + +... + +``` + +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. + +- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling + +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions + +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/templates.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/templates.mdx new file mode 100644 index 00000000000..790e3107f04 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/templates.mdx @@ -0,0 +1,97 @@ +# Templates + +Wails generates projects from pre-created templates. In v1, this was a difficult to maintain set of projects that were subject to going out of date. In v2, to empower the community, a couple of new features have been added for templates: + +- Ability to generate projects from [Remote Templates](../reference/cli.mdx#remote-templates) +- Tooling to help create your own templates + +## Creating Templates + +To create a template, you can use the `wails generate template` command. To generate a default template, run: + +`wails generate template -name mytemplate` + +This creates the directory "mytemplate" with default files: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### Template Overview + +The default template consists of the following files and directories: + +| Filename / Dir | Description | +| --------------- | -------------------------------------------- | +| NEXTSTEPS.md | Instructions on how to complete the template | +| README.md | The README published with the template | +| app.tmpl.go | `app.go` template file | +| frontend/ | The directory containing frontend assets | +| go.mod.tmpl | `go.mod` template file | +| main.tmpl.go | `main.go` template file | +| template.json | The template metadata | +| wails.tmpl.json | `wails.json` template file | + +At this point it is advisable to follow the steps in `NEXTSTEPS.md`. + +## Creating a Template from an Existing Project + +It's possible to create a template from an existing frontend project by passing the path to the project when generating the template. We will now walk through how to create a Vue 3 template: + +- Install the vue cli: `npm install -g @vue/cli` +- Create the default project: `vue create vue3-base` + - Select `Default (Vue 3) ([Vue 3] babel, eslint)` +- After the project has been generated, run: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- The template may now be customised as specified in the `NEXTSTEPS.md` file +- Once the files are ready, it can be tested by running: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- To test the new project, run: `cd my-vue3-project` then `wails build` +- Once the project has compiled, run it: `.\build\bin\my-vue3-project.exe` +- You should have a fully functioning Vue3 application: + +```mdx-code-block +
+ +
+``` + +## Publishing Templates + +Publishing a template is simply pushing the files to GitHub. The following best practice is encouraged: + +- Remove any unwanted files and directories (such as `.git`) from your frontend directory +- Ensure that `template.json` is complete, especially `helpurl` +- Push the files to GitHub +- Create a PR on the [Community Templates](../community/templates.mdx) page +- Announce the template on the [Template Announcement](https://github.com/wailsapp/wails/discussions/825) discussion board diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx new file mode 100644 index 00000000000..1c3b6a9cc52 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# Troubleshooting + +An assortment of troubleshooting tips. + +## The `wails` command appears to be missing? + +If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment variable. You will also normally need to close and reopen any open command prompts so that changes to the environment made by the installer are reflected at the command prompt. + +## My application is displaying a white/blank screen + +Check that your application includes the assets from the correct directory. In your `main.go` file, you will have something similar to the following code: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +Check that `frontend/dist` contains your application assets. + +### Mac + +If this happens on Mac, try adding the following to your `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Mac application not valid + +If your built application looks like this in finder: + +```mdx-code-block +

+ +

+``` + +it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to the `build/darwin` directory. + +## My application is not displaying the correct icon in Windows Explorer + +If your application is not displaying the correct icon, try deleting the hidden `IconCache.db` file located in the `C:\Users\<your username>\AppData\Local` directory. This will force Windows to rebuild the icon cache. + +Source: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## Cannot call backend method from frontend with variadic arguments + +If you have a backend method defined with variadic parameters, eg: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +calling this method from the frontend like this will fail: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Workaround: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Credit: https://github.com/wailsapp/wails/issues/1186 + +## I'm having getting proxy errors when trying to install Wails + +If you are getting errors like this: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +it's probably because the official Go Proxy is being blocked (Users in China have reported this). The solution is to set up the proxy manually, eg: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +Source: https://github.com/wailsapp/wails/issues/1233 + +## The generated TypeScript doesn't have the correct types + +Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, it is possible to specify what types should be generated using the `ts_type` struct tag. For more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). + +## When I navigate away from `index.html`, I am unable to call methods on the frontend + +If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding the following imports to the `` section of any new page you navigate to: + +```html + + + + +``` + +Source: https://github.com/wailsapp/wails/discussions/1512 + +## I get `too many open files` errors on my Mac when I run `wails dev` + +By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. This limit can be increased by running: `ulimit -n 1024` in the terminal. + +FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). + +## My Mac app gives me weird compilation errors + +A few users have reported seeing compilation errors such as the following: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. + +If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: + +`xcode-select -p` + +If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" + +It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine that may have different versions of Node installed, you may not be able to run your application. + +If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. + +## Build process stuck on "Generating bindings" + +Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/vscode.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/vscode.mdx new file mode 100644 index 00000000000..ed258656d63 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +This page is for miscellaneous tips and tricks when using Visual Studio Code with Wails. + +## Vetur Configuration + +Many thanks to [@Lyimmi](https://github.com/Lyimmi) for this tip. Originally posted [here](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349). + +Vetur is a popular plugin for Visual Studio Code that provides syntax highlighting and code completion for Vue projects. When loading a Wails project in VSCode, Vetur will throw an error as it is expecting to find the frontend project in the root directory. To fix this, you can do the following: + +Create a file named `vetur.config.js` in the project's root. + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +Next, configure `frontend/tsconfig.json`: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +This should enable you to now use Vetur as expected. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows-installer.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows-installer.mdx new file mode 100644 index 00000000000..2a360bb1cf5 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# NSIS installer + +```mdx-code-block +

+ +
+

+``` + +Wails supports generating Windows installers using the [NSIS installer](https://nsis.sourceforge.io/). + +## Installing NSIS + +### Windows + +The installer is available on the [NSIS Download](https://nsis.sourceforge.io/Download) page. + +Nếu bạn sử dụng chocolatey package manager, chạy đoạn mã lệnh sau: + +``` +choco install nsis +``` + +Nếu bạn cài đặt NSIS một cách thủ công, thì bạn phải thêm thư mục _Bin_, thư mục này chứa tệp tin `makensis.exe`, trong gói cài đặt NSIS vào đường dẫn thư mục gốc của bạn. [Đây](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/) là một ví dụ cụ thể về cách thức thêm đường dẫn trên Windows. + +### Linux + +Gói `NSIS` sẽ có sẵn thông qua quá trình phân phối của package manager. + +### MacOS + +NSIS sẽ có sẵn để cài đặt thông qua homebrew: `brew install nsis`. + +## Khởi Tạo Trình Cài Đặt + +Khi một dư án mới được khởi tạo, Wails sẽ tự động thiết lập các tệp cấu hình cho NSIS trong thư mục `build/windows/installer`. Các dữ liệu của cấu hình được đọc trong tệp `installer/info.json` và các cấu hình đó được sử dụng trong phần Info của file dự án `wails.json`: + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +Để khởi tạo trình cài đặt cho ứng dụng, bạn hãy dùng thêm lệnh `-nsis` phía sau lệnh `wails build`: + +``` +wails build -nsis +``` + +Trình cài đặt giờ đay đã sẵn sàng để sử dụng trong thue mục `build/bin`. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows.mdx new file mode 100644 index 00000000000..821808c0b8d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +This page has miscellaneous guides related to developing Wails applications for Windows. + +## Handling the WebView2 Runtime Dependency + +Wails applications built for Windows have a runtime requirement on the Microsoft [WebView2 Runtime](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). Windows 11 will have this installed by default, but some machines won't. Wails offers an easy approach to dealing with this dependency. + +By using the `-webview2` flag when building, you can decide what your application will do when a suitable runtime is not detected (including if the installed runtime is too old). The four options are: + +1. Download +2. Embed +3. Browser +4. Error + +### Download + +This option will prompt the user that no suitable runtime has been found and then offer to download and run the official bootstrapper from Microsoft's WebView2 site. If the user proceeds, the official bootstrapper will be downloaded and run. + +### Embed + +This option embeds the official bootstrapper within the application. If no suitable runtime has been found, the application will offer to run the bootstrapper. This adds ~150k to the binary size. + +### Browser + +This option will prompt the user that no suitable runtime has been found and then offer to open a browser to the official WebView2 page where the bootstrapper can be downloaded and installed. The application will then exit, leaving the installation up to the user. + +### Error + +If no suitable runtime is found, an error is given to the user and no further action taken. + +## Fixed version runtime + +Another way of dealing with webview2 dependency is shipping it yourself. You can download [fixed version runtime](https://developer.microsoft.com/microsoft-edge/webview2/#download-section) and bundle or download it with your application. + +Also, you should specify path to fixed version of webview2 runtime in the `windows.Options` structure when launching wails. + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +Note: When `WebviewBrowserPath` is specified, `error` strategy will be forced in case of minimal required version mismatch or invalid path to a runtime. + +## Spawning other programs + +When spawning other programs, such as scripts, you will see the window appear on the screen. To hide the window, you can use the following code: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +Solution provided by [sithembiso](https://github.com/sithembiso) on the [discussions board](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172). diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/howdoesitwork.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/howdoesitwork.mdx new file mode 100644 index 00000000000..44fa130cc74 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# How does it work? + +A Wails application is a standard Go application, with a webkit frontend. The Go part of the application consists of the application code and a runtime library that provides a number of useful operations, like controlling the application window. The frontend is a webkit window that will display the frontend assets. Also available to the frontend is a JavaScript version of the runtime library. Finally, it is possible to bind Go methods to the frontend, and these will appear as JavaScript methods that can be called, just as if they were local JavaScript methods. + +```mdx-code-block +
+ +
+``` + +## The Main Application + +### Overview + +The main application consists of a single call to `wails.Run()`. It accepts the application configuration which describes the size of the application window, the window title, what assets to use, etc. A basic application might look like this: + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### Options rundown + +This example has the following options set: + +- `Title` - The text that should appear in the window's title bar +- `Width` & `Height` - The dimensions of the window +- `Assets` - The application's frontend assets +- `OnStartup` - A callback for when the window is created and is about to start loading the frontend assets +- `OnShutdown` - A callback for when the application is about to quit +- `Bind` - A slice of struct instances that we wish to expose to the frontend + +A full list of application options can be found in the [Options Reference](reference/options). + +#### Assets + +The `Assets` option is mandatory as you can't have a Wails application without frontend assets. Those assets can be any files you would expect to find in a web application - html, js, css, svg, png, etc. **There is no requirement to generate asset bundles** - plain files will do. When the application starts, it will attempt to load `index.html` from your assets and the frontend will essentially work as a browser from that point on. It is worth noting that there is no requirement on where in the `embed.FS` the files live. It is likely that the embed path uses a nested directory relative to your main application code, such as `frontend/dist`: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +At startup, Wails will iterate the embedded files looking for the directory containing `index.html`. All other assets will be loaded relative to this directory. + +As production binaries use the files contained in `embed.FS`, there are no external files required to be shipped with the application. + +When running in development mode using the `wails dev` command, the assets are loaded off disk, and any changes result in a "live reload". The location of the assets will be inferred from the `embed.FS`. + +More details can be found in the [Application Development Guide](guides/application-development.mdx). + +#### Application Lifecycle Callbacks + +Just before the frontend is about to load `index.html`, a callback is made to the function provided in [OnStartup](reference/options.mdx#onstartup). A standard Go context is passed to this method. This context is required when calling the runtime so a standard pattern is to save a reference to in this method. Just before the application shuts down, the [OnShutdown](reference/options.mdx#onshutdown) callback is called in the same way, again with the context. There is also an [OnDomReady](reference/options.mdx#ondomready) callback for when the frontend has completed loading all assets in `index.html` and is equivalent of the [`body onload`](https://www.w3schools.com/jsref/event_onload.asp) event in JavaScript. It is also possible to hook into the window close (or application quit) event by setting the option [OnBeforeClose](reference/options.mdx#onbeforeclose). + +#### Method Binding + +The `Bind` option is one of the most important options in a Wails application. It specifies which struct methods to expose to the frontend. Think of structs like "controllers" in a traditional web application. When the application starts, it examines the struct instances listed in the `Bind` field in the options, determines which methods are public (starts with an uppercase letter) and will generate JavaScript versions of those methods that can be called by the frontend code. + +:::info Note + +Wails requires that you pass in an _instance_ of the struct for it to bind it correctly + +::: + +In this example, we create a new `App` instance and then add this instance to the `Bind` option in `wails.Run`: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +You may bind as many structs as you like. Just make sure you create an instance of it and pass it in `Bind`: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +When you run `wails dev` (or `wails generate module`), a frontend module will be generated containing the following: + +- JavaScript bindings for all bound methods +- TypeScript declarations for all bound methods +- TypeScript definitions for all Go structs used as inputs or outputs by the bound methods + +This makes it incredibly simple to call Go code from the frontend, using the same strongly typed datastructures. + +## The Frontend + +### Overview + +The frontend is a collection of files rendered by webkit. It's like a browser and webserver in one. There is virtually[^1] no limit to which frameworks or libraries you can use. The main points of interaction between the frontend and your Go code are: + +- Calling bound Go methods +- Calling runtime methods + +### Calling bound Go methods + +When you run your application with `wails dev`, it will automatically generate JavaScript bindings for your structs in a directory called `wailsjs/go` (You can also do this by running `wails generate module`). The generated files mirror the package names in your application. In the example above, we bind `app`, which has one public method `Greet`. This will lead to the generation of the following files: + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +Here we can see that there is a `main` package that contains the JavaScript bindings for the bound `App` struct, as well as the TypeScript declaration file for those methods. To call `Greet` from our frontend, we simply import the method and call it like a regular JavaScript function: + +```javascript +// ... +import { Greet } from "../wailsjs/go/main/App"; + +function doGreeting(name) { + Greet(name).then((result) => { + // Do something with result + }); +} +``` + +The TypeScript declaration file gives you the correct types for the bound methods: + +```ts +export function Greet(arg1: string): Promise; +``` + +The generated methods return a Promise. A successful call will result in the first return value from the Go call to be passed to the `resolve` handler. An unsuccessful call is when a Go method that has an error type as it's second return value, passes an error instance back to the caller. This is passed back via the `reject` handler. In the example above, `Greet` only returns a `string` so the JavaScript call will never reject - unless invalid data is passed to it. + +All data types are correctly translated between Go and JavaScript. Even structs. If you return a struct from a Go call, it will be returned to your frontend as a JavaScript class. + +:::info Note + +Struct fields _must_ have a valid `json` tag to be included in the generated TypeScript. + +Anonymous nested structs are not supported at this time. + +::: + +It is possible to send structs back to Go. Any JavaScript map/class passed as an argument that is expecting a struct, will be converted to that struct type. To make this process a lot easier, in `dev` mode, a TypeScript module is generated, defining all the struct types used in bound methods. Using this module, it's possible to construct and send native JavaScript objects to the Go code. + +There is also support for Go methods that use structs in their signature. All Go structs specified by a bound method (either as parameters or return types) will have TypeScript versions auto generated as part of the Go code wrapper module. Using these, it's possible to share the same data model between Go and JavaScript. + +Example: We update our `Greet` method to accept a `Person` instead of a string: + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +The `wailsjs/go/main/App.js` file will still have the following code: + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +But the `wailsjs/go/main/App.d.ts` file will be updated with the following code: + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +As we can see, the "main" namespace is imported from a new "models.ts" file. This file contains all the struct definitions used by our bound methods. In this example, this is a `Person` struct. If we look at `models.ts`, we can see how the models are defined: + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +So long as you have TypeScript as part of your frontend build configuration, you can use these models in the following way: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +The combination of generated bindings and TypeScript models makes for a powerful development environment. + +More information on Binding can be found in the [Binding Methods](guides/application-development.mdx#binding-methods) section of the [Application Development Guide](guides/application-development.mdx). + +### Calling runtime methods + +The JavaScript runtime is located at `window.runtime` and contains many methods to do various tasks such as emit an event or perform logging operations: + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +More details about the JS runtime can be found in the [Runtime Reference](reference/runtime/intro). + +[^1]: There is a very small subset of libraries that use features unsupported in WebViews. There are often alternatives and workarounds for such cases. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx new file mode 100644 index 00000000000..a8bb301fb23 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# Introduction + +Wails is a project that enables you to write desktop apps using Go and web technologies. + +Consider it a lightweight and fast Electron alternative for Go. You can easily build applications with the flexibility and power of Go, combined with a rich, modern frontend. + +### Features + +- Native Menus, Dialogs, Theming and Translucency +- Windows, macOS and linux support +- Built in templates for Svelte, React, Preact, Vue, Lit and Vanilla JS +- Easily call Go methods from JavaScript +- Automatic Go struct to TypeScript model generation +- No CGO or external DLLs required on Windows +- Live development mode using the power of [Vite](https://vitejs.dev/) +- Powerful CLI to easily Create, Build and Package applications +- A rich [runtime library](/docs/reference/runtime/intro) +- Applications built with Wails are Apple & Microsoft Store compliant + +This is [varly](https://varly.app) - a desktop application for MacOS & Windows written using Wails. Not only does it look great, it uses native menus and translucency - everything you'd expect from a modern native app. + +```mdx-code-block +

+ + + +

+``` + +### Quick Start Templates + +Wails comes with a number of pre-configured templates that allow you to get your application up and running quickly. There are templates for the following frameworks: Svelte, React, Vue, Preact, Lit and Vanilla. There are both JavaScript and TypeScript versions for each template. + +### Native Elements + +Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. + +**It does not embed a browser**, so it is resource efficient. Instead, it uses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. + +### Go & JavaScript Interoperability + +Wails automatically makes your Go methods available to JavaScript, so you can call them by name from your frontend! It even generates TypeScript models for the structs used by your Go methods, so you can pass the same data structures between Go and JavaScript. + +### Runtime Library + +Wails provides a runtime library, for both Go and JavaScript, that handles a lot of the things modern applications need, like Eventing, Logging, Dialogs, etc. + +### Live Development Experience + +#### Automatic Rebuilds + +When you run your application in "dev" mode, Wails will build your application as a native desktop application, but will read your assets from disk. It will detect any changes to your Go code and automatically rebuild and relaunch your application. + +#### Automatic Reloads + +When changes to your application assets are detected, your running application will "reload", reflecting your changes almost immediately. + +#### Develop your application in a Browser + +If you prefer to debug and develop in a browser then Wails has you covered. The running application also has a webserver that will run your application in any browser that connects to it. It will even refresh when your assets change on disk. + +### Production-ready Native Binaries + +When you're ready to do the final build of your application, the CLI will compile it down to a single executable, with all the assets bundled into it. On Windows and MacOS, it is possible to create a native package for distribution. The assets used in packaging (icon, info.plist, manifest file, etc) are part of your project and may be customised, giving you total control over how your applications are built. + +### Tooling + +The Wails CLI provides a hassle-free way to generate, build and bundle your applications. It will do the heavy lifting of creating icons, compiling your application with optimal settings and delivering a distributable, production ready binary. Choose from a number of starter templates to get up and running quickly! diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/cli.mdx new file mode 100644 index 00000000000..c7aea10a64d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -0,0 +1,240 @@ +--- +sidebar_position: 2 +--- + +# CLI + +The Wails CLI has a number of commands that are used for managing your projects. All commands are run in the following way: + +`wails ` + +## init + +`wails init` is used for generating projects. + +| Flag | Description | Default | +|:------------------ |:----------------------------------------------------------------------------------------------------------------------- |:-------------------:| +| -n "project name" | Name of the project. **Mandatory**. | | +| -d "project dir" | Project directory to create | Name of the project | +| -g | Initialise git repository | | +| -l | List available project templates | | +| -q | Suppress output to console | | +| -t "template name" | The project template to use. This can be the name of a default template or a URL to a remote template hosted on github. | vanilla | +| -ide | Generate IDE project files | | +| -f | Force build application | false | + +Example: `wails init -n test -d mytestproject -g -ide vscode -q` + +This will generate a a project called "test" in the "mytestproject" directory, initialise git, generate vscode project files and do so silently. + +More information on using IDEs with Wails can be found [here](../guides/ides.mdx). + +### Remote Templates + +Remote templates (hosted on GitHub) are supported and can be installed by using the template's project URL. + +Example: `wails init -n test -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +A list of community maintained templates can be found [here](../community/templates.mdx) + +:::warning Attention + +**The Wails project does not maintain, is not responsible nor liable for 3rd party templates!** + +If you are unsure about a template, inspect `package.json` and `wails.json` for what scripts are run and what packages are installed. + +::: + +## build + +`wails build` is used for compiling your project to a production-ready binary. + +| Flag | Description | Default | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Cleans the `build/bin` directory | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Force build application | | +| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -m | Skip mod tidy before compile | | +| -nopackage | Do not package application | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o filename | Output filename | | +| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | +| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Build with Go's race detector | | +| -s | Skip building the frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | +| -trimpath | Remove all file system paths from the resulting executable. | | +| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | +| -upx | Compress final binary using "upx" | | +| -upxflags | Flags to pass to upx | | +| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | +| -windowsconsole | Keep the console window for Windows builds | | + +For a detailed description of the `webview2` flag, please refer to the [Windows](../guides/windows.mdx) Guide. + +If you prefer to build using standard Go tooling, please consult the [Manual Builds](../guides/manual-builds.mdx) guide. + +Example: + +`wails build -clean -o myproject.exe` + +:::info + +On Mac, the application will be bundled with `Info.plist`, not `Info.dev.plist`. + +::: + +:::info UPX on Apple Silicon + +There are [issues](https://github.com/upx/upx/issues/446) with using UPX with Apple Silicon. + +::: + +:::info UPX on Windows + +Some Antivirus vendors false positively mark `upx` compressed binaries as virus, see [issue](https://github.com/upx/upx/issues/437). + +::: + +### Platforms + +Supported platforms are: + +| Platform | Description | +|:---------------- |:--------------------------------------------- | +| darwin | MacOS + architecture of build machine | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | MacOS AMD64+ARM64 universal application | +| windows | Windows 10/11 + architecture of build machine | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + architecture of build machine | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## doctor + +`wails doctor` will run diagnostics to ensure that your system is ready for development. + +Example: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## dev + +`wails dev` is used to run your application in a "live development" mode. This means: + +- The application's `go.mod` will be updated to use the same version of Wails as the CLI +- The application is compiled and run automatically +- A watcher is started and will trigger a rebuild of your dev app if it detects changes to your go files +- A webserver is started on `http://localhost:34115` which serves your application (not just frontend) over http. This allows you to use your favourite browser development extensions +- All application assets are loaded from disk. If they are changed, the application will automatically reload (not rebuild). All connected browsers will also reload +- A JS module is generated that provides the following: +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- A second JS module is generated that provides a wrapper + TS declaration for the runtime +- On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. + +| Flag | Description | Default | +|:---------------------------- |:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:--------------------- | +| -appargs "args" | Arguments passed to the application in shell style | | +| -assetdir "./path/to/assets" | Serve assets from the given directory instead of using the provided asset FS | Value in `wails.json` | +| -browser | Opens a browser to `http://localhost:34115` on startup | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debounce | The time to wait for reload after an asset change is detected | 100 (milliseconds) | +| -devserver "host:port" | The address to bind the wails dev server to | "localhost:34115" | +| -extensions | Extensions to trigger rebuilds (comma separated) | go | +| -forcebuild | Force build of application | | +| -frontenddevserverurl "url" | Use 3rd party dev server url to serve assets, EG Vite | "" | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -loglevel "loglevel" | Loglevel to use - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | Turn off colour cli output | false | +| -noreload | Disable automatic reload when assets change | | +| -nosyncgomod | Do not sync go.mod with the Wails version | false | +| -race | Build with Go's race detector | false | +| -reloaddirs | Additional directories to trigger reloads (comma separated) | Value in `wails.json` | +| -s | Skip building the frontend | false | +| -save | Saves the given `assetdir`, `reloaddirs`, `wailsjsdir`, `debounce`, `devserver` and `frontenddevserverurl` flags in `wails.json` to become the defaults for subsequent invocations. | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to compiler (quoted and space separated) | | +| -v | Verbosity level (0 - silent, 1 - standard, 2 - verbose) | 1 | +| -wailsjsdir | The directory to generate the generated Wails JS modules | Value in `wails.json` | + +Example: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +This command will do the following: + +- Build the application and run it (more details [here](../guides/manual-builds.mdx) +- Generate the Wails JS modules in `./frontend/src` +- Watch for updates to files in `./frontend/dist` and reload on any change +- Open a browser and connect to the application + +There is more information on using this feature with existing framework scripts [here](../guides/application-development.mdx#live-reloading). + +## generate + +### template + +Wails uses templates for project generation. The `wails generate template` command helps scaffold a template so that it may be used for generating projects. + +| Flag | Description | +|:---------------- |:------------------------------------------- | +| -name | The template name (Mandatory) | +| -frontend "path" | Path to frontend project to use in template | + +For more details on creating templates, consult the [Templates guide](../guides/templates.mdx). + +### module + +The `wails generate module` command allows you to manually generate the `wailsjs` directory for your application. + +## update + +`wails update` will update the version of the Wails CLI. + +| Flag | Description | +|:------------------ |:------------------------------------- | +| -pre | Update to latest pre-release version | +| -version "version" | Install a specific version of the CLI | + +## version + +`wails version` will simply output the current CLI version. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/menus.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/menus.mdx new file mode 100644 index 00000000000..ff9a2442281 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# Menus + +It is possible to add an application menu to Wails projects. This is achieved by defining a [Menu](#menu) struct and setting it in the [`Menu`](../reference/options.mdx#menu) application config, or by calling the runtime method [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). + +An example of how to create a menu: + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +It is also possible to dynamically update the menu, by updating the menu struct and calling [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). + +The example above uses helper methods, however it's possible to build the menu structs manually. + +## Menu + +A Menu is a collection of MenuItems: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +For the Application menu, each MenuItem represents a single menu such as "Edit". + +A simple helper method is provided for building menus: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +This makes the layout of the code more like that of a menu without the need to add the menu items manually after creating them. Alternatively, you can just create the menu items and add them to the menu manually. + +## MenuItem + +A MenuItem represents an item within a Menu. + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| Field | Type | Notes | +| ----------- | ------------------------------------ | ------------------------------------------------------------- | +| Label | string | The menu text | +| Accelerator | [\*keys.Accelerator](#accelerator) | Key binding for this menu item | +| Type | [Type](#type) | Type of MenuItem | +| Disabled | bool | Disables the menu item | +| Hidden | bool | Hides this menu item | +| Checked | bool | Adds check to item (Checkbox & Radio types) | +| SubMenu | [\*Menu](#menu) | Sets the submenu | +| Click | [Callback](#callback) | Callback function when menu clicked | +| Role | string | Defines a [role](#role) for this menu item. Mac only for now. | + +### Accelerator + +Accelerators (sometimes called keyboard shortcuts) define a binding between a keystroke and a menu item. Wails defines an Accelerator as a combination or key + [Modifier](#modifier). They are available in the `"github.com/wailsapp/wails/v2/pkg/menu/keys"` package. + +Example: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +Keys are any single character on a keyboard with the exception of `+`, which is defined as `plus`. Some keys cannot be represented as characters so there are a set of named characters that may be used: + +| | | | | +|:-----------:|:-----:|:-----:|:---------:| +| `backspace` | `f1` | `f16` | `f31` | +| `tab` | `f2` | `f17` | `f32` | +| `return` | `f3` | `f18` | `f33` | +| `enter` | `f4` | `f19` | `f34` | +| `escape` | `f5` | `f20` | `f35` | +| `left` | `f6` | `f21` | `numlock` | +| `right` | `f7` | `f22` | | +| `up` | `f8` | `f23` | | +| `down` | `f9` | `f24` | | +| `space` | `f10` | `f25` | | +| `delete` | `f11` | `f36` | | +| `home` | `f12` | `f37` | | +| `end` | `f13` | `f38` | | +| `page up` | `f14` | `f39` | | +| `page down` | `f15` | `f30` | | + +Wails also supports parsing accelerators using the same syntax as Electron. This is useful for storing accelerators in config files. + +Example: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### Modifier + +The following modifiers are keys that may be used in combination with the accelerator key: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +A number of helper methods are available to create Accelerators using modifiers: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +Modifiers can be combined using `keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)`: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### Type + +Each menu item must have a type and there are 5 types available: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +For convenience, helper methods are provided to quickly create a menu item: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +You can also create menu items directly on a menu by using the "Add" helpers: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +A note on radio groups: A radio group is defined as a number of radio menu items that are next to each other in the menu. This means that you do not need to group items together as it is automatic. However, that also means you cannot have 2 radio groups next to each other - there must be a non-radio item between them. + +### Callback + +Each menu item may have a callback that is executed when the item is clicked: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +The function is given a `CallbackData` struct which indicates which menu item triggered the callback. This is useful when using radio groups that may share a callback. + +### Role + +:::info Roles + +Roles are currently supported on Mac only. + +::: + +A menu item may have a role, which is essentially a pre-defined menu item. We currently support the following roles: + +| Role | Description | +| ------------ | ------------------------------------------------------------------------ | +| AppMenuRole | The standard Mac application menu. Can be created using `menu.AppMenu()` | +| EditMenuRole | The standard Mac edit menu. Can be created using `menu.EditMenu()` | diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/options.mdx new file mode 100644 index 00000000000..46d1c071b60 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -0,0 +1,853 @@ +--- +sidebar_position: 3 +--- + +# Options + +## Application Options + +The `Options.App` struct contains the application configuration. It is passed to the `wails.Run()` method: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // User messages that can be customised + Messages *windows.Messages + // OnSuspend is called when Windows enters low power mode + OnSuspend func() + // OnResume is called when Windows resumes from low power mode + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### Title + +The text shown in the window's title bar. + +Name: Title
Type: `string` + +### Width + +The initial width of the window. + +Name: Width
Type: `int`
Default: 1024. + +### Height + +The initial height of the window. + +Name: Height
Type: `int`
Default: 768 + +### DisableResize + +By default, the main window is resizable. Setting this to `true` will keep it a fixed size. + +Name: DisableResize
Type: `bool` + +### Fullscreen + +Deprecated: Please use [WindowStartState](#windowstartstate). + +### WindowStartState + +Defines how the window should present itself at startup. + +| Value | Win | Mac | Lin | +| ---------- | --- | --- | --- | +| Fullscreen | ✅ | ✅ | ✅ | +| Maximised | ✅ | ✅ | ✅ | +| Minimised | ✅ | ❌ | ✅ | + +Name: WindowStartState
Type: `options.WindowStartState` + +### Frameless + +When set to `true`, the window will have no borders or title bar. Also see [Frameless Windows](../guides/frameless.mdx). + +Name: Frameless
Type: `bool` + +### MinWidth + +This sets the minimum width for the window. If the value given in `Width` is less than this value, the window will be set to `MinWidth` by default. + +Name: MinWidth
Type: `int` + +### MinHeight + +This sets the minimum height for the window. If the value given in `Height` is less than this value, the window will be set to `MinHeight` by default. + +Name: MinHeight
Type: `int` + +### MaxWidth + +This sets the maximum width for the window. If the value given in `Width` is more than this value, the window will be set to `MaxWidth` by default. + +Name: MaxWidth
Type: `int` + +### MaxHeight + +This sets the maximum height for the window. If the value given in `Height` is more than this value, the window will be set to `MaxHeight` by default. + +Name: MaxHeight
Type: `int` + +### StartHidden + +When set to `true`, the application will be hidden until [WindowShow](../reference/runtime/window.mdx#windowshow) is called. + +Name: StartHidden
Type: `bool` + +### HideWindowOnClose + +By default, closing the window will close the application. Setting this to `true` means closing the window will + +hide the window instead. + +Name: HideWindowOnClose
Type: `bool` + +### BackgroundColour + +This value is the default background colour of the window. Example: options.NewRGBA(255,0,0,128) - Red at 50% transparency + +Name: BackgroundColour
Type: `*options.RGBA`
Default: white + +### AlwaysOnTop + +Indicates that the window should stay above other windows when losing focus. + +Name: AlwaysOnTop
Type: `bool` + +### Assets + +Deprecated: Please use Assets on [AssetServer specific options](#assetserver). + +### AssetsHandler + +Deprecated: Please use AssetsHandler on [AssetServer specific options](#assetserver). + +### AssetServer + +This defines AssetServer specific options. It allows to customize the AssetServer with static assets, serving assets dynamically with an `http.Handler` or hook into the request chain with an `assetserver.Middleware`. + +Not all features of an `http.Request` are currently supported, please see the following feature matrix: + +| Feature | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +Name: AssetServer
Type: `*assetserver.Options` + +#### Assets + +The static frontend assets to be used by the application. + +A GET request is first tried to be served from this `fs.FS`. If the `fs.FS` returns `os.ErrNotExist` for that file, the request handling will fallback to the [Handler](#handler) and tries to serve the GET request from it. + +If set to nil, all GET requests will be forwarded to [Handler](#handler). + +Name: Assets
Type: `fs.FS` + +#### Handler + +The assets handler is a generic `http.Handler` for fallback handling of assets that can't be found. + +The handler will be called for every GET request that can't be served from [Assets](#assets), due to `os.ErrNotExist`. Furthermore all non GET requests will always be served from this Handler. If not defined, the result is the following in cases where the Handler would have been called: + +- GET request: `http.StatusNotFound` +- Other request: `http.StatusMethodNotAllowed` + +NOTE: When used in combination with a Frontend DevServer there might be limitations, eg. Vite serves the index.html on every path, that does not contain a file extension. + +Name: AssetsHandler
Type: `http.Handler` + +#### Middleware + +Middleware is a HTTP Middleware which allows to hook into the AssetServer request chain. It allows to skip the default request handler dynamically, e.g. implement specialized Routing etc. The Middleware is called to build a new `http.Handler` used by the AssetSever and it also receives the default handler used by the AssetServer as an argument. + +If not defined, the default AssetServer request chain is executed. + +Name: Middleware
Type: `assetserver.Middleware` + +### Menu + +The menu to be used by the application. More details about Menus in the [Menu Reference](../reference/runtime/menu.mdx). + +:::note + +On Mac, if no menu is specified, a default menu will be created. + +::: + +Name: Menu
Type: `*menu.Menu` + +### Logger + +The logger to be used by the application. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: Logger
Type: `logger.Logger`
Default: Logs to Stdout + +### LogLevel + +The default log level. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevel
Type: `logger.LogLevel`
Default: `Info` in dev mode, `Error` in production mode + +### LogLevelProduction + +The default log level for production builds. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevelProduction
Type: `logger.LogLevel`
Default: `Error` + +### OnStartup + +This callback is called after the frontend has been created, but before `index.html` has been loaded. It is given the application context. + +Name: OnStartup
Type: `func(ctx context.Context)` + +### OnDomReady + +This callback is called after the frontend has loaded `index.html` and its resources. It is given the application context. + +Name: OnDomReady
Type: `func(ctx context.Context)` + +### OnShutdown + +This callback is called after the frontend has been destroyed, just before the application terminates. It is given the application context. + +Name: OnShutdown
Type: `func(ctx context.Context)` + +### OnBeforeClose + +If this callback is set, it will be called when the application is about to quit, either by clicking the window close button or calling `runtime.Quit`. Returning true will cause the application to continue, false will continue shutdown as normal. This is good for confirming with the user that they wish to exit the program. + +Example: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +Name: OnBeforeClose
Type: `func(ctx context.Context) bool` + +### CSSDragProperty + +Indicates the CSS property to use to identify which elements can be used to drag the window. Default: `--wails-draggable`. + +Name: CSSDragProperty
Type: `string` + +### CSSDragValue + +Indicates what value the `CSSDragProperty` style should have to drag the window. Default: `drag`. + +Name: CSSDragValue
Type: `string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenu enables the browser's default context-menu in production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : + +| CSS Style | Behavior | +| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `--default-contextmenu: auto;` | (**default**) will show the default context menu only if :
contentEditable is true OR text has been selected OR element is input or textarea | +| `--default-contextmenu: show;` | will always show the default context menu | +| `--default-contextmenu: hide;` | will always hide the default context menu | + +This rule is inherited like any normal CSS rule, so nesting works as expected. + +:::note +This filtering functionality is only enabled in production, so in development and in debug build, the full context-menu is always available everywhere. +::: + +:::warning +This filtering functionality is NOT a security measure, the developer should expect that the full context-menu could be leaked anytime which could contain commands like (Download image, Reload, Save webpage), if this is a concern, the developer SHOULD NOT enable the default context-menu. +::: + + +Name: EnableDefaultContextMenu
Type: `bool` + +### EnableFraudulentWebsiteDetection + +EnableFraudulentWebsiteDetection enables scan services for fraudulent content, such as malware or phishing attempts. These services might send information from your app like URLs navigated to and possibly other content to cloud services of Apple and Microsoft. + +Name: EnableFraudulentWebsiteDetection
Type: `bool` + +### ZoomFactor + +Name: ZoomFactor
Type: `float64` + +This defines the zoom factor for the WebView2. This is the option matching the Edge user activated zoom in or out. + +### IsZoomControlEnabled + +Name: IsZoomControlEnabled
Type: `bool` + +This enables the zoom factor to be changed by the user. Please note that the zoom factor can be set in the options while disallowing the user to change it at runtime (f.e. for a kiosk application or similar). + +### Bind + +A slice of struct instances defining methods that need to be bound to the frontend. + +Name: Bind
Type: `[]interface{}` + +### ErrorFormatter + +A function that determines how errors are formatted when returned by a JS-to-Go method call. The returned value will be marshalled as JSON. + +Name: ErrorFormatter
Type: `func (error) any` + +### Windows + +This defines [Windows specific options](#windows). + +Name: Windows
Type: `*windows.Options` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent). + +For Windows 11 versions before build 22621, this will use the [BlurBehind](https://learn.microsoft.com/en-us/windows/win32/dwm/blur-ovw) method for translucency, which can be slow. For Windows 11 versions after build 22621, this will enable the newer translucency types that are much faster. By default, the type of translucency used will be determined by Windows. To configure this, use the [BackdropType](#BackdropType) option. + +Name: WindowIsTranslucent
Type: `bool` + +#### BackdropType + +:::note + +Requires Windows 11 build 22621 or later. + +::: + +Sets the translucency type of the window. This is only applicable if [WindowIsTranslucent](#WindowIsTranslucent) is set to `true`. + +Name: BackdropType
Type `windows.BackdropType` + +The value can be one of the following: + +| Value | Description | +| ------- | ----------------------------------------------------------------------------------------- | +| Auto | Let Windows decide which backdrop to use | +| None | Do not use translucency | +| Acrylic | Use [Acrylic](https://learn.microsoft.com/en-us/windows/apps/design/style/acrylic) effect | +| Mica | Use [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) effect | +| Tabbed | Use Tabbed. This is a backdrop that is similar to Mica. | + +#### DisableWindowIcon + +Setting this to `true` will remove the icon in the top left corner of the title bar. + +Name: DisableWindowIcon
Type: `bool` + +#### DisableFramelessWindowDecorations + +Setting this to `true` will remove the window decorations in [Frameless](#Frameless) mode. This means there will be no 'Aero Shadow' and no 'Rounded Corners' shown for the window. Please note that 'Rounded Corners' are only supported on Windows 11. + +Name: DisableFramelessWindowDecorations
Type: `bool` + +#### WebviewUserDataPath + +This defines the path where the WebView2 stores the user data. If empty `%APPDATA%\[BinaryName.exe]` will be used. + +Name: WebviewUserDataPath
Type: `string` + +#### WebviewBrowserPath + +This defines the path to a directory with WebView2 executable files and libraries. If empty, webview2 installed in the system will be used. + +Important information about distribution of fixed version runtime: + +- [How to get and extract runtime](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [Known issues for fixed version](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [The path of fixed version of the WebView2 Runtime should not contain \Edge\Application\.](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +Name: WebviewBrowserPath
Type: `string` + +#### Theme + +Minimum Windows Version: Windows 10 2004/20H1 + +This defines the theme that the application should use: + +| Value | Description | +| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------- | +| SystemDefault | _Default_. The theme will be based on the system default. If the user changes their theme, the application will update to use the new setting | +| Dark | The application will use a dark theme exclusively | +| Light | The application will use a light theme exclusively | + +Name: Theme
Type: `windows.Theme` + +#### CustomTheme + +:::note + +Minimum Windows Version: Windows 10/11 2009/21H2 Build 22000 + +::: + +Allows you to specify custom colours for TitleBar, TitleText and Border for both light and dark mode, as well as when the window is active or inactive. + +Name: CustomTheme
Type: `windows.CustomTheme` + +##### CustomTheme type + +The CustomTheme struct uses `int32` to specify the colour values. These are in the standard(!) Windows format of: `0x00BBGGAA`. A helper function is provided to do RGB conversions into this format: `windows.RGB(r,g,b uint8)`. + +NOTE: Any value not provided will default to black. + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +Example: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### Messages + +A struct of strings used by the webview2 installer if a valid webview2 runtime is not found. + +Name: Messages
Type: `*windows.Messages` + +Customise this for any language you choose to support. + +#### ResizeDebounceMS + +ResizeDebounceMS is the amount of time to debounce redraws of webview2 when resizing the window. The default value (0) will perform redraws as fast as it can. + +Name: ResizeDebounceMS
Type: `uint16` + +#### OnSuspend + +If set, this function will be called when Windows initiates a switch to low power mode (suspend/hibernate) + +Name: OnSuspend
Type: `func()` + +#### OnResume + +If set, this function will be called when Windows resumes from low power mode (suspend/hibernate) + +Name: OnResume
Type: `func()` + +#### WebviewGpuIsDisabled + +Setting this to `true` will disable GPU hardware acceleration for the webview. + +Name: WebviewGpuIsDisabled
Type: `bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +This defines [Mac specific options](#mac). + +Name: Mac
Type: `*mac.Options` + +#### TitleBar + +The TitleBar struct provides the ability to configure the look and feel of the title bar. + +Name: TitleBar
Type: [`*mac.TitleBar`](#titlebar-struct) + +##### Titlebar struct + +The titlebar of the application can be customised by using the TitleBar options: + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| Name | Description | +| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| TitlebarAppearsTransparent | Makes the titlebar transparent. This has the effect of hiding the titlebar and the content fill the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | Hides the title of the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | Removes [NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/) from the style mask | +| FullSizeContent | Makes the webview fill the entire window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | Adds a default toolbar to the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | Removes the line beneath the toolbar. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +Preconfigured titlebar settings are available: + +| Setting | Example | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +Example: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +Click [here](https://github.com/lukakerr/NSWindowStyles) for some inspiration on customising the titlebar. + +#### Appearance + +Appearance is used to set the style of your app in accordance with Apple's [NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc) names. + +Name: Appearance
Type: [`mac.AppearanceType`](#appearance-type) + +##### Appearance type + +You can specify the application's [appearance](https://developer.apple.com/documentation/appkit/nsappearance?language=objc). + +| Value | Description | +| ----------------------------------------------------- | --------------------------------------------------------------- | +| DefaultAppearance | DefaultAppearance uses the default system value | +| NSAppearanceNameAqua | The standard light system appearance | +| NSAppearanceNameDarkAqua | The standard dark system appearance | +| NSAppearanceNameVibrantLight | The light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastAqua | A high-contrast version of the standard light system appearance | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | A high-contrast version of the standard dark system appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | A high-contrast version of the light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | A high-contrast version of the dark vibrant appearance | + +Example: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent) to make frosty-looking applications. + +Name: WindowIsTranslucent
Type: `bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### About + +This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. + +Name: About
Type: [`*mac.AboutInfo`](#about-struct) + +##### About struct + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +If these settings are provided, an "About" menu item will appear in the app menu (when using the `AppMenu` role). Given this configuration: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +The "About" menu item will appear in the app menu: + +```mdx-code-block +
+ +
+
+``` + +When clicked, that will open an about message box: + +```mdx-code-block +
+ +
+
+``` + +### Linux + +This defines [Linux specific options](#linux). + +Name: Linux
Type: `*linux.Options` + +#### Icon + +Sets up the icon representing the window. This icon is used when the window is minimized (also known as iconified). + +Name: Icon
Type: `[]byte` + +Some window managers or desktop environments may also place it in the window frame, or display it in other contexts. On others, the icon is not used at all, so your mileage may vary. + +NOTE: Gnome on Wayland at least does not display this icon. To have a application icon there, a `.desktop` file has to be used. On KDE it should work. + +The icon should be provided in whatever size it was naturally drawn; that is, don’t scale the image before passing it. Scaling is postponed until the last minute, when the desired final size is known, to allow best quality. + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Some window managers may ignore it, or result in a black window. + +Name: WindowIsTranslucent
Type: `bool` + +#### WebviewGpuPolicy + +This option is used for determining the webview's hardware acceleration policy. + +Name: WebviewGpuPolicy
Type: [`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
Default: `WebviewGpuPolicyAlways` + +##### WebviewGpuPolicy type + +| Value | Description | +| ------------------------ | -------------------------------------------------------------------- | +| WebviewGpuPolicyAlways | Hardware acceleration is always enabled | +| WebviewGpuPolicyOnDemand | Hardware acceleration is enabled/disabled as request by web contents | +| WebviewGpuPolicyNever | Hardware acceleration is always disabled | + +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` property differs form the executable's filename. + +Name: ProgramName
Type: string
+ +### Debug + +This defines [Debug specific options](#Debug) that apply to debug builds. + +Name: Debug
Type: `options.Debug` + +#### OpenInspectorOnStartup + +Setting this to `true` will open the WebInspector on startup of the application. + +Name: OpenInspectorOnStartup
Type: `bool` + +[^1]: This requires WebKit2GTK 2.36+ support and your app needs to be build with the build tag `webkit2_36` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.36 for your app. +[^2]: This requires WebKit2GTK 2.40+ support and your app needs to be build with the build tag `webkit2_40` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.40 for your app. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/project-config.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/project-config.mdx new file mode 100644 index 00000000000..8e763502bcc --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/project-config.mdx @@ -0,0 +1,92 @@ +--- +sidebar_position: 5 +--- + +# Project Config + +The project config resides in the `wails.json` file in the project directory. The structure of the config is: + +```json5 +{ + // Project config version + "version": "", + // The project name + "name": "", + // Relative path to the directory containing the compiled assets, this is normally inferred and could be left empty + "assetdir": "", + // Additional directories to trigger reloads (comma separated), this is only used for some advanced asset configurations + "reloaddirs": "", + // The directory where the build files reside. Defaults to 'build' + "build:dir": "", + // Relative path to the frontend directory. Defaults to 'frontend' + "frontend:dir": "", + // The command to install node dependencies, run in the frontend directory - often `npm install` + "frontend:install": "", + // The command to build the assets, run in the frontend directory - often `npm run build` + "frontend:build": "", + // This command has been replaced by frontend:dev:build. If frontend:dev:build is not specified will falls back to this command. \nIf this command is also not specified will falls back to frontend:build + "frontend:dev": "", + // This command is the dev equivalent of frontend:build. If not specified falls back to frontend:dev + "frontend:dev:build": "", + // This command is the dev equivalent of frontend:install. If not specified falls back to frontend:install + "frontend:dev:install": "", + // This command is run in a separate process on `wails dev`. Useful for 3rd party watchers or starting 3d party dev servers + "frontend:dev:watcher": "", + // URL to a 3rd party dev server to be used to serve assets, EG Vite. \nIf this is set to 'auto' then the devServerUrl will be inferred from the Vite output + "frontend:dev:serverUrl": "", + // Relative path to the directory that the auto-generated JS modules will be created + "wailsjsdir": "", + // The name of the binary + "outputfilename": "", + // The default time the dev server waits to reload when it detects a change in assets + "debounceMS": 100, + // Address to bind the wails dev sever to. Default: localhost:34115 + "devServer": "", + // Arguments passed to the application in shell style when in dev mode + "appargs": "", + // Defines if build hooks should be run though they are defined for an OS other than the host OS. + "runNonNativeBuildHooks": false, + "preBuildHooks": { + // The command that will be executed before a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed before a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed before every build: ${platform} is replaced with the "GOOS/GOARCH". + "*/*": "" + }, + "postBuildHooks": { + // The command that will be executed after a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed after a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed after every build: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. + "*/*": "" + }, + // Data used to populate manifests and version info. + "info": { + // The company name. Default: [The project name] + "companyName": "", + // The product name. Default: [The project name] + "productName": "", + // The version of the product. Default: '1.0.0' + "productVersion": "", + // The copyright of the product. Default: 'Copyright.........' + "copyright": "", + // A short comment of the app. Default: 'Built using Wails (https://wails.app)' + "comments": "" + }, + // 'multiple': One installer per architecture. 'single': Single universal installer for all architectures being built. Default: 'multiple' + "nsisType": "", + // Whether the app should be obfuscated. Default: false + "obfuscated": "", + // The arguments to pass to the garble command when using the obfuscated flag + "garbleargs": "" +} +``` + +This file is read by the Wails CLI when running `wails build` or `wails dev`. + +The `assetdir`, `reloaddirs`, `wailsjsdir`, `debounceMS`, `devserver` and `frontenddevserverurl` flags in `wails build/dev` will update the project config +and thus become defaults for subsequent runs. + +The JSON Schema for this file is located [here](https://wails.io/schemas/config.v2.json). diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/browser.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/browser.mdx new file mode 100644 index 00000000000..c71ec4a3a29 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# Browser + +These methods are related to the system browser. + +### BrowserOpenURL + +Opens the given URL in the system browser. + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/clipboard.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..805f68ed917 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# Clipboard + +This part of the runtime provides access to the operating system's clipboard.
The current implementation only handles text. + +### ClipboardGetText + +This method reads the currently stored text from the clipboard. + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
Returns: a string (if the clipboard is empty an empty string will be returned) or an error. + +JS: `ClipboardGetText(): Promise`
Returns: a promise with a string result (if the clipboard is empty an empty string will be returned). + +### ClipboardSetText + +This method writes a text to the clipboard. + +Go: `ClipboardSetText(ctx context.Context, text string) error`
Returns: an error if there is any. + +JS: `ClipboardSetText(text: string): Promise`
Returns: a promise with true result if the text was successfully set on the clipboard, false otherwise. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/dialog.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..9b3e6cf145c --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# Dialog + +This part of the runtime provides access to native dialogs, such as File Selectors and Message boxes. + +:::info JavaScript + +Dialog is currently unsupported in the JS runtime. + +::: + +### OpenDirectoryDialog + +Opens a dialog that prompts the user to select a directory. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Returns: Selected directory (blank if the user cancelled) or an error + +### OpenFileDialog + +Opens a dialog that prompts the user to select a file. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Returns: Selected file (blank if the user cancelled) or an error + +### OpenMultipleFilesDialog + +Opens a dialog that prompts the user to select multiple files. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +Returns: Selected files (nil if the user cancelled) or an error + +### SaveFileDialog + +Opens a dialog that prompts the user to select a filename for the purposes of saving. Can be customised using [SaveDialogOptions](#savedialogoptions). + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +Returns: The selected file (blank if the user cancelled) or an error + +### MessageDialog + +Displays a message using a message dialog. Can be customised using [MessageDialogOptions](#messagedialogoptions). + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +Returns: The text of the selected button or an error + +## Options + +### OpenDialogOptions + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| Field | Description | Win | Mac | Lin | +| -------------------------- | ---------------------------------------------- | --- | --- | --- | +| DefaultDirectory | The directory the dialog will show when opened | ✅ | ✅ | ✅ | +| DefaultFilename | The default filename | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | A list of file filters | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Show files hidden by the system | | ✅ | ✅ | +| CanCreateDirectories | Allow user to create directories | | ✅ | | +| ResolvesAliases | If true, returns the file not the alias | | ✅ | | +| TreatPackagesAsDirectories | Allow navigating into packages | | ✅ | | + +### SaveDialogOptions + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| Field | Description | Win | Mac | Lin | +| -------------------------- | ---------------------------------------------- | --- | --- | --- | +| DefaultDirectory | The directory the dialog will show when opened | ✅ | ✅ | ✅ | +| DefaultFilename | The default filename | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | A list of file filters | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Show files hidden by the system | | ✅ | ✅ | +| CanCreateDirectories | Allow user to create directories | | ✅ | | +| TreatPackagesAsDirectories | Allow navigating into packages | | ✅ | | + +### MessageDialogOptions + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| Field | Description | Win | Mac | Lin | +| ------------- | -------------------------------------------------------------------------- | -------------- | --- | --- | +| Type | The type of message dialog, eg question, info... | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| Message | The message to show the user | ✅ | ✅ | ✅ | +| Buttons | A list of button titles | | ✅ | | +| DefaultButton | The button with this text should be treated as default. Bound to `return`. | ✅[*](#windows) | ✅ | | +| CancelButton | The button with this text should be treated as cancel. Bound to `escape` | | ✅ | | + +#### Windows + +Windows has standard dialog types in which the buttons are not customisable. The value returned will be one of: "Ok", "Cancel", "Abort", "Retry", "Ignore", "Yes", "No", "Try Again" or "Continue". + +For Question dialogs, the default button is "Yes" and the cancel button is "No". This can be changed by setting the `DefaultButton` value to `"No"`. + +Example: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linux has standard dialog types in which the buttons are not customisable. The value returned will be one of: "Ok", "Cancel", "Yes", "No" + +#### Mac + +A message dialog on Mac may specify up to 4 buttons. If no `DefaultButton` or `CancelButton` is given, the first button is considered default and is bound to the `return` key. + +For the following code: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, +}) +``` + +the first button is shown as default: + +```mdx-code-block +
+ +
+
+``` + +And if we specify `DefaultButton` to be "two": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +the second button is shown as default. When `return` is pressed, the value "two" is returned. + +```mdx-code-block +
+ +
+
+``` + +If we now specify `CancelButton` to be "three": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +the button with "three" is shown at the bottom of the dialog. When `escape` is pressed, the value "three" is returned: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### DialogType + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### FileFilter + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +Windows allows you to use multiple file filters in dialog boxes. Each FileFilter will show up as a separate entry in the dialog: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linux allows you to use multiple file filters in dialog boxes. Each FileFilter will show up as a separate entry in the dialog: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +Mac dialogs only have the concept of a single set of patterns to filter files. If multiple FileFilters are provided, Wails will use all the Patterns defined. + +Example: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +This will result in the Open File dialog using `*.png,*.jpg,*.mov,*.mp4` as a filter. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/events.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/events.mdx new file mode 100644 index 00000000000..856ba6f0c26 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# Events + +The Wails runtime provides a unified events system, where events can be emitted or received by either Go or JavaScript. Optionally, data may be passed with the events. Listeners will receive the data in the local data types. + +### EventsOn + +This method sets up a listener for the given event name. When an event of type `eventName` is [emitted](#EventsEmit), the callback is triggered. Any additional data sent with the emitted event will be passed to the callback. It returns a function to cancel the listener. + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff + +This method unregisters the listener for the given event name, optionally multiple listeneres can be unregistered via `additionalEventNames`. + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce + +This method sets up a listener for the given event name, but will only trigger once. It returns a function to cancel the listener. + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple + +This method sets up a listener for the given event name, but will only trigger a maximum of `counter` times. It returns a function to cancel the listener. + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit + +This method emits the given event. Optional data may be passed with the event. This will trigger any event listeners. + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/intro.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/intro.mdx new file mode 100644 index 00000000000..aae8efccccf --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# Introduction + +The runtime is a library that provides utility methods for your application. There is both a Go and JavaScript runtime and the aim is to try and keep them at parity where possible. + +It has utility methods for: + +- [Window](window.mdx) +- [Menu](menu.mdx) +- [Dialog](dialog.mdx) +- [Events](events.mdx) +- [Browser](browser.mdx) +- [Log](log.mdx) +- [Clipboard](clipboard.mdx) + +The Go Runtime is available through importing `github.com/wailsapp/wails/v2/pkg/runtime`. All methods in this package take a context as the first parameter. This context should be obtained from the [OnStartup](../options.mdx#onstartup) or [OnDomReady](../options.mdx#ondomready) hooks. + +:::info Note + +Whilst the context will be provided to the [OnStartup](../options.mdx#onstartup) method, there's no guarantee the runtime will work in this method as the window is initialising in a different thread. If you wish to call runtime methods at startup, use [OnDomReady](../options.mdx#ondomready). + +::: + +The JavaScript library is available to the frontend via the `window.runtime` map. There is a runtime package generated when using `dev` mode that provides TypeScript declarations for the runtime. This should be located in the `wailsjs` directory in your frontend directory. + +### Hide + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +Hides the application. + +:::info Note + +On Mac, this will hide the application in the same way as the `Hide` menu item in standard Mac applications. This is different to hiding the window, but the application still being in the foreground. For Windows and Linux, this is currently the same as `WindowHide`. + +::: + +### Show + +Shows the application. + +:::info Note + +On Mac, this will bring the application back into the foreground. For Windows and Linux, this is currently the same as `WindowShow`. + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### Quit + +Quits the application. + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### Environment + +Returns details of the current environment. + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### EnvironmentInfo + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/log.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/log.mdx new file mode 100644 index 00000000000..06f0423b0a8 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# Log + +The Wails runtime provides a logging mechanism that may be called from Go or JavaScript. Like most loggers, there are a number of log levels: + +- Trace +- Debug +- Info +- Warning +- Error +- Fatal + +The logger will output any log message at the current, or higher, log level. Example: The `Debug` log level will output all messages except `Trace` messages. + +### LogPrint + +Logs the given message as a raw message. + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf + +Logs the given message as a raw message. + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace + +Logs the given message at the `Trace` log level. + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef + +Logs the given message at the `Trace` log level. + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug + +Logs the given message at the `Debug` log level. + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf + +Logs the given message at the `Debug` log level. + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo + +Logs the given message at the `Info` log level. + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof + +Logs the given message at the `Info` log level. + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning + +Logs the given message at the `Warning` log level. + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf + +Logs the given message at the `Warning` log level. + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError + +Logs the given message at the `Error` log level. + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf + +Logs the given message at the `Error` log level. + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal + +Logs the given message at the `Fatal` log level. + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf + +Logs the given message at the `Fatal` log level. + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel + +Sets the log level. In JavaScript, the number relates to the following log levels: + +| Value | Log Level | +| ----- | --------- | +| 1 | Trace | +| 2 | Debug | +| 3 | Info | +| 4 | Warning | +| 5 | Error | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## Using a Custom Logger + +A custom logger may be used by providing it using the [Logger](../options.mdx#logger) application option. The only requirement is that the logger implements the `logger.Logger` interface defined in `github.com/wailsapp/wails/v2/pkg/logger`: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/menu.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/menu.mdx new file mode 100644 index 00000000000..8a0c79e371c --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# Menu + +These methods are related to the application menu. + +:::info JavaScript + +Menu is currently unsupported in the JS runtime. + +::: + +### MenuSetApplicationMenu + +Sets the application menu to the given [menu](../menus.mdx). + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu + +Updates the application menu, picking up any changes to the menu passed to `MenuSetApplicationMenu`. + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/screen.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/screen.mdx new file mode 100644 index 00000000000..457c92ebff9 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# Screen + +These methods provide information about the currently connected screens. + +### ScreenGetAll + +Returns a list of currently connected screens. + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### Screen + +Go struct: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript interface: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx new file mode 100644 index 00000000000..15f555c5a79 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# Window + +These methods give control of the application window. + +### WindowSetTitle + +Sets the text in the window title bar. + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen + +Makes the window full screen. + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen + +Restores the previous window dimensions and position prior to full screen. + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen + +Returns true if the window is full screen. + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter + +Centers the window on the monitor the window is currently on. + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS + +Executes arbitrary JS code in the window. + +This method runs the code in the browser asynchronously and returns immediately. If the script causes any errors, they will only be available in the browser console. + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload + +Performs a "reload" (Reloads current page). + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp + +Reloads the application frontend. + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme + +Windows only. + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +Sets window theme to system default (dark/light). + +### WindowSetLightTheme + +Windows only. + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +Sets window theme to light. + +### WindowSetDarkTheme + +Windows only. + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +Sets window theme to dark. + +### WindowShow + +Shows the window, if it is currently hidden. + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowHide + +Hides the window, if it is currently visible. + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal + +Returns true if the window not minimised, maximised or fullscreen. + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize + +Sets the width and height of the window. + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowGetSize + +Gets the width and height of the window. + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize + +Sets the minimum window size. Will resize the window if the window is currently smaller than the given dimensions. + +Setting a size of `0,0` will disable this constraint. + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize + +Sets the maximum window size. Will resize the window if the window is currently larger than the given dimensions. + +Setting a size of `0,0` will disable this constraint. + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop + +Sets the window AlwaysOnTop or not on top. + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition + +Sets the window position relative to the monitor the window is currently on. + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition + +Gets the window position relative to the monitor the window is currently on. + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximise + +Maximises the window to fill the screen. + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise + +Restores the window to the dimensions and position prior to maximising. + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised + +Returns true if the window is maximised. + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise + +Toggles between Maximised and UnMaximised. + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise + +Minimises the window. + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise + +Restores the window to the dimensions and position prior to minimising. + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised + +Returns true if the window is minimised. + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour + +Sets the background colour of the window to the given RGBA colour definition. This colour will show through for all transparent pixels. + +Valid values for R, G, B and A are 0-255. + +:::info Windows + +On Windows, only alpha values of 0 or 255 are supported. Any value that is not 0 will be considered 255. + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## TypeScript Object Definitions + +### Position + +```ts +interface Position { + x: number; + y: number; +} +``` + +### Size + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/dogsapi.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..1af16f7745a --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# Dogs API + +```mdx-code-block +
+ +
+
+``` + +:::note + +This tutorial has been kindly provided by [@tatadan](https://twitter.com/tatadan) and forms part of their [Wails Examples Repository](https://github.com/tataDan/wails-v2-examples). + +::: + +In this tutorial we are going to develop an application that retrieves photos of dogs from the web and then displays them. + +### Create the project + +Let's create the application. From a terminal enter: `wails init -n dogs-api -t svelte` + +Note: We could optionally add `-ide vscode` or `-ide goland` to the end of this command if you wanted to add IDE support. + +Now let's `cd dogs-api` and start editing the project files. + +### Remove unused code + +We will start by removing some elements that we know we will not use: + +- Open `app.go` and remove the following lines: + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- Open `frontend/src/App.svelte` and delete all lines. +- Delete the `frontend/src/assets/images/logo-universal.png` file + +### Creating our application + +Now let's add our new Go code. + +Add the following struct declarations to `app.go` before the function definitions: + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +Add the following functions to `app.go` (perhaps after the existing function definitions): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +Modify the `import` section of `app.go` to look like this: + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +Add the following lines to `frontend/src/App.svelte`: + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### Testing the application + +To generate the bindings and test the application, run `wails dev`. + +### Compiling the application + +To compile the application to a single, production grade binary, run `wails build`. diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/helloworld.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/helloworld.mdx new file mode 100644 index 00000000000..ea350890049 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# Hello World + +The aim of this tutorial is to get you up and running with the most basic application using Wails. You will be able to: + +- Create a new Wails application +- Build the application +- Run the application + +:::note + +This tutorial uses Windows as the target platform. Output will vary slightly depending on your operating system. + +::: + +## Create a new Wails application + +To create a new Wails application using the default vanilla JS template, you need to run the following command: + +```bash +wails init -n helloworld +``` + +You should see something similar to the following: + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +This will create a new directory called `helloworld` in the current directory. In this directory, you will find a number of files: + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## Build the application + +To build the application, change to the new `helloworld` project directory and run the following command: + +```bash +wails build +``` + +You should see something like the following: + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +This has compiled the application and saved it in the `build/bin` directory. + +## Run the application + +If we view the `build/bin` directory in Windows Explorer, we should see our project binary: + +```mdx-code-block +
+ +
+
+``` + +We can run it by simply double-clicking the `helloworld.exe` file. + +On Mac, Wails generates a `helloworld.app` file which can be run by double-clicking it. + +On Linux, you can run the application using `./helloworld` from the `build/bin` directory. + +You should see the application working as expected: + +```mdx-code-block +
+ +
+
+``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.3.1.json b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.3.1.json new file mode 100644 index 00000000000..e413aa9df6f --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.3.1.json @@ -0,0 +1,38 @@ +{ + "version.label": { + "message": "v2.3.1", + "description": "The label for version v2.3.1" + }, + "sidebar.docs.category.Getting Started": { + "message": "Hướng Dẫn Nhanh", + "description": "The label for category Getting Started in sidebar docs" + }, + "sidebar.docs.category.Reference": { + "message": "Tham Khảo", + "description": "The label for category Reference in sidebar docs" + }, + "sidebar.docs.category.Runtime": { + "message": "Bộ Công Cụ Runtime", + "description": "The label for category Runtime in sidebar docs" + }, + "sidebar.docs.category.Community": { + "message": "Cộng Đồng", + "description": "The label for category Community in sidebar docs" + }, + "sidebar.docs.category.Showcase": { + "message": "Trưng Bày", + "description": "The label for category Showcase in sidebar docs" + }, + "sidebar.docs.category.Guides": { + "message": "Định Hướng", + "description": "The label for category Guides in sidebar docs" + }, + "sidebar.docs.category.Tutorials": { + "message": "Thực Hành", + "description": "The label for category Tutorials in sidebar docs" + }, + "sidebar.docs.link.Contributing": { + "message": "Đóng Góp", + "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.4.0.json b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.4.0.json new file mode 100644 index 00000000000..ff5f2e9c8e8 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.4.0.json @@ -0,0 +1,38 @@ +{ + "version.label": { + "message": "v2.4.0", + "description": "The label for version v2.4.0" + }, + "sidebar.docs.category.Getting Started": { + "message": "Hướng Dẫn Nhanh", + "description": "The label for category Getting Started in sidebar docs" + }, + "sidebar.docs.category.Reference": { + "message": "Tham Khảo", + "description": "The label for category Reference in sidebar docs" + }, + "sidebar.docs.category.Runtime": { + "message": "Bộ Công Cụ Runtime", + "description": "The label for category Runtime in sidebar docs" + }, + "sidebar.docs.category.Community": { + "message": "Cộng Đồng", + "description": "The label for category Community in sidebar docs" + }, + "sidebar.docs.category.Showcase": { + "message": "Trưng Bày", + "description": "The label for category Showcase in sidebar docs" + }, + "sidebar.docs.category.Guides": { + "message": "Định Hướng", + "description": "The label for category Guides in sidebar docs" + }, + "sidebar.docs.category.Tutorials": { + "message": "Thực Hành", + "description": "The label for category Tutorials in sidebar docs" + }, + "sidebar.docs.link.Contributing": { + "message": "Đóng Góp", + "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.5.0.json b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.5.0.json new file mode 100644 index 00000000000..af3a23601d9 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.5.0.json @@ -0,0 +1,38 @@ +{ + "version.label": { + "message": "v2.5.0", + "description": "The label for version v2.5.0" + }, + "sidebar.docs.category.Getting Started": { + "message": "Hướng Dẫn Nhanh", + "description": "The label for category Getting Started in sidebar docs" + }, + "sidebar.docs.category.Reference": { + "message": "Tham Khảo", + "description": "The label for category Reference in sidebar docs" + }, + "sidebar.docs.category.Runtime": { + "message": "Bộ Công Cụ Runtime", + "description": "The label for category Runtime in sidebar docs" + }, + "sidebar.docs.category.Community": { + "message": "Cộng Đồng", + "description": "The label for category Community in sidebar docs" + }, + "sidebar.docs.category.Showcase": { + "message": "Trưng Bày", + "description": "The label for category Showcase in sidebar docs" + }, + "sidebar.docs.category.Guides": { + "message": "Định Hướng", + "description": "The label for category Guides in sidebar docs" + }, + "sidebar.docs.category.Tutorials": { + "message": "Thực Hành", + "description": "The label for category Tutorials in sidebar docs" + }, + "sidebar.docs.link.Contributing": { + "message": "Đóng Góp", + "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.6.0.json b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.6.0.json new file mode 100644 index 00000000000..2d5505b0d36 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/version-v2.6.0.json @@ -0,0 +1,38 @@ +{ + "version.label": { + "message": "v2.6.0", + "description": "The label for version v2.6.0" + }, + "sidebar.docs.category.Getting Started": { + "message": "Hướng Dẫn Nhanh", + "description": "The label for category Getting Started in sidebar docs" + }, + "sidebar.docs.category.Reference": { + "message": "Tham Khảo", + "description": "The label for category Reference in sidebar docs" + }, + "sidebar.docs.category.Runtime": { + "message": "Bộ Công Cụ Runtime", + "description": "The label for category Runtime in sidebar docs" + }, + "sidebar.docs.category.Community": { + "message": "Cộng Đồng", + "description": "The label for category Community in sidebar docs" + }, + "sidebar.docs.category.Showcase": { + "message": "Trưng Bày", + "description": "The label for category Showcase in sidebar docs" + }, + "sidebar.docs.category.Guides": { + "message": "Định Hướng", + "description": "The label for category Guides in sidebar docs" + }, + "sidebar.docs.category.Tutorials": { + "message": "Thực Hành", + "description": "The label for category Tutorials in sidebar docs" + }, + "sidebar.docs.link.Contributing": { + "message": "Đóng Góp", + "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" + } +} diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx new file mode 100644 index 00000000000..4a332a43083 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx @@ -0,0 +1,751 @@ +# Nhật Ký Phiên Bản + +Tất cả những thay đổi đáng chú ý sẽ được thêm vào tài liệu này. + +Định dạng văn bản dựa trên [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) và dự án này tuân thủ [Semantic Versioning](https://semver.org/spec/v2.0.0.html). + +- `Added` cho các tính năng mới nhất. +- `Changed` cho các chỉnh sửa của các tính năng sẵn có. +- `Deprecated` cho các tính năng mà sắp bị gỡ bỏ ở các phiên bản tiếp theo. +- `Removed` các tính năng đã được gỡ bỏ. +- `Fixed` cho các lỗi hoặc bug đã được sửa. +- `Security` cho các trường hợp vá lỗ hổng bảo mật. + +## [Unreleased] + +### Added + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### Changed + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### Fixed + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + +## v2.6.0 - 2023-09-06 + +### Breaking Changes + +- AssetServer RequestURI and URL are now RFC and Go Docs compliant for server requests. This means Scheme, Host and Fragments are not provided anymore. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2722) + +### Fixed + +- Avoid app crashing when the Linux GTK key is empty. Fixed by @aminya in [PR](https://github.com/wailsapp/wails/pull/2672) +- Fix issue where app would exit before main() on linux if $DISPLAY env var was not set. Fixed by @phildrip in [PR](https://github.com/wailsapp/wails/pull/2841) +- Fixed a race condition when positioning the window on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2850) +- Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) +- Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) +- Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) +- Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) +- Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) + +### Added + +- Added correct NodeJS and Docker package names for DNF package manager of Fedora 38. Added by @aranggitoar in [PR](https://github.com/wailsapp/wails/pull/2790) +- Added `-devtools` production build flag. Added by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2725) +- Added `EnableDefaultContextMenu` option to allow enabling the browser's default context-menu in production . Added by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2733) +- Added smart functionality for the default context-menu in production with CSS styles to control it. Added by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2748) +- Added custom error formatting to allow passing structured errors back to the frontend. +- Added sveltekit.mdx guide. Added by @figuerom16 in [PR](https://github.com/wailsapp/wails/pull/2771) +- Added ProgramName option to [linux.Options](/docs/reference/options#linux). Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2817) +- Added new community template wails-sveltekit-ts. Added by [@haukened](https://github.com/haukened) in [PR](https://github.com/wailsapp/wails/pull/2851) +- Added support for retrieving the logical and physical screen size in the screen api. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2856) +- Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) +- Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) +- Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) + +### Changed + +- Now uses new `go-webview2` module. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2687). +- Changed styling of `doctor` command. Changed by @MarvinJWendt in [PR](https://github.com/wailsapp/wails/pull/2660) +- Enable HiDPI option by default in windows nsis installer. Changed by @5aaee9 in [PR](https://github.com/wailsapp/wails/pull/2694) +- Now debug builds include the un-minified version of the runtime JS with source maps . Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2745) + +## v2.5.1 - 2023-05-16 + +### Breaking Changes + +- The Go WebView2Loader allowed env variable and registry overrides to change the behaviour of WebView2. This is not possible when using the native WebView2Loader with Wails and should not be possible according to [PR](https://github.com/wailsapp/wails/pull/1771). Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2668) +- `wails dev` now uses the custom schemes `wails://` on macOS and Linux for all Vite versions. This also fixes missing reloads after multiple fast savings on Linux and Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2664) + +### Fixed + +- Fixed segfaults during fast reloads of requests on macOS. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2664) +- Fixed devserver on Linux for older WebKit2GTK versions < 2.36. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2664) +- Fixed devserver on Windows that might have triggered the WebView2 into a hang. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2664) + +## v2.5.0 - 2023-05-13 + +### Breaking Changes + +- `wails dev` now uses the custom schemes `wails://` on macOS and Linux if Vite >= `v3.0.0` is used. This makes the dev application consistent in behaviour with the final production application and fixes some long-standing inconsistencies. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2610) + +### Added + +- Added Nodejs version in `wails doctor`. Added by @misitebao in [PR](https://github.com/wailsapp/wails/pull/2546) +- Added support for WebKit2GTK 2.40+ on Linux. This brings additional features for the [AssetServer](/docs/reference/options#assetserver), like support for HTTP Request Bodies. The app must be compiled with the Go build tag `webkit2_40` to activate support for this features. This also bumps the minimum requirement of WebKit2GTK to 2.40 for your app. Added by @stffabi in this [PR](https://github.com/wailsapp/wails/pull/2592) +- macOS: Added Window menu role with well known shortcuts "Minimize, Full-Screen and Zoom". Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2586) +- macOS: Added "Hide, Hide Others, Show All“ to appmenu. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2586) +- Windows: Added flag to disable the WebView2 `RendererCodeIntegrity` checks, please see the comment on the flag for more information. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2627) +- Windows: Added handling of WebView2 process crashes, for unrecoverable errors an error message is shown that the app needs to be restarted. If an error occurs that shows a chromium error page, make sure the Window and the WebView2 is visible. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2627) + +### Changed + +- [v3] Typescript model generation using `StructDef`s from new AST-based parser. Added by @ATenderholt in [PR1](https://github.com/wailsapp/wails/pull/2428/files) and [PR2](https://github.com/wailsapp/wails/pull/2485). + +### Fixed + +- Fixed console printing in `wails generate template`. Fixed by @misitebao in [PR](https://github.com/wailsapp/wails/pull/2483) +- Fixed unicode encoding of strings for multi-rune characters. Fixed by @joshbuddy in [PR](https://github.com/wailsapp/wails/pull/2509) +- Fixed `-skipbindings` flag in `wails dev`. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2584) +- Fixed `runtime.MenuUpdateApplicationMenu` on macOS. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2588) +- Fixed add package name for `libwebkit`/`pkg-config` and use shell.RunCommandWithENV instead of shell.RunCommand in `zypper.go`. Fixed by @wgjtyu in [PR](https://github.com/wailsapp/wails/pull/2593) +- Make sure to start the CommonFileDialogs on Windows on the Main-Thread. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2606) +- Fixed generated typescript bindings. Fixed by @dominictwlee in [PR](https://github.com/wailsapp/wails/pull/2552) +- Fixed `StartHidden` option not working on Linux. Fixed by @LGiki in [PR](https://github.com/wailsapp/wails/pull/2624) + +## v2.4.1 - 2023-03-20 + +### Changed + +- Support single clicks on items with `--wails-draggable: drag` again on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2482) + +### Fixed + +- Fixed panic when using `wails dev` and the AssetServer tried to log to the logger. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2481) +- Fixed compatibility with WebView2 Runtime > `110.0.1587.69` which showed a `connection refused` html page before doing a reload of the frontend. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2496) + +## v2.4.0 - 2023-03-08 + +### Added + +- Added Webview GPU acceleration options for [Windows](/docs/reference/options#webviewgpuisdisabled) and [Linux](/docs/reference/options#webviewgpupolicy). Added by @Lyimmi in [PR](https://github.com/wailsapp/wails/pull/2266) +- Added `EnableFraudulentWebsiteDetection` option to opt-in to scan services for fraudulent content, such as malware or phishing attempts. Older releases had the scan services per default activated. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2269) +- Allow an [AssetServer Middleware](/docs/reference/options#middleware) to specify the `Content-Type` of a file served by the [Assets](/docs/reference/options#assets-1) `fs.FS`. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2286) +- The AssetServer now detects more mimetypes by extension, e.g. `.mjs`. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2391) + +### Changed + +- Improved fullscreen mode for frameless window on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2279), [PR](https://github.com/wailsapp/wails/pull/2288) and [PR](https://github.com/wailsapp/wails/pull/2299) +- On Windows unmaximising a window has no effect anymore when the window is in fullscreen mode, this makes it consistent with e.g. macOS. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2279) +- Frameless resize now sets the cursor on documentElement, otherwise resizing cursor won't be shown outside of the body rectangle. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2289) +- Improved the `--wails-draggable` experience to be more reactive. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2302) +- NSIS template now installs the shortcuts for all users and not only for the current user. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2373) + +### Fixed + +- Fixed failing build hooks when `build/bin` was missing. Fixed by @Lyimmi in [PR](https://github.com/wailsapp/wails/pull/2273) +- Fixed fullscreen mode for frameless window on Windows to fully cover the taskbar when changing into fullscreen from maximised state. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2279) +- Fixed set window background colour on Windows when setting the colour via runtime. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2279) +- Fixed the showing of a white border around a fullscreen window when `DisableWindowIcon` is active on Windows. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2299) +- Fixed the sometimes lagging drag experience with `--wails-draggable` on Windows. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2302) +- Fixed applying the default arch to platform flag in wails cli. If only a `GOOS` has been supplied as platform flag e.g. `wails build --platform windows` the current architecture wasn't applied and the build failed. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2309) +- Fixed a segfault on opening the inspector on older macOS versions. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2397) +- Fixed the macos single architecture builds not respecting an output file name specified with the '-o' flag. Fixed by @gwynforthewyn in [PR](https://github.com/wailsapp/wails/pull/2358) +- Fixed `undo`/`redo` on macOS. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2430) +- Fixed `Events*` runtime functions in JavaScript not returning the function to cancel the watcher. Fixed by @zllovesuki in [PR](https://github.com/wailsapp/wails/pull/2434) +- Fixed AppOptions merging defaults when a custom logger is used. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2452) +- Fixed race condition and missing unlock of mutex in events handling. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2453) + +## v2.3.0 - 2022-12-29 + +### Added + +- Added `OpenInspectorOnStartup` to debug options to allow opening the WebInspector during startup of the application in debug mode. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2080) +- On macOS `wails doctor` now also shows the version of Xcode installed. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2089) +- The [AssetServer](/docs/reference/options#assetserver) now supports handling range-requests if the [Assets](/docs/reference/options/#assets-1) `fs.FS` provides an `io.ReadSeeker`. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2091) +- Add new property for the `wails.json` config file - `bindings`. More information on the new property can be found in the updated [schema](/schemas/config.v2.json). Properties `prefix` and `suffix` allow you to control the generated TypeScript entity name in the `model.ts` file. Added by @OlegGulevskyy in [PR](https://github.com/wailsapp/wails/pull/2101) +- The `WindowSetAlwaysOnTop` method is now exposed in the JS runtime. Fixed by @gotid in [PR](https://github.com/wailsapp/wails/pull/2128) +- The [AssetServer](/docs/reference/options#assetserver) now supports serving the index.html file when requesting a directory. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2110) +- Added support for WebKit2GTK 2.36+ on Linux. This brings additional features for the [AssetServer](/docs/reference/options#assetserver), like support for HTTP methods and Headers. The app must be compiled with the Go build tag `webkit2_36` to activate support for this features. This also bumps the minimum requirement of WebKit2GTK to 2.36 for your app. Fixed by @stffabi in this [PR](https://github.com/wailsapp/wails/pull/2151) +- Added support for file input selection on macOS. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2209) +- Added support Request/Response streaming of the AssetServer on macOS. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2219) +- Added request path checks for the AssetServer. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2235) +- Added new App Store guide. Added by @achhabra2 in [PR](https://github.com/wailsapp/wails/pull/2142) +- Added extra libwebkit check for dnf package manager (required for latest Fedora). Fixed by @NullCode1337 in this [PR](https://github.com/wailsapp/wails/pull/2218) +- Added new `-nomodsync` flag to disable the automatic syncing of the `go.mod` file. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2215) +- Added support for adding prefix and postfix text to generated Typescript classes. Added by @OlegGulevskyy in [PR](https://github.com/wailsapp/wails/pull/2101) +- Added reference to NSIS as optional dependency. Added by @acheong08 in [PR](https://github.com/wailsapp/wails/pull/2070) +- Added Korean translation for the website. Added by @cybertramp in [PR](https://github.com/wailsapp/wails/pull/2093) + +### Fixed + +- The `noreload` flag in wails dev wasn't applied. Fixed by @stffabi in this [PR](https://github.com/wailsapp/wails/pull/2081) +- `build/bin` folder was duplicating itself on each reload in `wails dev` mode. Fixed by @OlegGulevskyy in this [PR](https://github.com/wailsapp/wails/pull/2103) +- Prevent a thin white line at the bottom of a frameless window on Windows. Fixed by @stffabi in this [PR](https://github.com/wailsapp/wails/pull/2111) +- Better signal handling for Linux. Fixed by @leaanthony in this [PR](https://github.com/wailsapp/wails/pull/2152) +- Running the docs locally was broken due to mandatory crowdin token requirement. Fixed by @OlegGulevskyy in this [PR](https://github.com/wailsapp/wails/pull/2231) +- Generated typescript for Go functions that returned a single error was incorrect. Fixed by @ATenderholt in this [PR](https://github.com/wailsapp/wails/pull/2247) +- The right mouse event was not being raised correctly for Linux. Fixed by @leaanthony in this [PR](https://github.com/wailsapp/wails/pull/2190) +- Remove extra spaces in Wails version when querying `go.mod`. Fixed by @matholt in this [PR](https://github.com/wailsapp/wails/pull/2197) +- Fix go tests failure on Mac. Fixed by @avengerweb in this [PR](https://github.com/wailsapp/wails/pull/2187) +- Fix go test errors. Fixed by @misitebao & @avengerweb in [these](https://github.com/wailsapp/wails/pull/2169) [PRs](https://github.com/wailsapp/wails/pull/2187) +- Added `WindowSetAlwaysOnTop` to the JS runtime. Fixed by @gotid in [PR](https://github.com/wailsapp/wails/pull/2128) +- Fixed obfuscated flag typo in docs. Fixed by @ckilb in [PR](https://github.com/wailsapp/wails/pull/2102) +- Fixed React Hash Router link in docs. Fixed by @marvinhosea in [PR](https://github.com/wailsapp/wails/pull/2050) + +### Changed + +- Improve error message if no `index.html` could be found in the assets and validate assetserver options. Changed by @stffabi in this [PR](https://github.com/wailsapp/wails/pull/2110) +- Promote the Go WebView2Loader from experimental to stable. This means now per default all Wails build use the new loader introduced with `v2.2.0`. The old loader remains usable with build tag `native_webview2loader` for the next few releases. Changed by @stffabi in this [PR](https://github.com/wailsapp/wails/pull/2199) +- Refactored CLI. Changed by @leaanthony in this [PR](https://github.com/wailsapp/wails/pull/2123) +- Remove unreachable code. Changed by @tmclane in this [PR](https://github.com/wailsapp/wails/pull/2182) + +## v2.2.0 - 2022-11-09 + +### Added + +- Wails now uses a purpose built, native Go implementation of Microsoft's webview2loader dll. This means there is no need to embed the `Webview2Loader.dll` file in your binaries, meaning filesizes will be ~130k smaller! Thanks to @stffabi for this [awesome contribution](https://github.com/wailsapp/wails/pull/1974)! +- This release provides much more control over custom asset handling via the new [AssetServer](https://wails.io/docs/reference/options#assetserver) options. This allows you to provide your own custom asset handler and hook into the request chain through middleware. Thanks to @stffabi for this [awesome contribution](https://github.com/wailsapp/wails/pull/2016) and @mholt for the [initial idea](https://github.com/wailsapp/wails/issues/2007) and extensive testing. +- It is now possible to customise the layout of your Wails projects using 2 new options in `wails.json`: + - `build:dir` can be used to specify where the build files reside + - `frontend:dir` can be used to specify where the frontend files reside + - If `go.mod` is not found in the same directory as `wails.json`, the cli will look up the parent directories to find it. Fixed by @leaanthony in this [PR](https://github.com/wailsapp/wails/pull/2009) +- Colour output in the CLI can now be turned off by using the `--nocolor` flag. This is useful for CI/CD pipelines. Thanks to @scottopell for the [PR](https://github.com/wailsapp/wails/pull/1947) +- A JSON schema definition for the `wails.json` file has been added. IDEs should now provide code complete when editing `wails.json`. Thanks to @binyamin for the [PR](https://github.com/wailsapp/wails/pull/1864) +- The `EventsOn*` methods now return a function that can be called to deregister the listener. Thanks to @joshbuddy for the [PR](https://github.com/wailsapp/wails/pull/1969) + +### Fixed + +- Webview2 on Windows returns a potential whitespace when defining the style like this style="--wails-draggable: drag". Fixed by @stffabi in https://github.com/wailsapp/wails/pull/1989 +- Bound structs that had `struct{}` field types would cause the TypeScript generation to fail. Thanks to @ParkourLiu for the [PR](https://github.com/wailsapp/wails/pull/1999) +- When maximising a frameless window on Windows with multiple monitors, the window could sometimes become blank. Thanks to @stffabi for the [fix](https://github.com/wailsapp/wails/pull/2043) + +### Changed + +- The troubleshooting guide was updated to provide guidance when [Vite fails to start](https://wails.io/docs/guides/troubleshooting#cannot-start-service-host-version-xxx-does-not-match-binary-version-xxx). Thanks to @willdot for the [PR](https://github.com/wailsapp/wails/pull/2000). +- English, Chinese and Japanese documentation updates. Thanks to @misitebao. + +### Deprecated + +- The [AssetsHandler](https://wails.io/docs/reference/options#assetshandler) option has been deprecated. Please use the [AssetServer](https://wails.io/docs/reference/options#assetserver) option instead. + +### New Contributors + +- @willdot made their first contribution in https://github.com/wailsapp/wails/pull/2000 +- @ParkourLiu made their first contribution in https://github.com/wailsapp/wails/pull/1999 +- @binyamin made their first contribution in https://github.com/wailsapp/wails/pull/1864 +- @joshbuddy made their first contribution in https://github.com/wailsapp/wails/pull/1969 +- @sgosiaco made their first contribution in https://github.com/wailsapp/wails/pull/2062 + +## v2.1.0 - 2022-10-18 + +### Removed + +- The `RGBA` option in `options.App` has now been removed. Use `BackgroundColour` instead. + +### Added + +- [Support for defaulting to No button in Windows dialogs](https://wails.io/docs/reference/runtime/dialog/#windows) - @leaanthony in https://github.com/wailsapp/wails/pull/1875 +- Add missing resize for frameless window on Linux - @Lyimmi in https://github.com/wailsapp/wails/pull/1918 +- [Add window transparency for linux](https://wails.io/docs/reference/options#windowistranslucent-2) - @Lyimmi in https://github.com/wailsapp/wails/pull/1926 +- [Add WindowExecJS method](https://wails.io/docs/reference/runtime/window#windowexecjs) - @leaanthony in https://github.com/wailsapp/wails/pull/1927 +- [Add support for `Info.dev.plist` on macOS](https://wails.io/docs//reference/cli#dev) - @leaanthony in https://github.com/wailsapp/wails/pull/1960 +- [Add ZoomFactor get/set and add the respective windows only options](https://wails.io/docs/reference/options#zoomfactor) - @pierrejoye in https://github.com/wailsapp/wails/pull/1463 + +### Fixed + +- Embed directories auto-created if they don't exist - @leaanthony in https://github.com/wailsapp/wails/pull/1983 +- Quote command arguments if they have a space - @leaanthony in https://github.com/wailsapp/wails/pull/1892 +- Fixed Linux frameless window drag - @Lyimmi in https://github.com/wailsapp/wails/pull/1916 +- Fix gtk_window_begin_resize_drag's mouse button - @Lyimmi in https://github.com/wailsapp/wails/pull/1920 +- Fix binding generation special cases - @JulioDRF in https://github.com/wailsapp/wails/pull/1902 +- Remove the `.git` directory in the template - @misitebao in https://github.com/wailsapp/wails/pull/1929 +- Fix wails dev - @JulioDRF in https://github.com/wailsapp/wails/pull/1931 +- Fix for considering new `go` files in dev filesystem watcher - @scottopell in https://github.com/wailsapp/wails/pull/1946 +- Prevent type parsing to interfere with TypeScript package name - @ValentinTrinque in https://github.com/wailsapp/wails/pull/1942 +- [dev] Do not try to infer assetdir from fs.FS when a frontend dev server is in use - @stffabi in https://github.com/wailsapp/wails/pull/1972 +- Fix init command not listed in wails help message - @lyon-lee-dev in https://github.com/wailsapp/wails/pull/1976 + +### Changed + +- Add PR checks - @leaanthony in https://github.com/wailsapp/wails/pull/1879 +- Auto label project cards - @leaanthony in https://github.com/wailsapp/wails/pull/1881 +- Add issue translator - @leaanthony in https://github.com/wailsapp/wails/pull/1891 +- Update label names in the issue template - @misitebao in https://github.com/wailsapp/wails/pull/1893 +- obfuscated instead of obfuscate in the docs - @arifali123 in https://github.com/wailsapp/wails/pull/1895 +- [assetHandler] Remove redundant log prefix - @stffabi in https://github.com/wailsapp/wails/pull/1896 +- [dev] Do not generate bindings in the dev app itself - @stffabi in https://github.com/wailsapp/wails/pull/1899 +- Update Chinese translation - @almas1992 in https://github.com/wailsapp/wails/pull/1894 +- Refactor app - @leaanthony in https://github.com/wailsapp/wails/pull/1909 +- Update documents - @misitebao in https://github.com/wailsapp/wails/pull/1907 https://github.com/wailsapp/wails/pull/1936 +- Adding Tutorial link - @raguay in https://github.com/wailsapp/wails/pull/1903 +- Add react-ts-vite-tailwind template - @hotafrika in https://github.com/wailsapp/wails/pull/1930 +- Update README.zh-Hans.md - @o8x in https://github.com/wailsapp/wails/pull/1949 +- Add Elm Tailwind CSS community template - @rnice01 in https://github.com/wailsapp/wails/pull/1939 +- Chore/generate sponsors - @leaanthony in https://github.com/wailsapp/wails/pull/1965 +- Use swc + pnpm for website - @leaanthony in https://github.com/wailsapp/wails/pull/1966 +- Sort structs in models.ts - @leaanthony in https://github.com/wailsapp/wails/pull/1961 +- Update Sponsor Image - @github-actions in https://github.com/wailsapp/wails/pull/1973 +- docs: sync documents - @misitebao in https://github.com/wailsapp/wails/pull/1968 +- Update events.mdx - @cuigege in https://github.com/wailsapp/wails/pull/1979 + +### New Contributors + +- @arifali123 made their first contribution in https://github.com/wailsapp/wails/pull/1895 +- @almas1992 made their first contribution in https://github.com/wailsapp/wails/pull/1894 +- @JulioDRF made their first contribution in https://github.com/wailsapp/wails/pull/1902 +- @hotafrika made their first contribution in https://github.com/wailsapp/wails/pull/1930 +- @scottopell made their first contribution in https://github.com/wailsapp/wails/pull/1946 +- @o8x made their first contribution in https://github.com/wailsapp/wails/pull/1949 +- @rnice01 made their first contribution in https://github.com/wailsapp/wails/pull/1939 +- @cuigege made their first contribution in https://github.com/wailsapp/wails/pull/1979 +- @lyon-lee-dev made their first contribution in https://github.com/wailsapp/wails/pull/1976 + +## v2.0.0 - 2022-09-22 + +### Fixed + +- Fix buildtags parsing if only one tag is specified by @stffabi in https://github.com/wailsapp/wails/pull/1858 +- Use embed all to include all files in templates by @stffabi in https://github.com/wailsapp/wails/pull/1862 + +### Changed + +- Bump minimum required Go version to 1.18 by @stffabi in https://github.com/wailsapp/wails/pull/1854 +- Add check for minimum required Go version by @stffabi in https://github.com/wailsapp/wails/pull/1853 +- chore: update README and workflows by @misitebao in https://github.com/wailsapp/wails/pull/1848 +- Update introduction.mdx by @ParvinEyvazov in https://github.com/wailsapp/wails/pull/1863 +- Releasetest/release workflow by @leaanthony in https://github.com/wailsapp/wails/pull/1869 +- Optimize documentation website by @misitebao in https://github.com/wailsapp/wails/pull/1849 + +### New Contributors + +- @ParvinEyvazov made their first contribution in https://github.com/wailsapp/wails/pull/1863 + +## v2.0.0-rc.1 - 2022-09-13 + +### Deprecated + +- The `-noGen` flag for the `wails dev` command has been replaced with `-skipbindings`. This is to align with the `wails build` command. + +### Added + +- Add garble support by @leaanthony in https://github.com/wailsapp/wails/pull/1793 +- Make draggable CSS property customisable by @leaanthony in https://github.com/wailsapp/wails/pull/1828 +- Add Some WindowState by @zandercodes in https://github.com/wailsapp/wails/pull/1349 +- Make EventsOff capable of removing multiple listeners by @Lyimmi in https://github.com/wailsapp/wails/pull/1822 + +### Fixed + +- Use `Promise` when Go routine does not output by @SheetJSDev in https://github.com/wailsapp/wails/pull/1821 +- preact-ts template build fix by @Debdut in https://github.com/wailsapp/wails/pull/1781 +- fix frontend/tsconfig.js by @Lyimmi in https://github.com/wailsapp/wails/pull/1795 +- fix: fix bugs in website by @misitebao in https://github.com/wailsapp/wails/pull/1810 +- Fix vue-ts template by @leaanthony in https://github.com/wailsapp/wails/pull/1813 +- Remove duplicate defs in win32/window.go by @AlbinoDrought in https://github.com/wailsapp/wails/pull/1832 + +### Changed + +- Upgrade React to use Vite v3 by @leaanthony in https://github.com/wailsapp/wails/pull/1744 +- Upgrade Lit to use Vite v3 by @leaanthony in https://github.com/wailsapp/wails/pull/1745 +- Support vite3 for Vue by @leaanthony in https://github.com/wailsapp/wails/pull/1746 +- Preact templates for vite 3 by @leaanthony in https://github.com/wailsapp/wails/pull/1770 +- Prevent env variables and registry overrides from changing behaviour by @stffabi in https://github.com/wailsapp/wails/pull/1771 +- Use go implementation to retrieve the version of a fixed runtime by @stffabi in https://github.com/wailsapp/wails/pull/1790 +- Change contribution guide type from "doc" to "page" by @misitebao in https://github.com/wailsapp/wails/pull/1777 +- feat(website): repair document content by @misitebao in https://github.com/wailsapp/wails/pull/1775 +- chore: sort out files by @misitebao in https://github.com/wailsapp/wails/pull/1776 +- Add Korean to doc by @jaesung9507 in https://github.com/wailsapp/wails/pull/1774 +- Add README.ja.md by @shinshin86 in https://github.com/wailsapp/wails/pull/1783 +- Reorganized contribution guidelines page by @misitebao in https://github.com/wailsapp/wails/pull/1784 +- fix(website): fix link by @misitebao in https://github.com/wailsapp/wails/pull/1785 +- Update templates.mdx by @KiddoV in https://github.com/wailsapp/wails/pull/1799 +- Better watcher by @leaanthony in https://github.com/wailsapp/wails/pull/1827 +- Only set GDK_BACKEND to "x11" if GDK_BACKEND is unset and XDG_SESSION_TYPE is not "wayland" by @prurigro in https://github.com/wailsapp/wails/pull/1811 +- Optimize images by @imgbot in https://github.com/wailsapp/wails/pull/1812 + +### New Contributors + +- @shinshin86 made their first contribution in https://github.com/wailsapp/wails/pull/1783 +- @Debdut made their first contribution in https://github.com/wailsapp/wails/pull/1781 +- @KiddoV made their first contribution in https://github.com/wailsapp/wails/pull/1799 +- @zandercodes made their first contribution in https://github.com/wailsapp/wails/pull/1349 +- @prurigro made their first contribution in https://github.com/wailsapp/wails/pull/1811 +- @SheetJSDev made their first contribution in https://github.com/wailsapp/wails/pull/1821 +- @AlbinoDrought made their first contribution in https://github.com/wailsapp/wails/pull/1832 +- @imgbot made their first contribution in https://github.com/wailsapp/wails/pull/1812 + +## v2.0.0-beta.44 - 2022-08-20 + +### Deprecated + +The `data-wails-drag` attribute is being deprecated in favour of the following CSS style: `style="--wails-draggable:drag"`. You can use `style="--wails-draggable:no-drag"` to disable the drag behaviour. For this release only, you can test this by setting the following application option: + +```go +Experimental: &options.Experimental{ + UseCSSDrag: true, +}, +``` + +### Added + +- Set file permissions for generated files by @leaanthony in https://github.com/wailsapp/wails/pull/1763 +- Experimental support for CSS Drag detection by @leaanthony in https://github.com/wailsapp/wails/pull/1750 + +### Fixed + +- Bug fix collecting of output binaries for platforms by @stffabi in https://github.com/wailsapp/wails/pull/1715 +- Fix registration of exposed fields by @ValentinTrinque in https://github.com/wailsapp/wails/pull/1727 +- Fix column widths for doctor command's dependencies table by @ianmjones in https://github.com/wailsapp/wails/pull/1717 +- Do not generate bindings for `OnBeforeClose` method + +### Changed + +- Misc code refactors and removal by @leaanthony in https://github.com/wailsapp/wails/pull/1713 +- Add react-router to routing.mdx by @Maicarons2022 in https://github.com/wailsapp/wails/pull/1755 +- Add Japanese to doc by @RyoTagami in https://github.com/wailsapp/wails/pull/1716 +- Added EmailIt and Modal File Manager by @raguay in https://github.com/wailsapp/wails/pull/1728 +- Adding my ScriptBar program by @raguay in https://github.com/wailsapp/wails/pull/1761 +- Link to general webview2 runtime download page and not to a specific language by @stffabi in https://github.com/wailsapp/wails/pull/1764 +- Updated translations in https://github.com/wailsapp/wails/pull/1719 and https://github.com/wailsapp/wails/pull/1720 +- Remove text outside style's brackets by @DragoSpiro98 in https://github.com/wailsapp/wails/pull/1765 + +### New Contributors + +- @RyoTagami made their first contribution in https://github.com/wailsapp/wails/pull/1716 +- @raguay made their first contribution in https://github.com/wailsapp/wails/pull/1728 +- @Maicarons2022 made their first contribution in https://github.com/wailsapp/wails/pull/1755 +- @DragoSpiro98 made their first contribution in https://github.com/wailsapp/wails/pull/1765 + +## v2.0.0-beta.43 - 2022-08-08 + +### Added + +- Add support for retrieving the release notes from Github by @leaanthony in https://github.com/wailsapp/wails/pull/1679 +- Add `frontend:dev:install` configuration by @LGiki in https://github.com/wailsapp/wails/pull/1666 + +### Fixed + +- Fix formatting of some error messages by @stffabi in https://github.com/wailsapp/wails/pull/1665 +- Windows dialogs now work when window is not visible yet by @leaanthony in https://github.com/wailsapp/wails/pull/1662 +- Multiple fixes for MacOS asset requests by @stffabi in https://github.com/wailsapp/wails/pull/1668 and https://github.com/wailsapp/wails/pull/1681 +- Fix for Go 1.19 by @stffabi in https://github.com/wailsapp/wails/pull/1689 +- Removed Linux warnings by @leaanthony in https://github.com/wailsapp/wails/pull/1656 +- Better support for doubleclick events in drag regions by @leaanthony in https://github.com/wailsapp/wails/pull/1704 +- Allow MacOS frameless window to be miniturisable by @leaanthony in https://github.com/wailsapp/wails/pull/1705 + +### Changed + +- add wails-sveltekit-template by @h8gi in https://github.com/wailsapp/wails/pull/1671 +- wails doctor now reports correct MacOS os id by @stffabi in https://github.com/wailsapp/wails/pull/1673 +- Update application-development.mdx by @SamHennessy in https://github.com/wailsapp/wails/pull/1682 +- Move SetMin/Max calls to main thread by @leaanthony in https://github.com/wailsapp/wails/pull/1684 +- Change `frontend:dev` to `frontend:dev:build` by @LGiki in https://github.com/wailsapp/wails/pull/1691 +- Build frontend only before starting the dev watcher command by @stffabi in https://github.com/wailsapp/wails/pull/1694 +- Improve error message for auto dev server discovery without a dev watcher by @stffabi in https://github.com/wailsapp/wails/pull/1711 + +### New Contributors + +- @h8gi made their first contribution in https://github.com/wailsapp/wails/pull/1671 +- @SamHennessy made their first contribution in https://github.com/wailsapp/wails/pull/1682 + +## v2.0.0-beta.42 - 2022-07-25 + +### Added + +- Added `options.NewRGBA` and `options.NewRGB` functions to create `*options.RGBA` by @leaanthony + +### Fixed + +- Fixed initial build of frontend when using `wails dev` on new projects by @leaanthony in https://github.com/wailsapp/wails/pull/1650 +- Ignore empty install command when running `wails dev` by @stffabi in https://github.com/wailsapp/wails/pull/1651 +- Fixed error reporting in templates +- BackgroundColour documentation fix +- Generalize manual compile steps [Documentation] by @acheong08 in https://github.com/wailsapp/wails/pull/1644 + +## v2.0.0-beta.40 - 2022-07-24 + +### Added + +- Add Show() and Hide() to runtime to show/hide application by @leaanthony in https://github.com/wailsapp/wails/pull/1599 +- Override target platform/arch using GOOS and GOARCH environment variables by @leaanthony in https://github.com/wailsapp/wails/pull/1618 +- Add option to skip frontend rebuild in dev mode by @leaanthony in https://github.com/wailsapp/wails/pull/1632 + +### Fixed + +- Update svelte templates to use vite 3 by @leaanthony in https://github.com/wailsapp/wails/pull/1643 +- Fix plain template by @stffabi in https://github.com/wailsapp/wails/pull/1609 +- Fix Website layout by @leaanthony in https://github.com/wailsapp/wails/pull/1616 +- Fixed typo in documentation page docs/howdoesitwork by @MyNameIsAres in https://github.com/wailsapp/wails/pull/1636 +- Use scaling when setting min/max window by @leaanthony in https://github.com/wailsapp/wails/pull/1557 + +### Changed + +- Install dev dependencies before starting dev mode by @leaanthony in https://github.com/wailsapp/wails/pull/1615 +- Translate and fix website text by @misitebao in https://github.com/wailsapp/wails/pull/1525 +- docs: add MyNameIsAres as a contributor for doc by @allcontributors in https://github.com/wailsapp/wails/pull/1638 +- Deprecate Fullscreen appoption by @acheong08 in https://github.com/wailsapp/wails/pull/1640 + +### Deprecated + +- The `Fullscreen` application option is deprecated. Please use [`WindowStartState`](https://wails.io/docs/reference/options#windowstartstate) instead. + +### New Contributors + +- @MyNameIsAres made their first contribution in https://github.com/wailsapp/wails/pull/1636 + +## v2.0.0-beta.39.2 - 2022-07-20 + +### Added + +- Update example for macOS Menu by @acheong08 in https://github.com/wailsapp/wails/pull/1600 + +### Fixed + +- Reinstate Go 1.17 compatibility by @leaanthony in https://github.com/wailsapp/wails/pull/1605 + +## v2.0.0-beta.39 - 2022-07-19 + +### Added + +- New screen dimensions runtime API by @skamensky in https://github.com/wailsapp/wails/pull/1519 +- Auto discover vite devserver port by @leaanthony in https://github.com/wailsapp/wails/pull/1547 +- Add nixpkgs support to doctor command. by @ianmjones in https://github.com/wailsapp/wails/pull/1551 +- New pre-build hooks feature by @leaanthony in https://github.com/wailsapp/wails/pull/1578 +- New production log level option by @leaanthony in https://github.com/wailsapp/wails/pull/1555 + +### Fixed + +- Fix stack corruption in Windows when using ICoreWebView2HttpHeadersCollectionIterator by @stffabi in https://github.com/wailsapp/wails/pull/1589 +- Move WindowGet\* to main thread by @leaanthony in https://github.com/wailsapp/wails/pull/1464 +- Allow -appargs flag to pass flags to binary. by @ianmjones in https://github.com/wailsapp/wails/pull/1534 +- Fix checking for installed apt package in none English session. by @ianmjones in https://github.com/wailsapp/wails/pull/1548 +- Fix OnBeforeClose code for Mac by @leaanthony in https://github.com/wailsapp/wails/pull/1558 +- Support Maps in TS conversion by @leaanthony in https://github.com/wailsapp/wails/pull/1435 +- Check for line length when scanning for local devserver url by @leaanthony in https://github.com/wailsapp/wails/pull/1566 +- Remove usage of unsafe.Pointer in winc by @stffabi and @leaanthony in https://github.com/wailsapp/wails/pull/1556 + +### Changed + +- Rename WindowSetRGBA -> WindowSetBackgroundColour by @leaanthony in https://github.com/wailsapp/wails/pull/1506 +- Improvements to the dev command by @stffabi in https://github.com/wailsapp/wails/pull/1510 +- Update vscode template by @leaanthony in https://github.com/wailsapp/wails/pull/1398 +- Bump svelte from 3.42.2 to 3.49.0 in /v2/internal/frontend/runtime/dev by @dependabot in https://github.com/wailsapp/wails/pull/1572 +- Bump svelte from 3.42.5 to 3.49.0 in /v2/internal/frontend/runtime by @dependabot in https://github.com/wailsapp/wails/pull/1573 +- Add troubleshooting for `Not Found` error by @acheong08 in https://github.com/wailsapp/wails/pull/1586 +- Docs/better homepage by @leaanthony in https://github.com/wailsapp/wails/pull/1591 + +### New Contributors + +- @skamensky made their first contribution in https://github.com/wailsapp/wails/pull/1519 +- @acheong08 made their first contribution in https://github.com/wailsapp/wails/pull/1586 + +**Full Changelog**: https://github.com/wailsapp/wails/compare/v2.0.0-beta.38...v2.0.0-beta.39 + +## v2.0.0-beta.38 - 2022-06-27 + +### Added + +- Add race detector to build & dev by @Lyimmi in https://github.com/wailsapp/wails/pull/1426 +- [linux] Support `linux/arm` architecture by @Lyimmi in https://github.com/wailsapp/wails/pull/1427 +- Create gitignore when using `-g` option by @jaesung9507 in https://github.com/wailsapp/wails/pull/1430 +- [windows] Add Suspend/Resume callback support by @leaanthony in https://github.com/wailsapp/wails/pull/1474 +- Add runtime function `WindowSetAlwaysOnTop` by @chenxiao1990 in https://github.com/wailsapp/wails/pull/1442 +- [windows] Allow setting browser path by @NanoNik in https://github.com/wailsapp/wails/pull/1448 + +### Fixed + +- [linux] Improve switching to main thread for callbacks by @stffabi in https://github.com/wailsapp/wails/pull/1392 +- [windows] Fix WebView2 minimum runtime version check by @stffabi in https://github.com/wailsapp/wails/pull/1456 +- [linux] Fix apt command syntax (#1458) by @abtin in https://github.com/wailsapp/wails/pull/1461 +- [windows] Set Window Background colour if provided + debounce redraw option by @leaanthony in https://github.com/wailsapp/wails/pull/1466 +- Fix small typo in docs by @LukenSkyne in https://github.com/wailsapp/wails/pull/1449 +- Fix the url to surge by @andywenk in https://github.com/wailsapp/wails/pull/1460 +- Fixed theme change at runtime by @leaanthony in https://github.com/wailsapp/wails/pull/1473 +- Fix: Don't stop if unable to remove temporary bindings build by @leaanthony in https://github.com/wailsapp/wails/pull/1465 +- [windows] Pass the correct installationStatus to the webview installation strategy by @stffabi in https://github.com/wailsapp/wails/pull/1483 +- [windows] Make `SetBackgroundColour` compatible for `windows/386` by @stffabi in https://github.com/wailsapp/wails/pull/1493 +- Fix lit-ts template by @Orijhins in https://github.com/wailsapp/wails/pull/1494 + +### Changed + +- [windows] Load WebView2 loader from embedded only by @stffabi in https://github.com/wailsapp/wails/pull/1432 +- Add showcase entry for October + update homepage carousel entry for October by @marcus-crane in https://github.com/wailsapp/wails/pull/1436 +- Always use return in wrapped method by @leaanthony in https://github.com/wailsapp/wails/pull/1410 +- [windows] Unlock OSThread after native calls have been finished by @stffabi in https://github.com/wailsapp/wails/pull/1441 +- Add `BackgroundColour` and deprecate `RGBA` by @leaanthony in https://github.com/wailsapp/wails/pull/1475 +- AssetsHandler remove retry logic in dev mode by @stffabi in https://github.com/wailsapp/wails/pull/1479 +- Add Solid JS template to docs by @sidwebworks in https://github.com/wailsapp/wails/pull/1492 +- Better signal handling by @leaanthony in https://github.com/wailsapp/wails/pull/1488 +- Chore/react 18 create root by @tomanagle in https://github.com/wailsapp/wails/pull/1489 + +### New Contributors + +- @jaesung9507 made their first contribution in https://github.com/wailsapp/wails/pull/1430 +- @LukenSkyne made their first contribution in https://github.com/wailsapp/wails/pull/1449 +- @andywenk made their first contribution in https://github.com/wailsapp/wails/pull/1460 +- @abtin made their first contribution in https://github.com/wailsapp/wails/pull/1461 +- @chenxiao1990 made their first contribution in https://github.com/wailsapp/wails/pull/1442 +- @NanoNik made their first contribution in https://github.com/wailsapp/wails/pull/1448 +- @sidwebworks made their first contribution in https://github.com/wailsapp/wails/pull/1492 +- @tomanagle made their first contribution in https://github.com/wailsapp/wails/pull/1489 + +## v2.0.0-beta.37 - 2022-05-26 + +### Added + +- Add `nogen` flag in wails dev command by @mondy in https://github.com/wailsapp/wails/pull/1413 +- Initial support for new native translucency in Windows Preview by @leaanthony in https://github.com/wailsapp/wails/pull/1400 + +### Fixed + +- Bugfix/incorrect bindings by @leaanthony in https://github.com/wailsapp/wails/pull/1383 +- Fix runtime.js events by @polikow in https://github.com/wailsapp/wails/pull/1369 +- Fix docs formatting by @antimatter96 in https://github.com/wailsapp/wails/pull/1372 +- Events | fixes #1388 by @lambdajack in https://github.com/wailsapp/wails/pull/1390 +- bugfix: correct typo by @tmclane in https://github.com/wailsapp/wails/pull/1391 +- Fix typo in docs by @LGiki in https://github.com/wailsapp/wails/pull/1393 +- Fix typo bindings.js to ipc.js by @rayshoo in https://github.com/wailsapp/wails/pull/1406 +- Make sure to execute the menu callbacks on a new goroutine by @stffabi in https://github.com/wailsapp/wails/pull/1403 +- Update runtime.d.ts & templates by @Yz4230 in https://github.com/wailsapp/wails/pull/1421 +- Add missing className to input in React and Preact templates by @edwardbrowncross in https://github.com/wailsapp/wails/pull/1419 + +### Changed + +- Improve multi-platform builds by @stffabi in https://github.com/wailsapp/wails/pull/1373 +- During wails dev only use reload logic if no AssetsHandler are in use by @stffabi in https://github.com/wailsapp/wails/pull/1385 +- Update events.mdx by @Junkher in https://github.com/wailsapp/wails/pull/1387 +- Add Next.js template by @LGiki in https://github.com/wailsapp/wails/pull/1394 +- Add docs on wails generate module by @TechplexEngineer in https://github.com/wailsapp/wails/pull/1414 +- Add macos custom menu EditMenu tips by @daodao97 in https://github.com/wailsapp/wails/pull/1423 + +### New Contributors + +- @polikow made their first contribution in https://github.com/wailsapp/wails/pull/1369 +- @antimatter96 made their first contribution in https://github.com/wailsapp/wails/pull/1372 +- @Junkher made their first contribution in https://github.com/wailsapp/wails/pull/1387 +- @lambdajack made their first contribution in https://github.com/wailsapp/wails/pull/1390 +- @LGiki made their first contribution in https://github.com/wailsapp/wails/pull/1393 +- @rayshoo made their first contribution in https://github.com/wailsapp/wails/pull/1406 +- @TechplexEngineer made their first contribution in https://github.com/wailsapp/wails/pull/1414 +- @mondy made their first contribution in https://github.com/wailsapp/wails/pull/1413 +- @Yz4230 made their first contribution in https://github.com/wailsapp/wails/pull/1421 +- @daodao97 made their first contribution in https://github.com/wailsapp/wails/pull/1423 +- @edwardbrowncross made their first contribution in https://github.com/wailsapp/wails/pull/1419 + +## v2.0.0-beta.36 - 2022-04-27 + +### Fixed + +- [v2] Validate devServer property to be of the correct form by [@stffabi](https://github.com/stffabi) in https://github.com/wailsapp/wails/pull/1359 +- [v2, darwin] Initialize native variables on stack to prevent segfault by [@stffabi](https://github.com/stffabi) in https://github.com/wailsapp/wails/pull/1362 +- Vue-TS template fix + +### Changed + +- Added `OnStartup` method back to default templates + +## v2.0.0-beta.35 - 2022-04-27 + +### Breaking Changes + +- When data was sent to the `EventsOn` callback, it was being sent as a slice of values, instead of optional parameters to the method. `EventsOn` now works as expected, but you will need to update your code if you currently use this. [More information](https://github.com/wailsapp/wails/issues/1324) +- The broken `bindings.js` and `bindings.d.ts` files have been replaced by a new JS/TS code generation system. More details [here](https://wails.io/docs/howdoesitwork#calling-bound-go-methods) + +### Added + +- **New Templates**: Svelte, React, Vue, Preact, Lit and Vanilla templates, both JS and TS versions. `wails init -l` for more info. +- Default templates now powered by [Vite](https://vitejs.dev). This enables lightning fast reloads when you use `wails dev`! +- Add support for external frontend development servers. See `frontend:dev:serverUrl` in the [project config](https://wails.io/docs/reference/project-config) - [@stffabi](https://github.com/stffabi) +- [Fully configurable dark mode](https://wails.io/docs/reference/options#theme) for Windows. +- Hugely improved [WailsJS generation](https://wails.io/docs/howdoesitwork#calling-bound-go-methods) (both JavaScript and TypeScript) +- Wails doctor now reports information about the wails installation - [@stffabi](https://github.com/stffabi) +- Added docs for [code-signing](https://wails.io/docs/guides/signing) and [NSIS installer](https://wails.io/docs/guides/windows-installer) - [@gardc](https://github.com/gardc) +- Add support for `-trimpath` [build flag](https://wails.io/docs/reference/cli#build) +- Add support for a default AssetsHandler - [@stffabi](https://github.com/stffabi) + +### Fixed + +- Improved mimetype detection for BOM marker and comments - [@napalu](https://github.com/napalu) +- Remove duplicate mimetype entries - [@napalu](https://github.com/napalu) +- Remove duplicate TypeScript imports in generated definition files - [@adalessa](https://github.com/adalessa) +- Add missing method declaration - [@adalessa](https://github.com/adalessa) +- Fix Linux sigabrt on start - [@napalu](https://github.com/napalu) +- Double Click event now works on elements with `data-wails-drag` attribute - [@jicg](https://github.com/jicg) +- Suppress resizing during minimize of a frameless window - [@stffabi](https://github.com/stffabi) +- Fixed TS/JS generation for Go methods with no returns +- Fixed WailsJS being generated in project directory + +### Changed + +- Website docs are now versioned +- Improved `runtime.Environment` call +- Improve the close action for Mac +- A bunch of dependabot security updates +- Improved website content - [@misitebao](https://github.com/misitebao) +- Upgrade issue template - [@misitebao](https://github.com/misitebao) +- Convert documents that don't require version management to individual pages + - [@misitebao](https://github.com/misitebao) +- Website now using Algolia search + +## v2.0.0-beta.34 - 2022-03-26 + +### Added + +- Add support for 'DomReady' callback on linux by [@napalu](https://github.com/napalu) in #1249 +- MacOS - Show extension by default by [@leaanthony](https://github.com/leaanthony) in #1228 + +### Fixed + +- [v2, nsis] Seems like / as path separator works only for some directives in a cross platform way by [@stffabi](https://github.com/stffabi) in #1227 +- import models on binding definition by [@adalessa](https://github.com/adalessa) in #123 +- Use local search on website by [@leaanthony](https://github.com/leaanthony) in #1234 +- Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 +- Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 +- [v2, windows] Fix maximised start state by [@stffabi](https://github.com/stffabi) in #1243 +- Ensure Linux IsFullScreen uses GDK_WINDOW_STATE_FULLSCREEN bitmask appropriately. by [@ianmjones](https://github.com/ianmjones) in #1245 +- Fix memory leak in ExecJS for Mac by [@leaanthony](https://github.com/leaanthony) in #1230 +- Fix, or at least a workaround, for (#1232) by [@BillBuilt](https://github.com/BillBuilt) in #1247 +- [v2] Use os.Args[0] for self starting wails by [@stffabi](https://github.com/stffabi) in #1258 +- [v2, windows] Windows switch scheme: https -> http by @stefpap in #1255 +- Ensure Focus is regained by Webview2 when tabbing by [@leaanthony](https://github.com/leaanthony) in #1257 +- Try to focus window when Show() is called. by [@leaanthony](https://github.com/leaanthony) in #1212 +- Check system for user installed Linux dependencies by [@leaanthony](https://github.com/leaanthony) in #1180 + +### Changed + +- feat(website): sync documents and add content by [@misitebao](https://github.com/misitebao) in #1215 +- refactor(cli): optimize default templates by [@misitebao](https://github.com/misitebao) in #1214 +- Run watcher after initial build by [@leaanthony](https://github.com/leaanthony) in #1216 +- Feature/docs update by [@leaanthony](https://github.com/leaanthony) in #1218 +- feat(website): optimize website and sync documents by [@misitebao](https://github.com/misitebao) in #1219 +- docs: sync documents by [@misitebao](https://github.com/misitebao) in #1224 +- Default index page by [@leaanthony](https://github.com/leaanthony) in #1229 +- Build added win32 compatibility by [@fengweiqiang](https://github.com/fengweiqiang) in #1238 +- docs: sync documents by [@misitebao](https://github.com/misitebao) in #1260 + +## v2.0.0-beta.33 - 2022-03-05 + +### Added + +- NSIS Installer support for creating installers for Windows applications - Thanks [@stffabi](https://github.com/stffabi) 🎉 +- New frontend:dev:watcher command to spin out 3rd party watchers when using wails dev - Thanks [@stffabi](https://github.com/stffabi)🎉 +- Remote templates now support version tags - Thanks [@misitebao](https://github.com/misitebao) 🎉 + +### Fixed + +- A number of fixes for ARM Linux providing a huge improvement - Thanks [@ianmjones](https://github.com/ianmjones) 🎉 +- Fixed potential Nil reference when discovering the path to `index.html` +- Fixed crash when using `runtime.Log` methods in a production build +- Improvements to internal file handling meaning webworkers will now work on Windows - Thanks [@stffabi](https://github.com/stffabi)🎉 + +### Changed + +- The Webview2 bootstrapper is now run as a normal user and doesn't require admin rights +- The docs have been improved and updated +- Added troubleshooting guide diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/coc.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/coc.mdx new file mode 100644 index 00000000000..a027c6273df --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-pages/coc.mdx @@ -0,0 +1,83 @@ +# Contributor Covenant Code of Conduct + +## Our Pledge + +We as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, caste, color, religion, or sexual identity and orientation. + +We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community. + +## Our Standards + +Examples of behavior that contributes to a positive environment for our community include: + +- Demonstrating empathy and kindness toward other people +- Being respectful of differing opinions, viewpoints, and experiences +- Giving and gracefully accepting constructive feedback +- Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience +- Focusing on what is best not just for us as individuals, but for the overall community + +Examples of unacceptable behavior include: + +- The use of sexualized language or imagery, and sexual attention or advances of any kind +- Trolling, insulting or derogatory comments, and personal or political attacks +- Public or private harassment +- Publishing others' private information, such as a physical or email address, without their explicit permission +- Other conduct which could reasonably be considered inappropriate in a professional setting + +## Enforcement Responsibilities + +Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful. + +Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate. + +## Scope + +This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to the community leaders responsible for enforcement at coc@wails.io. All complaints will be reviewed and investigated promptly and fairly. + +All community leaders are obligated to respect the privacy and security of the reporter of any incident. + +## Enforcement Guidelines + +Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct: + +### 1. Correction + +**Community Impact**: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community. + +**Consequence**: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested. + +### 2. Warning + +**Community Impact**: A violation through a single incident or series of actions. + +**Consequence**: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban. + +### 3. Temporary Ban + +**Community Impact**: A serious violation of community standards, including sustained inappropriate behavior. + +**Consequence**: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban. + +### 4. Permanent Ban + +**Community Impact**: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals. + +**Consequence**: A permanent ban from any sort of public interaction within the community. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.1, available at [https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1]. + +Community Impact Guidelines were inspired by [Mozilla's code of conduct enforcement ladder][mozilla coc]. + +For answers to common questions about this code of conduct, see the FAQ at [https://www.contributor-covenant.org/faq][faq]. Translations are available at [https://www.contributor-covenant.org/translations][translations]. + +[homepage]: https://www.contributor-covenant.org +[v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html +[mozilla coc]: https://github.com/mozilla/diversity +[faq]: https://www.contributor-covenant.org/faq +[translations]: https://www.contributor-covenant.org/translations diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/community-guide.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/community-guide.mdx new file mode 100644 index 00000000000..8bc5a3e193a --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-pages/community-guide.mdx @@ -0,0 +1,142 @@ +# Community Guide + +The number of Wails users is growing at an incredible rate, and if you're reading this, chances are you're ready to join. So... welcome! + +## Resources + +### Code of Conduct + +This [Code of Conduct](./coc) is an easy guide to develop the technical communities in which we participate. + +### Stay in the Know + +- Follow our [official Twitter account](https://twitter.com/wailsapp). + +### Get Support + +- [GitHub](https://github.com/wailsapp/wails) - If you have a bug to report or feature to request, that's what the GitHub issues are for. Please respect the rules specified in each repository's issue template. +- [Discord](https://discord.gg/JDdSxwjhGf) - A place for Wails devs to meet and chat in real time. +- [QQ Group(中文)](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - A Wails group for Chinese developers to communicate, where you can get help from other developers. + +### Explore the Ecosystem + +- [The Awesome Wails Page](https://github.com/wailsapp/awesome-wails) - See what other awesome resources have been published by other awesome people. + +## Ways of contributing + +Wails is an open source, community driven project. We welcome anyone to join us in contributing to the project. This documentation is aimed at anyone wishing to get familiar with the project and the development processes. + +There are many ways to contribute to the project: + +- Developing new features +- Fixing bugs +- Testing +- Documenting features +- Writing tutorials / guides +- Helping others on the issues + discussions boards + +Guides for these have been created in their own sections. Before getting started, please introduce yourself in the [Contributing to Wails](https://github.com/wailsapp/wails/discussions/1520) discussion. + +### Developing New Features + +We are always keen to add features to Wails and expand on what the project can do. The process for adding new features are as follows: + +- Pick an enhancement ticket with the "TODO" label. It's preferable to select one from the current [Backlog](https://github.com/orgs/wailsapp/projects/1/views/1) but the choice is yours. +- Before developing, check that the ticket includes the following information: + - The purpose of the enhancement + - What is out of scope for the enhancement + - What platforms the enhancement targets (most features should be cross-platform unless there's a very specific reason) +- If the ticket does not include this information, feel free to request the information from the person who opened the ticket. Sometimes placeholder tickets are created and require more details +- Comment on the ticket stating if you wish to develop the feature +- Clone the repository and create a branch with the format `feature/_` +- New features often require documentation so please ensure you have also added or updated the documentation as part of the changes +- Once the feature is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. +- Once all the testing is completed, please update the status of the PR from draft and leave a message. + +:::note +There is nothing stopping you from opening a ticket and working on it yourself, but please be aware that all enhancement requests are reviewed for good fit. Not all ideas will be selected, so it's best to have discussion about the enhancement first. +::: + +:::warning +Any PRs opened without a corresponding ticket may be rejected. +::: + +### Fixing Bugs + +The process for fixing bugs are as follows: + +- Check the current [Backlog](https://github.com/orgs/wailsapp/projects/1/views/1) and select a bug to fix +- Before developing, check that the ticket includes the following information: +- The scope of the issue including platforms affected +- The steps to reproduce. Sometimes bugs are opened that are not Wails issues and the onus is on the reporter to prove that it is a Wails issue with a [minimal reproducible example](https://stackoverflow.com/help/minimal-reproducible-example) +- The output of `wails doctor` +- A test that can reproduce the bug +- If the ticket does not include this information, feel free to request the information from the person who opened the ticket. +- Comment on the ticket stating you wish to develop a fix +- Clone the repository and create a branch with the format `bugfix/_` +- Once the fix is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. +- Once all the testing is completed, please update the status of the PR from draft and leave a message. + +:::note +There is nothing stopping you from opening a ticket and working on it yourself, but please be aware that all bugfixes should be discussed as the approach may have unintended side effects. +::: + +:::warning +Any PRs opened without a corresponding ticket may be rejected. +::: + +### Testing + +Testing is vitally important to ensure quality in the project. There are a couple of scenarios where testing can really help the project: + +- Testing if a bug is reproducible on your local system +- Testing PRs to ensure that they work correctly + +If you chose to test if someone's bug report is reproducible on your local system, then feel free to add a comment on the ticket confirming this with the output of `wails doctor`. + +To test PRs, choose a PR to test and check if the PR description has the testing scenarios listed. If not, please ask the person who opened the PR to provide that list. Once you have determined a valid test scenario, please report your findings on the PR. + +If you ever need more clarity or help on testing, please ask a question in the [Contributing to Wails](https://github.com/wailsapp/wails/discussions/1520) discussion or on slack. + +### Documenting + +This website is also the main documentation site for the project. Sometimes this gets out of date and needs some slight adjustments. Some of the documentation isn't written to the best standards either. Developing documentation is hard and so any contribution to this is greatly appreciated. Features without documentation are unfinished so to the project, it's _as important_ as the code. + +We generally do not create tickets for updating documentation so if there is text you think should be updated or rephrased then feel free to submit a PR for that. This site is in the main repository under the `website` directory. We use [Docusaurus](https://docusaurus.io/) to create the site so there is plenty of existing documentation and tutorials around to get started. + +To set up a local documentation development environment, do the following: + +- [Install npm](https://docs.npmjs.com/cli/v8/configuring-npm/install) +- `cd website` +- `npm install` +- `npm run start` + +After it has all installed and is running, you should see the site at [`http://localhost:3000`](http://localhost:3000). Any changes made to the site text will be immediately reflected in the browser. + +#### Versioning + +We employ a versioning system where we have the "latest" documentation AKA "Next Version" which has all the changes that have occurred since the last release. We also keep the last release documentation as well as the version before that. + +There isn't usually a reason to update released documentation so we don't generally update the documents in the `versioned_docs` or `versioned_sidebars` directories. + +The "next version" docs are mainly in `website/docs` with some "version independent" documents in `src/pages`. Any updates should be made in the `website/docs` directory. + +#### Languages + +The default documents of the Wails project are English documents. We use the "crowdin" tool to translate documents in other languages and synchronize them to the website. You can [join our project](https://crowdin.com/project/wails) and submit your translations to make contributions. + +##### Add new language + +If you want to add a new language to the documentation, please follow the prompts to [fill in and submit an Issue](https://github.com/wailsapp/wails/issues/new?assignees=&labels=documentation&template=documentation.yml). After being confirmed by the maintainer, we will add the language to the "crowdin" and you will then be able to submit your translation. + +### Helping Others + +A great way to contribute to the project is to help others who are experiencing difficulty. This is normally reported as a ticket or a message on the Wails discord server. Even just clarifying the issue can really help out. Sometimes, when an issue is discussed and gets resolved, we create a guide out of it to help others who face the same issues. + +To join the Wails discord server, click [here](https://discord.gg/JDdSxwjhGf). + +:::note + +Work In Progress + +::: diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx new file mode 100644 index 00000000000..af5cd28c530 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx @@ -0,0 +1,268 @@ +# Credits + +- [Lea Anthony](https://github.com/leaanthony) - Project owner, lead developer +- [Stffabi](https://github.com/stffabi) - Technical lead, developer and maintainer +- [Misite Bao](https://github.com/misitebao) - Documentation wizard, Chinese translation, Windows testing, Bug finder general +- [Travis McLane](https://github.com/tmclane) - Cross-compilation work, MacOS testing +- [Lyimmi](https://github.com/Lyimmi) - All things Linux + +## Sponsors + + + +## Contributors + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Lea Anthony
Lea Anthony

💻 🤔 🎨 🖋 💡 🧑‍🏫 📆 🔧 🐛 📝 🚧 📦 👀 💬 🔬 ⚠️ 📢 👀 📖
stffabi
stffabi

💻 🤔 🎨 🐛 🚧 📦 👀 💬 🔬 👀 📖 ⚠️
Travis McLane
Travis McLane

💻 🔬 📦 🤔 🐛 👀 ⚠️ 💬 📖
Misite Bao
Misite Bao

📖 🌍 🔬 🚧
Byron Chris
Byron Chris

💻 🔬 🚧 🐛 👀 ⚠️ 💬 🤔 🎨 📦 🚇
konez2k
konez2k

💻 📦 🤔
Dario Emerson
Dario Emerson

💻 🐛 🤔 ⚠️
Ian M. Jones
Ian M. Jones

💻 🐛 🤔 ⚠️ 👀 📦
marktohark
marktohark

💻
Ryan H
Ryan H

💻
Cody Bentley
Cody Bentley

💻 📦 🤔 💵
Florent
Florent

💻 🐛
Alexander Hudek
Alexander Hudek

💻 💵
Tim Kipp
Tim Kipp

💻
Altynbek Kaliakbarov
Altynbek Kaliakbarov

💻
Nikolai Zimmermann
Nikolai Zimmermann

💻
k-muchmore
k-muchmore

💻
Snider
Snider

💻 🤔 📖 💵
Albert Sun
Albert Sun

💻 ⚠️
Ariel
Ariel

💻 🐛
Ilgıt Yıldırım
Ilgıt Yıldırım

💻 🐛 💵
Toyam Cox
Toyam Cox

💻 📦 🐛
hi019
hi019

💻 🐛
Arthur Wiebe
Arthur Wiebe

💻 🐛
Balakrishna Prasad Ganne
Balakrishna Prasad Ganne

💻
BillBuilt
BillBuilt

💻 📦 🤔 💬 💵
Eng Zer Jun
Eng Zer Jun

🚧 💻
LGiki
LGiki

📖
Lontten
Lontten

📖
Lukas Crepaz
Lukas Crepaz

💻 🐛
Marcus Crane
Marcus Crane

🐛 📖 💵
Qais Patankar
Qais Patankar

📖
Wakeful-Cloud
Wakeful-Cloud

💻 🐛
Zámbó, Levente
Zámbó, Levente

💻 📦 🐛 ⚠️
Ironpark
Ironpark

💻 🤔
mondy
mondy

💻 📖
Benjamin Ryan
Benjamin Ryan

🐛
fallendusk
fallendusk

📦 💻
Mat Ryer
Mat Ryer

💻 🤔 🐛
Abtin
Abtin

💻 🐛
Adrian Lanzafame
Adrian Lanzafame

📦 💻
Aleksey Polyakov
Aleksey Polyakov

🐛 💻
Alexander Matviychuk
Alexander Matviychuk

💻 📦
AlienRecall
AlienRecall

💻 📦
Aman
Aman

📖
Amaury Tobias Quiroz
Amaury Tobias Quiroz

💻 🐛
Andreas Wenk
Andreas Wenk

📖
Antonio Stanković
Antonio Stanković

💻 📦
Arpit Jain
Arpit Jain

📖
Austin Schey
Austin Schey

💻 🐛
Benjamin Thomas
Benjamin Thomas

💻 📦 🤔
Bertram Truong
Bertram Truong

💻 🐛
Blake Bourque
Blake Bourque

📖
Denis
Denis

📖
diogox
diogox

💻 📦
Dmitry Gomzyakov
Dmitry Gomzyakov

💻 📦
Edward Browncross
Edward Browncross

💻
Elie Grenon
Elie Grenon

💻
Florian Didron
Florian Didron

💻 🐛 🤔 ⚠️ 👀 📦
GargantuaX
GargantuaX

💵
Igor Minin
Igor Minin

💻 🐛
Jae-Sung Lee
Jae-Sung Lee

💻 🤔
Jarek
Jarek

💻 📦
Junker
Junker

📖
Kris Raney
Kris Raney

💻 🐛
Luken
Luken

📖
Mark Stenglein
Mark Stenglein

💻 🐛
buddyabaddon
buddyabaddon

💻
MikeSchaap
MikeSchaap

💻 🐛
NYSSEN Michaël
NYSSEN Michaël

💻 🐛
Nan0
Nan0

💻 🤔 ⚠️ 👀
oskar
oskar

📖
Pierre Joye
Pierre Joye

💻 🐛 🤔 ⚠️
Reuben Thomas-Davis
Reuben Thomas-Davis

💻 🐛
Robin
Robin

💻 🐛
Sebastian Bauer
Sebastian Bauer

💻 🤔 ⚠️ 👀 💬
Sidharth Rathi
Sidharth Rathi

📖 🐛
Sithembiso Khumalo
Sithembiso Khumalo

💻 🐛
Soheib El-Harrache
Soheib El-Harrache

💻 🐛 💵
Sophie Au
Sophie Au

💻 🐛
Stefanos Papadakis
Stefanos Papadakis

💻 🐛
Steve Chung
Steve Chung

💻 🐛
Timm Ortloff
Timm Ortloff

📖
Tom
Tom

💻
Valentin Trinqué
Valentin Trinqué

💻 🐛
mattn
mattn

💻 🐛
bearsh
bearsh

💻 🤔 📖
chenxiao
chenxiao

💻 🤔 📖
fengweiqiang
fengweiqiang

💻 📦
flin7
flin7

📖
fred21O4
fred21O4

📖
gardc
gardc

📖
rayshoo
rayshoo

📖
Ishiyama Yuzuki
Ishiyama Yuzuki

💻 🐛
佰阅
佰阅

💻
刀刀
刀刀

📖 🐛
归位
归位

💻 🐛
skamensky
skamensky

💻 🤔 📖
dependabot[bot]
dependabot[bot]

💻 🚧
Damian Sieradzki
Damian Sieradzki

💵
John Dorman
John Dorman

💵
Ian Sinnott
Ian Sinnott

💵
Arden Shackelford
Arden Shackelford

💵
Bironou
Bironou

💵
CharlieGo_
CharlieGo_

💵
overnet
overnet

💵
jugglingjsons
jugglingjsons

💵
Selvin Ortiz
Selvin Ortiz

💵
ZanderCodes
ZanderCodes

💵
Michael Voronov
Michael Voronov

💵
letheanVPN
letheanVPN

💵
Tai Groot
Tai Groot

💵
easy-web-it
easy-web-it

💵
Michael Olson
Michael Olson

💵
EdenNetwork Italia
EdenNetwork Italia

💵
ondoki
ondoki

💵
QuEST Rail LLC
QuEST Rail LLC

💵
Gilgameš
Gilgameš

💵
Bernt-Johan Bergshaven
Bernt-Johan Bergshaven

💵
Liam Bigelow
Liam Bigelow

💵
Nick Arellano
Nick Arellano

💵
Frank Chiarulli Jr.
Frank Chiarulli Jr.

💵
Tyler
Tyler

💵
Trea Hauet
Trea Hauet

💵
Kent 'picat' Gruber
Kent 'picat' Gruber

💵
tc-hib
tc-hib

💵
Antonio
Antonio

📖
MyNameIsAres
MyNameIsAres

📖
Maicarons J
Maicarons J

📖
kiddov
kiddov

📖 💵 ⚠️ 🤔
Nicolas Coutin
Nicolas Coutin

💵
Parvin Eyvazov
Parvin Eyvazov

📖
github-actions[bot]
github-actions[bot]

💻
Oleg Gulevskyy
Oleg Gulevskyy

💻 📖 🚧 📦
Richard Guay
Richard Guay

📖
Adam Tenderholt
Adam Tenderholt

💻
JulioDRF
JulioDRF

💻
Scott Opell
Scott Opell

💻
Vadim Shchepotev
Vadim Shchepotev

💻
Will Andrews
Will Andrews

💻
Gwyn
Gwyn

💻 👀 💬 🔬
希嘉嘉
希嘉嘉

💻
ALMAS
ALMAS

💻
Alex
Alex

💻
Arif Ali
Arif Ali

💻
Artur Siarohau
Artur Siarohau

💻
Binyamin Aron Green
Binyamin Aron Green

💻
Brian Dwyer
Brian Dwyer

💻
Christian Kilb
Christian Kilb

💻
David Florness
David Florness

📖
David Walton
David Walton

💻
Debdut Karmakar
Debdut Karmakar

💻
Dieter Zhu
Dieter Zhu

💻
Fredrik Holmqvist
Fredrik Holmqvist

💻
Giovanni Palma
Giovanni Palma

💻
Hao
Hao

💻
Igor Sementsov
Igor Sementsov

💻
Johannes Haseitl
Johannes Haseitl

💻
Joshua Hull
Joshua Hull

💻
Joshua Mangiola
Joshua Mangiola

📖
Kevin MacMartin
Kevin MacMartin

💻
Liang Li
Liang Li

💻
Marvin Collins Hosea
Marvin Collins Hosea

💻
Matt Holt
Matt Holt

💻
Niklas
Niklas

💻
Andy Hsu
Andy Hsu

💻
NullCode
NullCode

💻
Oussama Sethoum
Oussama Sethoum

💻
ParkourLiu
ParkourLiu

💻
Rachel Chen
Rachel Chen

💻
Rob Nice
Rob Nice

💻
Ryo TAGAMI
Ryo TAGAMI

💻
Sam Hennessy
Sam Hennessy

💻
Sean
Sean

💻
Sean Gosiaco
Sean Gosiaco

💻
Eric P Sheets
Eric P Sheets

💻
Supian M
Supian M

💻
Watson-Sei
Watson-Sei

💻 📖
Yuki Shindo
Yuki Shindo

💻
cuigege
cuigege

💻
cybertramp
cybertramp

💻
hiroki yagi
hiroki yagi

💻
imgbot[bot]
imgbot[bot]

💻
juju
juju

💻
Michael Eatherly
Michael Eatherly

💻
tk
tk

💻
allcontributors[bot]
allcontributors[bot]

📖
wander
wander

📖
+ + + +## Special Mentions + +- [John Chadwick](https://github.com/jchv) - His amazing work on [go-webview2](https://github.com/jchv/go-webview2) and [go-winloader](https://github.com/jchv/go-winloader) have made the Windows version possible. +- [Tad Vizbaras](https://github.com/tadvi) - His winc project was the first step down the path to a pure Go Wails. +- [Mat Ryer](https://github.com/matryer) - For advice, support and bants. +- [Byron Chris](https://github.com/bh90210) - For his long term contributions to this project. +- [Dustin Krysak](https://wiki.ubuntu.com/bashfulrobot) - His support and feedback has been invaluable. +- [Justen Walker](https://github.com/justenwalker/) - For helping wrangle COM issues which got v2 over the line. +- [Wang, Chi](https://github.com/patr0nus/) - The DeskGap project was a huge influence on the direction of Wails v2. +- [Serge Zaitsev](https://github.com/zserge) - Whilst Wails does not use the Webview project, it is still a source of inspiration. diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/faq.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/faq.mdx new file mode 100644 index 00000000000..98b68d55794 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-pages/faq.mdx @@ -0,0 +1,9 @@ +# FAQ + +## What's with the name? + +When I saw WebView, I thought "What I really want is tooling around building a WebView app, a bit like Rails is to Ruby". So initially it was a play on words (**W**ebview on R**ails**). It just so happened to also be a homophone of the English name for the [Country](https://en.wikipedia.org/wiki/Wales) I am from. So it stuck. + +## Is this an alternative to Electron? + +Depends on your requirements. It's designed to make it easy for Go programmers to make lightweight desktop applications or add a frontend to their existing applications. Wails v2 does offer native elements such as menus and dialogs, so it is becoming a lightweight electron alternative. diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/markdown-page.md b/website/i18n/vi/docusaurus-plugin-content-pages/markdown-page.md new file mode 100644 index 00000000000..9756c5b6685 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-pages/markdown-page.md @@ -0,0 +1,7 @@ +--- +title: Markdown page example +--- + +# Markdown page example + +You don't need React to write simple standalone pages. diff --git a/website/i18n/vi/docusaurus-theme-classic/footer.json b/website/i18n/vi/docusaurus-theme-classic/footer.json new file mode 100644 index 00000000000..20f38e25af4 --- /dev/null +++ b/website/i18n/vi/docusaurus-theme-classic/footer.json @@ -0,0 +1,62 @@ +{ + "link.title.Docs": { + "message": "Tài Liệu", + "description": "The title of the footer links column with title=Docs in the footer" + }, + "link.title.Community": { + "message": "Cộng Đồng", + "description": "The title of the footer links column with title=Community in the footer" + }, + "link.title.More": { + "message": "Bổ Xung", + "description": "The title of the footer links column with title=More in the footer" + }, + "link.item.label.Introduction": { + "message": "Lời Giới Thiệu", + "description": "The label of footer link with label=Introduction linking to /docs/introduction" + }, + "link.item.label.Getting Started": { + "message": "Hướng Dẫn Nhanh", + "description": "The label of footer link with label=Getting Started linking to /docs/gettingstarted/installation" + }, + "link.item.label.Changelog": { + "message": "Nhật Ký Phiên Bản", + "description": "The label of footer link with label=Changelog linking to /changelog" + }, + "link.item.label.Github": { + "message": "Github", + "description": "The label of footer link with label=Github linking to https://github.com/wailsapp/wails" + }, + "link.item.label.Twitter": { + "message": "X", + "description": "The label of footer link with label=Twitter linking to https://twitter.com/wailsapp" + }, + "link.item.label.Slack": { + "message": "Slack", + "description": "The label of footer link with label=Slack linking to https://gophers.slack.com/messages/CJ4P9F7MZ/" + }, + "link.item.label.Slack invite": { + "message": "Lời Mời Slack", + "description": "The label of footer link with label=Slack invite linking to https://invite.slack.golangbridge.org/" + }, + "link.item.label.Blog": { + "message": "Blog", + "description": "The label of footer link with label=Blog linking to /blog" + }, + "copyright": { + "message": "Bản Quyền Thuộc Về © 2022 Lea Anthony", + "description": "The footer copyright" + }, + "link.item.label.Awesome": { + "message": "Awesome", + "description": "The label of footer link with label=Awesome linking to https://github.com/wailsapp/awesome-wails" + }, + "link.item.label.Discord": { + "message": "Discord", + "description": "The label of footer link with label=Discord linking to https://discord.gg/JDdSxwjhGf" + }, + "logo.alt": { + "message": "Wails Logo", + "description": "The alt text of footer logo" + } +} diff --git a/website/i18n/vi/docusaurus-theme-classic/navbar.json b/website/i18n/vi/docusaurus-theme-classic/navbar.json new file mode 100644 index 00000000000..bfed862b88b --- /dev/null +++ b/website/i18n/vi/docusaurus-theme-classic/navbar.json @@ -0,0 +1,46 @@ +{ + "item.label.Sponsor": { + "message": "Tài Trợ", + "description": "Navbar item with label Sponsor" + }, + "item.label.Docs": { + "message": "Tài Liệu", + "description": "Navbar item with label Docs" + }, + "item.label.Blog": { + "message": "Blog", + "description": "Navbar item with label Blog" + }, + "item.label.GitHub": { + "message": "GitHub", + "description": "Navbar item with label GitHub" + }, + "item.label.About": { + "message": "Thông Tin", + "description": "Navbar item with label About" + }, + "item.label.FAQ": { + "message": "Câu Hỏi Thường Gặp", + "description": "Navbar item with label FAQ" + }, + "item.label.Changelog": { + "message": "Nhật Ký Phiên Bản", + "description": "Navbar item with label Changelog" + }, + "item.label.Community Guide": { + "message": "Hướng Dẫn của Cộng Đồng", + "description": "Navbar item with label Community Guide" + }, + "item.label.Credits": { + "message": "Tín Nhiệm", + "description": "Navbar item with label Credits" + }, + "item.label.Code of Conduct": { + "message": "Quy Tắc Ứng Xử", + "description": "Navbar item with label Code of Conduct" + }, + "logo.alt": { + "message": "Wails Logo", + "description": "The alt text of navbar logo" + } +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx index b2dd2b372ad..fb487e87c79 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -60,6 +60,10 @@ sidebar_position: 1 - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - 使用函数式编程和 **快速** 的热重载设置开发您的 GUI 应用程序 :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - 结合 Elm + Tailwind CSS + Wails 的力量 :muscle: ! 支持热重载。 +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + ## 纯 JavaScript (Vanilla) - [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - 一个只有基本 JavaScript、HTML 和 CSS 的模板 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx new file mode 100644 index 00000000000..b10c66bb373 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -0,0 +1,199 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +## Limitations: + +On Windows and Linux when associated file is opened, new instance of your app is launched. +Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx index 72028ea6b12..83de0fdfd8f 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/manual-builds.mdx @@ -80,7 +80,6 @@ Wails CLI 为项目做了很多繁重的工作,但有时需要手动构建项 #### 手动步骤 - 对于 dev 构建,最少的命令是:`go build -tags dev -gcflags "all=-N -l"` - - 在macos上需要加上环境变量 `export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l"` - 对于生产构建,最少的命令是:`go build -tags desktop,production -ldflags "-w -s -H windowsgui"` - 确保在与 `.syso` 文件相同的目录中进行编译 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx index 0bf892edb88..3f12baf7d60 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/troubleshooting.mdx @@ -159,6 +159,28 @@ In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/Sy 来源:https://github.com/wailsapp/wails/issues/1806 和 https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 +## My application won't compile on Mac + +如果您遇到这样的错误: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + -- ## 无法启动服务:主机版本“x.x.x”与二进制版本“x.x.x”不匹配 @@ -185,4 +207,162 @@ This is due to the default background of the webview being white. If you want to WebviewIsTransparent: true, }, }) -``` \ No newline at end of file +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/cli.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/cli.mdx index b157a1bb5be..d28d2732c30 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/cli.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/cli.mdx @@ -49,35 +49,35 @@ Wails CLI 有许多用于管理项目的命令。 所有命令都以此方式运 `wails build` 用于将您的项目编译为生产可用的二进制文件。 -| 标志 | 描述 | 默认 | -|:--------------- |:------------------------------------------------------------------------------------------------------------- |:---------------------------------------------------------------------------------------------------------- | -| -clean | 清理 `build/bin` 目录 | | -| -compiler "编译器" | 使用不同的 go 编译器来构建,例如 go1.15beta1 | go | -| -debug | Retains debug information in the application and shows the debug console. 允许在应用程序窗口中使用 devtools | | -| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used) | | -| -dryrun | 打印构建命令但不执行它 | | -| -f | 强制构建应用 | | -| -garbleargs | 传递给 garble 的参数 | `-literals -tiny -seed=random` | -| -ldflags "标志" | 传递给编译器的额外 ldflags | | -| -m | 编译前跳过 mod tidy | | -| -nopackage | 不打包应用程序 | | -| -nocolour | 在输出中禁用颜色 | | -| -nosyncgomod | 不同步 go.mod 中的 Wails 版本 | | -| -nsis | 为 Windows 生成 NSIS 安装程序 | | -| -o 文件名 | 输出文件名 | | -| -obfuscated | 使用 [garble](https://github.com/burrowers/garble) 混淆应用程序 | | -| -platform | 为指定的 [平台](../reference/cli#平台)(逗号分割)构建,例如: `windows/arm64`。 `windows/arm64`。 注意,如果不给出架构,则使用 `runtime.GOARCH`。 | 如果给定环境变量 platform = `GOOS` 否则等于 `runtime.GOOS`。
如果给定环境变量 arch = `GOARCH` 否则等于 `runtime.GOARCH`. | -| -race | 使用 Go 的竞态检测器构建 | | -| -s | 跳过前端构建 | | -| -skipbindings | 跳过 bindings 生成 | | -| -tags "额外标签" | 构建标签以传递给 Go 编译器。 必须引用。 空格或逗号(但不能同时使用)分隔 | | -| -trimpath | 从生成的可执行文件中删除所有文件系统路径。 | | -| -u | 更新项目的 `go.mod` 以使用与 CLI 相同版本的 Wails | | -| -upx | 使用 “upx” 压缩最终二进制文件 | | -| -upxflags | 传递给 upx 的标志 | | -| -v int | 详细级别 (0 - silent, 1 - default, 2 - verbose) | 1 | -| -webview2 | WebView2 安装策略:download,embed,browser,error. | download | -| -windowsconsole | 保留Windows构建控制台窗口 | | +| 标志 | 描述 | 默认 | +|:--------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:---------------------------------------------------------------------------------------------------------- | +| -clean | 清理 `build/bin` 目录 | | +| -compiler "编译器" | 使用不同的 go 编译器来构建,例如 go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. 允许在应用程序窗口中使用 devtools | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | 打印构建命令但不执行它 | | +| -f | 强制构建应用 | | +| -garbleargs | 传递给 garble 的参数 | `-literals -tiny -seed=random` | +| -ldflags "标志" | 传递给编译器的额外 ldflags | | +| -m | 编译前跳过 mod tidy | | +| -nopackage | 不打包应用程序 | | +| -nocolour | 在输出中禁用颜色 | | +| -nosyncgomod | 不同步 go.mod 中的 Wails 版本 | | +| -nsis | 为 Windows 生成 NSIS 安装程序 | | +| -o 文件名 | 输出文件名 | | +| -obfuscated | 使用 [garble](https://github.com/burrowers/garble) 混淆应用程序 | | +| -platform | 为指定的 [平台](../reference/cli#平台)(逗号分割)构建,例如: `windows/arm64`。 `windows/arm64`。 注意,如果不给出架构,则使用 `runtime.GOARCH`。 | 如果给定环境变量 platform = `GOOS` 否则等于 `runtime.GOOS`。
如果给定环境变量 arch = `GOARCH` 否则等于 `runtime.GOARCH`. | +| -race | 使用 Go 的竞态检测器构建 | | +| -s | 跳过前端构建 | | +| -skipbindings | 跳过 bindings 生成 | | +| -tags "额外标签" | 构建标签以传递给 Go 编译器。 必须引用。 空格或逗号(但不能同时使用)分隔 | | +| -trimpath | 从生成的可执行文件中删除所有文件系统路径。 | | +| -u | 更新项目的 `go.mod` 以使用与 CLI 相同版本的 Wails | | +| -upx | 使用 “upx” 压缩最终二进制文件 | | +| -upxflags | 传递给 upx 的标志 | | +| -v int | 详细级别 (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 安装策略:download,embed,browser,error. | download | +| -windowsconsole | 保留Windows构建控制台窗口 | | 有关 `webview2` 标志的详细描述,请参阅 [Windows 系统指南](../guides/windows)。 @@ -166,8 +166,8 @@ Your system is ready for Wails development! - 启动一个网络服务器 `http://localhost:34115`,通过 http 为您的应用程序(不仅仅是前端)提供服务。 这允许您使用您喜欢的浏览器开发扩展 - 所有应用程序资源都从磁盘加载。 如果它们被更改,应用程序将自动重新加载(而不是重新构建)。 所有连接的浏览器也将重新加载 - 生成的 JS 模块提供以下内容: - - 带有自动生成的 JSDoc 的 Go 方法的 JavaScript 包装器,提供代码提示 - - 您的 Go 结构的 TypeScript 版本,可以构建并传递给您的 go 方法 +- 带有自动生成的 JSDoc 的 Go 方法的 JavaScript 包装器,提供代码提示 +- 您的 Go 结构的 TypeScript 版本,可以构建并传递给您的 go 方法 - 生成第二个 JS 模块,为运行时提供包装器 + TS 声明 - 在 macOS 上,它将应用程序捆绑到一个 `.app` 文件中并运行它。 开发模式将使用 `build/darwin/Info.dev.plist` 。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/options.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/options.mdx index 0c43e683811..d6c559d8d69 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/options.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/options.mdx @@ -366,7 +366,7 @@ func (b *App) beforeClose(ctx context.Context) (prevent bool) { EnableDefaultContextMenu enables the browser's default context-menu in production. -By default, the browser's default context-menu is only available in development and in a `-debug` or `-devtools` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : @@ -593,6 +593,12 @@ ResizeDebounceMS 是调整窗口大小时去抖动 webview2 重绘的时间量 名称:WebviewGpuIsDisabled
类型:`bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + ### Mac 这定义了 [Mac 特定的选项](#mac)。 @@ -688,6 +694,42 @@ Mac: &mac.Options{ 名称:WindowIsTranslucent
类型:`bool` +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| 设置 | 描述 | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +示例: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### 关于 此配置允许您在“AppMenu”角色创建的应用程序菜单中设置“关于”菜单项的标题、消息和图标。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx index 57ff12d08d3..4b2142d8c3c 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx @@ -13,6 +13,28 @@ ## [即将发布] +### 新增 + +- Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) +- When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) +- Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) +- Added support for enabling/disabling fullscreen of the Webview on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2953) +- Added French README.fr.md page. Added by @nejos97 in [PR](https://github.com/wailsapp/wails/pull/2943) +- New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) +- Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) +- Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c + +### 变更 + +- AssetServer requests are now processed asynchronously without blocking the main thread on Windows. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) +- Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) +- Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) +#### 修复 + +- Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) + ## v2.6.0 - 2023-09-06 ### 重大变更 @@ -27,6 +49,7 @@ - Fixed `SetBackgroundColour` so it sets the window's background color to reduce resize flickering on Linux. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2853) - Fixed disable window resize option and wrong initial window size when its enabled. Added by @lyimmi in [PR](https://github.com/wailsapp/wails/pull/2863) - Fixed build hook command parsing. Added by @smac89 in [PR](https://github.com/wailsapp/wails/pull/2836) +- Fixed filesystem watcher from filtering top level project directory if binary name is included in .gitignore. Added by [@haukened](https://github.com/haukened) in [PR #2869](https://github.com/wailsapp/wails/pull/2869) - Fixed `-reloaddir` flag to watch additional directories (non-recursively). [@haukened](https://github.com/haukened) in [PR #2871](https://github.com/wailsapp/wails/pull/2871) - Fixed support for Go 1.21 `go.mod` files. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2876) @@ -44,6 +67,7 @@ - Added new community template wails-sveltekit-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2851) - Added support for print dialogs. Added by [@aangelisc](https://github.com/aangelisc) in [PR](https://github.com/wailsapp/wails/pull/2822) - Added new `wails dev -nogorebuild` flag to prevent restarts on back end file changes. [@haukened](https://github.com/haukened) in [PR #2870](https://github.com/wailsapp/wails/pull/2870) +- Added a guide to describe a cross-platform build using GitHub Actions. Added by [@dennypenta](https://github.com/dennypenta) in [PR](https://github.com/wailsapp/wails/pull/2879) ### 变更 @@ -680,9 +704,6 @@ Experimental: &options.Experimental{ - [v2, nsis] Seems like / as path separator works only for some directives in a cross platform way by [@stffabi](https://github.com/stffabi) in #1227 - import models on binding definition by [@adalessa](https://github.com/adalessa) in #123 - - 1 - - Use local search on website by [@leaanthony](https://github.com/leaanthony) in #1234 - Ensure binary resources can be served by [@napalu](https://github.com/napalu) in #1240 - Only retry loading assets when loading from disk by [@leaanthony](https://github.com/leaanthony) in #1241 From c24bd5e3e889a3bfe9f6fb431d9e12430e0b41b0 Mon Sep 17 00:00:00 2001 From: Andrey Pshenkin Date: Mon, 23 Oct 2023 11:31:56 +0100 Subject: [PATCH 26/87] Implement Single instance lock feature with passing arguments to initial instance (#2951) * implement MacOS openFile/openFiles events * wip: windows file association * fix macro import * add file icon copy * try copy icon * keep only required part of scripts * update config schema * fix json * set fileAssociation for mac via config * proper iconName handling * add fileAssociation icon generator * fix file association icons bundle * don't break compatibility * remove mimeType as not supported linux for now * add documentation * adjust config schema * restore formatting * try implement single instance lock with params passing * fix focusing * fix focusing * formatting * use channel buffer for second instance events * handle errors * add comment * remove unused option in file association * wip: linux single instance lock * wip: linux single instance * some experiments with making window active * try to use unminimise * remove unused * try present for window * try present for window * fix build * cleanup * cleanup * implement single instance lock on mac os * implement proper show for windows * proper unmimimise * get rid of openFiles mac os. change configuration structure * remove unused channel * remove unused function * add documentation for single instance lock * add PR link * changes after review * update docs * changes after review --------- Co-authored-by: Lea Anthony --- v2/go.mod | 1 + v2/go.sum | 2 + .../frontend/desktop/darwin/AppDelegate.h | 6 + .../frontend/desktop/darwin/AppDelegate.m | 22 +++ .../frontend/desktop/darwin/Application.h | 2 +- .../frontend/desktop/darwin/Application.m | 39 +++--- .../frontend/desktop/darwin/WailsContext.h | 3 + .../frontend/desktop/darwin/frontend.go | 15 +++ .../desktop/darwin/single_instance.go | 75 +++++++++++ v2/internal/frontend/desktop/darwin/window.go | 11 +- .../frontend/desktop/linux/frontend.go | 17 +++ .../frontend/desktop/linux/single_instance.go | 69 ++++++++++ .../frontend/desktop/windows/frontend.go | 17 +++ .../desktop/windows/single_instance.go | 126 ++++++++++++++++++ .../desktop/windows/winc/controlbase.go | 18 ++- .../frontend/desktop/windows/winc/form.go | 9 ++ .../desktop/windows/winc/w32/user32.go | 9 ++ v2/pkg/options/options.go | 28 ++++ website/docs/guides/file-association.mdx | 50 ++++++- website/docs/guides/single-instance-lock.mdx | 80 +++++++++++ website/src/pages/changelog.mdx | 1 + 21 files changed, 578 insertions(+), 22 deletions(-) create mode 100644 v2/internal/frontend/desktop/darwin/single_instance.go create mode 100644 v2/internal/frontend/desktop/linux/single_instance.go create mode 100644 v2/internal/frontend/desktop/windows/single_instance.go create mode 100644 website/docs/guides/single-instance-lock.mdx diff --git a/v2/go.mod b/v2/go.mod index 891928653bc..b707a5d5512 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -12,6 +12,7 @@ require ( github.com/fsnotify/fsnotify v1.4.9 github.com/go-git/go-git/v5 v5.3.0 github.com/go-ole/go-ole v1.2.6 + github.com/godbus/dbus/v5 v5.1.0 github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 github.com/google/uuid v1.3.0 github.com/jackmordaunt/icns v1.0.0 diff --git a/v2/go.sum b/v2/go.sum index c3faec7c0b5..dd11b8555ee 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -71,6 +71,8 @@ github.com/go-git/go-git/v5 v5.3.0/go.mod h1:xdX4bWJ48aOrdhnl2XqHYstHbbp6+LFS4r4 github.com/go-ole/go-ole v1.2.5/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= +github.com/godbus/dbus/v5 v5.1.0 h1:4KLkAxT3aOY8Li4FRJe/KvhoNFFxo0m6fNuFUO8QJUk= +github.com/godbus/dbus/v5 v5.1.0/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= diff --git a/v2/internal/frontend/desktop/darwin/AppDelegate.h b/v2/internal/frontend/desktop/darwin/AppDelegate.h index 7b533ad5ff7..4fa32233e8b 100644 --- a/v2/internal/frontend/desktop/darwin/AppDelegate.h +++ b/v2/internal/frontend/desktop/darwin/AppDelegate.h @@ -15,6 +15,8 @@ @property bool alwaysOnTop; @property bool startHidden; +@property (retain) NSString* singleInstanceUniqueId; +@property bool singleInstanceLockEnabled; @property bool startFullscreen; @property (retain) WailsWindow* mainWindow; @@ -22,4 +24,8 @@ extern void HandleOpenFile(char *); +extern void HandleSecondInstanceData(char * message); + +void SendDataToFirstInstance(char * singleInstanceUniqueId, char * text); + #endif /* AppDelegate_h */ diff --git a/v2/internal/frontend/desktop/darwin/AppDelegate.m b/v2/internal/frontend/desktop/darwin/AppDelegate.m index 6a9f5a4c379..55bc2b4a817 100644 --- a/v2/internal/frontend/desktop/darwin/AppDelegate.m +++ b/v2/internal/frontend/desktop/darwin/AppDelegate.m @@ -40,6 +40,28 @@ - (void)applicationDidFinishLaunching:(NSNotification *)aNotification { [self.mainWindow setCollectionBehavior:behaviour]; [self.mainWindow toggleFullScreen:nil]; } + + if ( self.singleInstanceLockEnabled ) { + [[NSDistributedNotificationCenter defaultCenter] addObserver:self + selector:@selector(handleSecondInstanceNotification:) name:self.singleInstanceUniqueId object:nil]; + } +} + +void SendDataToFirstInstance(char * singleInstanceUniqueId, char * message) { + [[NSDistributedNotificationCenter defaultCenter] + postNotificationName:[NSString stringWithUTF8String:singleInstanceUniqueId] + object:nil + userInfo:@{@"message": [NSString stringWithUTF8String:message]} + deliverImmediately:YES]; +} + +- (void)handleSecondInstanceNotification:(NSNotification *)note; +{ + if (note.userInfo[@"message"] != nil) { + NSString *message = note.userInfo[@"message"]; + const char* utf8Message = message.UTF8String; + HandleSecondInstanceData((char*)utf8Message); + } } - (void)dealloc { diff --git a/v2/internal/frontend/desktop/darwin/Application.h b/v2/internal/frontend/desktop/darwin/Application.h index 458016cc821..eb679f3b5eb 100644 --- a/v2/internal/frontend/desktop/darwin/Application.h +++ b/v2/internal/frontend/desktop/darwin/Application.h @@ -17,7 +17,7 @@ #define WindowStartsMinimised 2 #define WindowStartsFullscreen 3 -WailsContext* Create(const char* title, int width, int height, int frameless, int resizable, int fullscreen, int fullSizeContent, int hideTitleBar, int titlebarAppearsTransparent, int hideTitle, int useToolbar, int hideToolbarSeparator, int webviewIsTransparent, int alwaysOnTop, int hideWindowOnClose, const char *appearance, int windowIsTranslucent, int devtoolsEnabled, int defaultContextMenuEnabled, int windowStartState, int startsHidden, int minWidth, int minHeight, int maxWidth, int maxHeight, bool fraudulentWebsiteWarningEnabled, struct Preferences preferences); +WailsContext* Create(const char* title, int width, int height, int frameless, int resizable, int fullscreen, int fullSizeContent, int hideTitleBar, int titlebarAppearsTransparent, int hideTitle, int useToolbar, int hideToolbarSeparator, int webviewIsTransparent, int alwaysOnTop, int hideWindowOnClose, const char *appearance, int windowIsTranslucent, int devtoolsEnabled, int defaultContextMenuEnabled, int windowStartState, int startsHidden, int minWidth, int minHeight, int maxWidth, int maxHeight, bool fraudulentWebsiteWarningEnabled, struct Preferences preferences, int singleInstanceEnabled, const char* singleInstanceUniqueId); void Run(void*, const char* url); void SetTitle(void* ctx, const char *title); diff --git a/v2/internal/frontend/desktop/darwin/Application.m b/v2/internal/frontend/desktop/darwin/Application.m index e0c05212dca..aa35b2a6bea 100644 --- a/v2/internal/frontend/desktop/darwin/Application.m +++ b/v2/internal/frontend/desktop/darwin/Application.m @@ -14,15 +14,15 @@ #import "WailsMenu.h" #import "WailsMenuItem.h" -WailsContext* Create(const char* title, int width, int height, int frameless, int resizable, int fullscreen, int fullSizeContent, int hideTitleBar, int titlebarAppearsTransparent, int hideTitle, int useToolbar, int hideToolbarSeparator, int webviewIsTransparent, int alwaysOnTop, int hideWindowOnClose, const char *appearance, int windowIsTranslucent, int devtoolsEnabled, int defaultContextMenuEnabled, int windowStartState, int startsHidden, int minWidth, int minHeight, int maxWidth, int maxHeight, bool fraudulentWebsiteWarningEnabled, struct Preferences preferences) { - +WailsContext* Create(const char* title, int width, int height, int frameless, int resizable, int fullscreen, int fullSizeContent, int hideTitleBar, int titlebarAppearsTransparent, int hideTitle, int useToolbar, int hideToolbarSeparator, int webviewIsTransparent, int alwaysOnTop, int hideWindowOnClose, const char *appearance, int windowIsTranslucent, int devtoolsEnabled, int defaultContextMenuEnabled, int windowStartState, int startsHidden, int minWidth, int minHeight, int maxWidth, int maxHeight, bool fraudulentWebsiteWarningEnabled, struct Preferences preferences, int singleInstanceLockEnabled, const char* singleInstanceUniqueId) { + [NSApplication sharedApplication]; WailsContext *result = [WailsContext new]; result.devtoolsEnabled = devtoolsEnabled; result.defaultContextMenuEnabled = defaultContextMenuEnabled; - + if ( windowStartState == WindowStartsFullscreen ) { fullscreen = 1; } @@ -30,7 +30,7 @@ [result CreateWindow:width :height :frameless :resizable :fullscreen :fullSizeContent :hideTitleBar :titlebarAppearsTransparent :hideTitle :useToolbar :hideToolbarSeparator :webviewIsTransparent :hideWindowOnClose :safeInit(appearance) :windowIsTranslucent :minWidth :minHeight :maxWidth :maxHeight :fraudulentWebsiteWarningEnabled :preferences]; [result SetTitle:safeInit(title)]; [result Center]; - + switch( windowStartState ) { case WindowStartsMaximised: [result.mainWindow zoom:nil]; @@ -43,14 +43,19 @@ if ( startsHidden == 1 ) { result.startHidden = true; } - + if ( fullscreen == 1 ) { result.startFullscreen = true; } - + + if ( singleInstanceLockEnabled == 1 ) { + result.singleInstanceLockEnabled = true; + result.singleInstanceUniqueId = safeInit(singleInstanceUniqueId); + } + result.alwaysOnTop = alwaysOnTop; result.hideOnClose = hideWindowOnClose; - + return result; } @@ -181,7 +186,7 @@ void ToggleMaximise(void* inctx) { NSString *result = [NSString stringWithFormat:@"%d,%d",x,y]; return [result UTF8String]; } - + const bool IsFullScreen(void *inctx) { WailsContext *ctx = (__bridge WailsContext*) inctx; return [ctx IsFullScreen]; @@ -249,7 +254,7 @@ void ShowApplication(void *inctx) { void MessageDialog(void *inctx, const char* dialogType, const char* title, const char* message, const char* button1, const char* button2, const char* button3, const char* button4, const char* defaultButton, const char* cancelButton, void* iconData, int iconDataLength) { WailsContext *ctx = (__bridge WailsContext*) inctx; - + NSString *_dialogType = safeInit(dialogType); NSString *_title = safeInit(title); NSString *_message = safeInit(message); @@ -259,33 +264,33 @@ void MessageDialog(void *inctx, const char* dialogType, const char* title, const NSString *_button4 = safeInit(button4); NSString *_defaultButton = safeInit(defaultButton); NSString *_cancelButton = safeInit(cancelButton); - + ON_MAIN_THREAD( [ctx MessageDialog:_dialogType :_title :_message :_button1 :_button2 :_button3 :_button4 :_defaultButton :_cancelButton :iconData :iconDataLength]; ) } void OpenFileDialog(void *inctx, const char* title, const char* defaultFilename, const char* defaultDirectory, int allowDirectories, int allowFiles, int canCreateDirectories, int treatPackagesAsDirectories, int resolveAliases, int showHiddenFiles, int allowMultipleSelection, const char* filters) { - + WailsContext *ctx = (__bridge WailsContext*) inctx; NSString *_title = safeInit(title); NSString *_defaultFilename = safeInit(defaultFilename); NSString *_defaultDirectory = safeInit(defaultDirectory); NSString *_filters = safeInit(filters); - + ON_MAIN_THREAD( [ctx OpenFileDialog:_title :_defaultFilename :_defaultDirectory :allowDirectories :allowFiles :canCreateDirectories :treatPackagesAsDirectories :resolveAliases :showHiddenFiles :allowMultipleSelection :_filters]; ) } void SaveFileDialog(void *inctx, const char* title, const char* defaultFilename, const char* defaultDirectory, int canCreateDirectories, int treatPackagesAsDirectories, int showHiddenFiles, const char* filters) { - + WailsContext *ctx = (__bridge WailsContext*) inctx; NSString *_title = safeInit(title); NSString *_defaultFilename = safeInit(defaultFilename); NSString *_defaultDirectory = safeInit(defaultDirectory); NSString *_filters = safeInit(filters); - + ON_MAIN_THREAD( [ctx SaveFileDialog:_title :_defaultFilename :_defaultDirectory :canCreateDirectories :treatPackagesAsDirectories :showHiddenFiles :_filters]; ) @@ -367,6 +372,8 @@ void Run(void *inctx, const char* url) { delegate.mainWindow = ctx.mainWindow; delegate.alwaysOnTop = ctx.alwaysOnTop; delegate.startHidden = ctx.startHidden; + delegate.singleInstanceLockEnabled = ctx.singleInstanceLockEnabled; + delegate.singleInstanceUniqueId = ctx.singleInstanceUniqueId; delegate.startFullscreen = ctx.startFullscreen; NSString *_url = safeInit(url); @@ -395,7 +402,7 @@ void WindowPrint(void *inctx) { WailsContext *ctx = (__bridge WailsContext*) inctx; WKWebView* webView = ctx.webview; - // I think this should be exposed as a config + // I think this should be exposed as a config // It directly affects the printed output/PDF NSPrintInfo *pInfo = [NSPrintInfo sharedPrintInfo]; pInfo.horizontalPagination = NSPrintingPaginationModeAutomatic; @@ -417,4 +424,4 @@ void WindowPrint(void *inctx) { [po runOperationModalForWindow:ctx.mainWindow delegate:ctx.mainWindow.delegate didRunSelector:nil contextInfo:nil]; ) } -} \ No newline at end of file +} diff --git a/v2/internal/frontend/desktop/darwin/WailsContext.h b/v2/internal/frontend/desktop/darwin/WailsContext.h index 99b6ce2def7..ab5e5c2a201 100644 --- a/v2/internal/frontend/desktop/darwin/WailsContext.h +++ b/v2/internal/frontend/desktop/darwin/WailsContext.h @@ -40,6 +40,9 @@ @property bool startHidden; @property bool startFullscreen; +@property bool singleInstanceLockEnabled; +@property (retain) NSString* singleInstanceUniqueId; + @property (retain) NSEvent* mouseEvent; @property bool alwaysOnTop; diff --git a/v2/internal/frontend/desktop/darwin/frontend.go b/v2/internal/frontend/desktop/darwin/frontend.go index 66e5dc0cf91..7a4905eeb1d 100644 --- a/v2/internal/frontend/desktop/darwin/frontend.go +++ b/v2/internal/frontend/desktop/darwin/frontend.go @@ -39,6 +39,7 @@ var messageBuffer = make(chan string, 100) var requestBuffer = make(chan webview.Request, 100) var callbackBuffer = make(chan uint, 10) var openFilepathBuffer = make(chan string, 100) +var secondInstanceBuffer = make(chan options.SecondInstanceData, 1) type Frontend struct { @@ -109,6 +110,7 @@ func NewFrontend(ctx context.Context, appoptions *options.App, myLogger *logger. go result.startMessageProcessor() go result.startCallbackProcessor() go result.startFileOpenProcessor() + go result.startSecondInstanceProcessor() return result } @@ -119,6 +121,15 @@ func (f *Frontend) startFileOpenProcessor() { } } +func (f *Frontend) startSecondInstanceProcessor() { + for secondInstanceData := range secondInstanceBuffer { + if f.frontendOptions.SingleInstanceLock != nil && + f.frontendOptions.SingleInstanceLock.OnSecondInstanceLaunch != nil { + f.frontendOptions.SingleInstanceLock.OnSecondInstanceLaunch(secondInstanceData) + } + } +} + func (f *Frontend) startMessageProcessor() { for message := range messageBuffer { f.processMessage(message) @@ -162,6 +173,10 @@ func (f *Frontend) WindowSetDarkTheme() { func (f *Frontend) Run(ctx context.Context) error { f.ctx = ctx + if f.frontendOptions.SingleInstanceLock != nil { + SetupSingleInstance(f.frontendOptions.SingleInstanceLock.UniqueId) + } + var _debug = ctx.Value("debug") var _devtoolsEnabled = ctx.Value("devtoolsEnabled") diff --git a/v2/internal/frontend/desktop/darwin/single_instance.go b/v2/internal/frontend/desktop/darwin/single_instance.go new file mode 100644 index 00000000000..6f61ed1734f --- /dev/null +++ b/v2/internal/frontend/desktop/darwin/single_instance.go @@ -0,0 +1,75 @@ +//go:build darwin +// +build darwin + +package darwin + +/* +#cgo CFLAGS: -x objective-c +#cgo LDFLAGS: -framework Foundation -framework Cocoa +#import "AppDelegate.h" + +*/ +import "C" +import ( + "encoding/json" + "github.com/wailsapp/wails/v2/pkg/options" + "os" + "syscall" +) + +func SetupSingleInstance(uniqueID string) { + lockFilePath := os.TempDir() + lockFileName := uniqueID + ".lock" + _, err := createLockFile(lockFilePath + "/" + lockFileName) + + // if lockFile exist – send notification to second instance + if err != nil { + c := NewCalloc() + defer c.Free() + singleInstanceUniqueId := c.String(uniqueID) + + data, err := options.NewSecondInstanceData() + if err != nil { + return + } + + serialized, err := json.Marshal(data) + if err != nil { + return + } + + C.SendDataToFirstInstance(singleInstanceUniqueId, c.String(string(serialized))) + + os.Exit(0) + } +} + +//export HandleSecondInstanceData +func HandleSecondInstanceData(secondInstanceMessage *C.char) { + message := C.GoString(secondInstanceMessage) + + var secondInstanceData options.SecondInstanceData + + err := json.Unmarshal([]byte(message), &secondInstanceData) + if err == nil { + secondInstanceBuffer <- secondInstanceData + } +} + +// CreateLockFile tries to create a file with given name and acquire an +// exclusive lock on it. If the file already exists AND is still locked, it will +// fail. +func createLockFile(filename string) (*os.File, error) { + file, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0600) + if err != nil { + return nil, err + } + + err = syscall.Flock(int(file.Fd()), syscall.LOCK_EX|syscall.LOCK_NB) + if err != nil { + file.Close() + return nil, err + } + + return file, nil +} diff --git a/v2/internal/frontend/desktop/darwin/window.go b/v2/internal/frontend/desktop/darwin/window.go index c0f62741920..740848df144 100644 --- a/v2/internal/frontend/desktop/darwin/window.go +++ b/v2/internal/frontend/desktop/darwin/window.go @@ -58,6 +58,7 @@ func NewWindow(frontendOptions *options.App, debug bool, devtools bool) *Window startsHidden := bool2Cint(frontendOptions.StartHidden) devtoolsEnabled := bool2Cint(devtools) defaultContextMenuEnabled := bool2Cint(debug || frontendOptions.EnableDefaultContextMenu) + singleInstanceEnabled := bool2Cint(frontendOptions.SingleInstanceLock != nil) var fullSizeContent, hideTitleBar, hideTitle, useToolbar, webviewIsTransparent C.int var titlebarAppearsTransparent, hideToolbarSeparator, windowIsTranslucent C.int @@ -74,6 +75,12 @@ func NewWindow(frontendOptions *options.App, debug bool, devtools bool) *Window title = c.String(frontendOptions.Title) + singleInstanceUniqueIdStr := "" + if frontendOptions.SingleInstanceLock != nil { + singleInstanceUniqueIdStr = frontendOptions.SingleInstanceLock.UniqueId + } + singleInstanceUniqueId := c.String(singleInstanceUniqueIdStr) + enableFraudulentWebsiteWarnings := C.bool(frontendOptions.EnableFraudulentWebsiteDetection) if frontendOptions.Mac != nil { @@ -109,7 +116,9 @@ func NewWindow(frontendOptions *options.App, debug bool, devtools bool) *Window var context *C.WailsContext = C.Create(title, width, height, frameless, resizable, fullscreen, fullSizeContent, hideTitleBar, titlebarAppearsTransparent, hideTitle, useToolbar, hideToolbarSeparator, webviewIsTransparent, alwaysOnTop, hideWindowOnClose, appearance, windowIsTranslucent, devtoolsEnabled, defaultContextMenuEnabled, - windowStartState, startsHidden, minWidth, minHeight, maxWidth, maxHeight, enableFraudulentWebsiteWarnings, preferences) + windowStartState, startsHidden, minWidth, minHeight, maxWidth, maxHeight, enableFraudulentWebsiteWarnings, + preferences, singleInstanceEnabled, singleInstanceUniqueId, + ) // Create menu result := &Window{ diff --git a/v2/internal/frontend/desktop/linux/frontend.go b/v2/internal/frontend/desktop/linux/frontend.go index 70f8468b674..2410457b780 100644 --- a/v2/internal/frontend/desktop/linux/frontend.go +++ b/v2/internal/frontend/desktop/linux/frontend.go @@ -102,6 +102,8 @@ var initOnce = sync.Once{} const startURL = "wails://wails/" +var secondInstanceBuffer = make(chan options.SecondInstanceData, 1) + type Frontend struct { // Context @@ -201,6 +203,8 @@ func NewFrontend(ctx context.Context, appoptions *options.App, myLogger *logger. C.free(unsafe.Pointer(prgname)) } + go result.startSecondInstanceProcessor() + return result } @@ -235,6 +239,10 @@ func (f *Frontend) Run(ctx context.Context) error { } }() + if f.frontendOptions.SingleInstanceLock != nil { + SetupSingleInstance(f.frontendOptions.SingleInstanceLock.UniqueId) + } + f.mainWindow.Run(f.startURL.String()) return nil @@ -505,3 +513,12 @@ func (f *Frontend) startRequestProcessor() { func processURLRequest(request unsafe.Pointer) { requestBuffer <- webview.NewRequest(request) } + +func (f *Frontend) startSecondInstanceProcessor() { + for secondInstanceData := range secondInstanceBuffer { + if f.frontendOptions.SingleInstanceLock != nil && + f.frontendOptions.SingleInstanceLock.OnSecondInstanceLaunch != nil { + f.frontendOptions.SingleInstanceLock.OnSecondInstanceLaunch(secondInstanceData) + } + } +} diff --git a/v2/internal/frontend/desktop/linux/single_instance.go b/v2/internal/frontend/desktop/linux/single_instance.go new file mode 100644 index 00000000000..9d6426e141a --- /dev/null +++ b/v2/internal/frontend/desktop/linux/single_instance.go @@ -0,0 +1,69 @@ +//go:build linux +// +build linux + +package linux + +import ( + "encoding/json" + "github.com/godbus/dbus/v5" + "github.com/wailsapp/wails/v2/pkg/options" + "os" + "strings" +) + +type dbusHandler func(string) + +func (f dbusHandler) SendMessage(message string) *dbus.Error { + f(message) + return nil +} + +func SetupSingleInstance(uniqueID string) { + id := "wails_app_" + strings.ReplaceAll(strings.ReplaceAll(uniqueID, "-", "_"), ".", "_") + + dbusName := "org." + id + ".SingleInstance" + dbusPath := "/org/" + id + "/SingleInstance" + + conn, err := dbus.ConnectSessionBus() + // if we will reach any error during establishing connection or sending message we will just continue. + // It should not be the case that such thing will happen actually, but just in case. + if err != nil { + return + } + + f := dbusHandler(func(message string) { + var secondInstanceData options.SecondInstanceData + + err := json.Unmarshal([]byte(message), &secondInstanceData) + if err == nil { + secondInstanceBuffer <- secondInstanceData + } + }) + + err = conn.Export(f, dbus.ObjectPath(dbusPath), dbusName) + if err != nil { + return + } + + reply, err := conn.RequestName(dbusName, dbus.NameFlagDoNotQueue) + if err != nil { + return + } + + // if name already taken, try to send args to existing instance, if no success just launch new instance + if reply == dbus.RequestNameReplyExists { + data := options.SecondInstanceData{ + Args: os.Args[1:], + } + serialized, err := json.Marshal(data) + if err != nil { + return + } + + err = conn.Object(dbusName, dbus.ObjectPath(dbusPath)).Call(dbusName+".SendMessage", 0, string(serialized)).Store() + if err != nil { + return + } + os.Exit(1) + } +} diff --git a/v2/internal/frontend/desktop/windows/frontend.go b/v2/internal/frontend/desktop/windows/frontend.go index c388d738472..7671ad7420b 100644 --- a/v2/internal/frontend/desktop/windows/frontend.go +++ b/v2/internal/frontend/desktop/windows/frontend.go @@ -35,6 +35,8 @@ import ( const startURL = "http://wails.localhost/" +var secondInstanceBuffer = make(chan options.SecondInstanceData, 1) + type Screen = frontend.Screen type Frontend struct { @@ -113,6 +115,8 @@ func NewFrontend(ctx context.Context, appoptions *options.App, myLogger *logger. } result.assets = assets + go result.startSecondInstanceProcessor() + return result } @@ -137,6 +141,10 @@ func (f *Frontend) Run(ctx context.Context) error { f.chromium = edge.NewChromium() + if f.frontendOptions.SingleInstanceLock != nil { + SetupSingleInstance(f.frontendOptions.SingleInstanceLock.UniqueId) + } + mainWindow := NewWindow(nil, f.frontendOptions, f.versionInfo, f.chromium) f.mainWindow = mainWindow @@ -826,3 +834,12 @@ func (f *Frontend) ShowWindow() { func (f *Frontend) onFocus(arg *winc.Event) { f.chromium.Focus() } + +func (f *Frontend) startSecondInstanceProcessor() { + for secondInstanceData := range secondInstanceBuffer { + if f.frontendOptions.SingleInstanceLock != nil && + f.frontendOptions.SingleInstanceLock.OnSecondInstanceLaunch != nil { + f.frontendOptions.SingleInstanceLock.OnSecondInstanceLaunch(secondInstanceData) + } + } +} diff --git a/v2/internal/frontend/desktop/windows/single_instance.go b/v2/internal/frontend/desktop/windows/single_instance.go new file mode 100644 index 00000000000..222d9fe1a99 --- /dev/null +++ b/v2/internal/frontend/desktop/windows/single_instance.go @@ -0,0 +1,126 @@ +//go:build windows + +package windows + +import ( + "encoding/json" + "github.com/wailsapp/wails/v2/internal/frontend/desktop/windows/winc/w32" + "github.com/wailsapp/wails/v2/pkg/options" + "golang.org/x/sys/windows" + "os" + "syscall" + "unsafe" +) + +type COPYDATASTRUCT struct { + dwData uintptr + cbData uint32 + lpData uintptr +} + +// WMCOPYDATA_SINGLE_INSTANCE_DATA we define our own type for WM_COPYDATA message +const WMCOPYDATA_SINGLE_INSTANCE_DATA = 1542 + +func SendMessage(hwnd w32.HWND, data string) { + arrUtf16, _ := syscall.UTF16FromString(data) + + pCopyData := new(COPYDATASTRUCT) + pCopyData.dwData = WMCOPYDATA_SINGLE_INSTANCE_DATA + pCopyData.cbData = uint32(len(arrUtf16)*2 + 1) + pCopyData.lpData = uintptr(unsafe.Pointer(windows.StringToUTF16Ptr(data))) + + w32.SendMessage(hwnd, w32.WM_COPYDATA, 0, uintptr(unsafe.Pointer(pCopyData))) +} + +// SetupSingleInstance single instance Windows app +func SetupSingleInstance(uniqueId string) { + id := "wails-app-" + uniqueId + + className := id + "-sic" + windowName := id + "-siw" + mutexName := id + "sim" + + _, err := windows.CreateMutex(nil, false, windows.StringToUTF16Ptr(mutexName)) + + if err != nil { + if err == windows.ERROR_ALREADY_EXISTS { + // app is already running + hwnd := w32.FindWindowW(windows.StringToUTF16Ptr(className), windows.StringToUTF16Ptr(windowName)) + + if hwnd != 0 { + data := options.SecondInstanceData{ + Args: os.Args[1:], + } + serialized, _ := json.Marshal(data) + + SendMessage(hwnd, string(serialized)) + // exit second instance of app after sending message + os.Exit(0) + } + // if we got any other unknown error we will just start new application instance + } + } else { + createEventTargetWindow(className, windowName) + } +} + +func createEventTargetWindow(className string, windowName string) w32.HWND { + // callback handler in the event target window + wndProc := func( + hwnd w32.HWND, msg uint32, wparam w32.WPARAM, lparam w32.LPARAM, + ) w32.LRESULT { + if msg == w32.WM_COPYDATA { + ldata := (*COPYDATASTRUCT)(unsafe.Pointer(lparam)) + + if ldata.dwData == WMCOPYDATA_SINGLE_INSTANCE_DATA { + serialized := windows.UTF16PtrToString((*uint16)(unsafe.Pointer(ldata.lpData))) + + var secondInstanceData options.SecondInstanceData + + err := json.Unmarshal([]byte(serialized), &secondInstanceData) + + if err == nil { + secondInstanceBuffer <- secondInstanceData + } + } + + return w32.LRESULT(0) + } + + return w32.DefWindowProc(hwnd, msg, wparam, lparam) + } + + var class w32.WNDCLASSEX + class.Size = uint32(unsafe.Sizeof(class)) + class.Style = 0 + class.WndProc = syscall.NewCallback(wndProc) + class.ClsExtra = 0 + class.WndExtra = 0 + class.Instance = w32.GetModuleHandle("") + class.Icon = 0 + class.Cursor = 0 + class.Background = 0 + class.MenuName = nil + class.ClassName = windows.StringToUTF16Ptr(className) + class.IconSm = 0 + + w32.RegisterClassEx(&class) + + // create event window that will not be visible for user + hwnd := w32.CreateWindowEx( + 0, + windows.StringToUTF16Ptr(className), + windows.StringToUTF16Ptr(windowName), + 0, + 0, + 0, + 0, + 0, + w32.HWND_MESSAGE, + 0, + w32.GetModuleHandle(""), + nil, + ) + + return hwnd +} diff --git a/v2/internal/frontend/desktop/windows/winc/controlbase.go b/v2/internal/frontend/desktop/windows/winc/controlbase.go index b745cb1b01e..086609aedde 100644 --- a/v2/internal/frontend/desktop/windows/winc/controlbase.go +++ b/v2/internal/frontend/desktop/windows/winc/controlbase.go @@ -334,7 +334,23 @@ func (cba *ControlBase) ClientHeight() int { } func (cba *ControlBase) Show() { - w32.ShowWindow(cba.hwnd, w32.SW_SHOWDEFAULT) + // WindowPos is used with HWND_TOPMOST to guarantee bring our app on top + // force set our main window on top + w32.SetWindowPos( + cba.hwnd, + w32.HWND_TOPMOST, + 0, 0, 0, 0, + w32.SWP_SHOWWINDOW|w32.SWP_NOSIZE|w32.SWP_NOMOVE, + ) + // remove topmost to allow normal windows manipulations + w32.SetWindowPos( + cba.hwnd, + w32.HWND_NOTOPMOST, + 0, 0, 0, 0, + w32.SWP_SHOWWINDOW|w32.SWP_NOSIZE|w32.SWP_NOMOVE, + ) + // put main window on tops foreground + w32.SetForegroundWindow(cba.hwnd) } func (cba *ControlBase) Hide() { diff --git a/v2/internal/frontend/desktop/windows/winc/form.go b/v2/internal/frontend/desktop/windows/winc/form.go index 9b9cadb2ce4..8a42d63f399 100644 --- a/v2/internal/frontend/desktop/windows/winc/form.go +++ b/v2/internal/frontend/desktop/windows/winc/form.go @@ -136,6 +136,15 @@ func (fm *Form) Minimise() { } func (fm *Form) Restore() { + // SC_RESTORE param for WM_SYSCOMMAND to restore app if it is minimized + const SC_RESTORE = 0xF120 + // restore the minimized window, if it is + w32.SendMessage( + fm.hwnd, + w32.WM_SYSCOMMAND, + SC_RESTORE, + 0, + ) w32.ShowWindow(fm.hwnd, w32.SW_RESTORE) } diff --git a/v2/internal/frontend/desktop/windows/winc/w32/user32.go b/v2/internal/frontend/desktop/windows/winc/w32/user32.go index ec5e4a5960c..8ca72ce4b44 100644 --- a/v2/internal/frontend/desktop/windows/winc/w32/user32.go +++ b/v2/internal/frontend/desktop/windows/winc/w32/user32.go @@ -24,6 +24,7 @@ var ( procShowWindowAsync = moduser32.NewProc("ShowWindowAsync") procUpdateWindow = moduser32.NewProc("UpdateWindow") procCreateWindowEx = moduser32.NewProc("CreateWindowExW") + procFindWindowW = moduser32.NewProc("FindWindowW") procAdjustWindowRect = moduser32.NewProc("AdjustWindowRect") procAdjustWindowRectEx = moduser32.NewProc("AdjustWindowRectEx") procDestroyWindow = moduser32.NewProc("DestroyWindow") @@ -263,6 +264,14 @@ func CreateWindowEx(exStyle uint, className, windowName *uint16, return HWND(ret) } +func FindWindowW(className, windowName *uint16) HWND { + ret, _, _ := procFindWindowW.Call( + uintptr(unsafe.Pointer(className)), + uintptr(unsafe.Pointer(windowName))) + + return HWND(ret) +} + func AdjustWindowRectEx(rect *RECT, style uint, menu bool, exStyle uint) bool { ret, _, _ := procAdjustWindowRectEx.Call( uintptr(unsafe.Pointer(rect)), diff --git a/v2/pkg/options/options.go b/v2/pkg/options/options.go index b64befe12cf..1ecad7fb919 100644 --- a/v2/pkg/options/options.go +++ b/v2/pkg/options/options.go @@ -5,6 +5,8 @@ import ( "html" "io/fs" "net/http" + "os" + "path/filepath" "runtime" "github.com/wailsapp/wails/v2/pkg/options/assetserver" @@ -82,6 +84,8 @@ type App struct { // services of Apple and Microsoft. EnableFraudulentWebsiteDetection bool + SingleInstanceLock *SingleInstanceLock + Windows *windows.Options Mac *mac.Options Linux *linux.Options @@ -165,6 +169,30 @@ func MergeDefaults(appoptions *App) { processDragOptions(appoptions) } +type SingleInstanceLock struct { + // uniqueId that will be used for setting up messaging between instances + UniqueId string + OnSecondInstanceLaunch func(secondInstanceData SecondInstanceData) +} + +type SecondInstanceData struct { + Args []string + WorkingDirectory string +} + +func NewSecondInstanceData() (*SecondInstanceData, error) { + ex, err := os.Executable() + if err != nil { + return nil, err + } + workingDirectory := filepath.Dir(ex) + + return &SecondInstanceData{ + Args: os.Args[1:], + WorkingDirectory: workingDirectory, + }, nil +} + func processMenus(appoptions *App) { switch runtime.GOOS { case "darwin": diff --git a/website/docs/guides/file-association.mdx b/website/docs/guides/file-association.mdx index ff3ca6e6b53..71bbff37ead 100644 --- a/website/docs/guides/file-association.mdx +++ b/website/docs/guides/file-association.mdx @@ -85,6 +85,30 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. @@ -179,6 +203,26 @@ func main() { } ``` -## Limitations: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/docs/guides/single-instance-lock.mdx b/website/docs/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..644e1e0d80c --- /dev/null +++ b/website/docs/guides/single-instance-lock.mdx @@ -0,0 +1,80 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 10f80d73749..96c1145e98a 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -16,6 +16,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Added +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) From afe922a4e3303a6863a5692834c2ee74ded8e3e5 Mon Sep 17 00:00:00 2001 From: stffabi Date: Tue, 24 Oct 2023 08:59:39 +0200 Subject: [PATCH 27/87] [go-webview2] Bump to 1.0.10 (#3002) --- v2/go.mod | 2 +- v2/go.sum | 4 ++-- website/src/pages/changelog.mdx | 2 ++ 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/v2/go.mod b/v2/go.mod index b707a5d5512..4cb6c48ae4e 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -36,7 +36,7 @@ require ( github.com/tc-hib/winres v0.1.5 github.com/tidwall/sjson v1.1.7 github.com/tkrajina/go-reflector v0.5.6 - github.com/wailsapp/go-webview2 v1.0.7 + github.com/wailsapp/go-webview2 v1.0.10 github.com/wailsapp/mimetype v1.4.1 github.com/wzshiming/ctc v1.2.3 golang.org/x/mod v0.12.0 diff --git a/v2/go.sum b/v2/go.sum index dd11b8555ee..1d3da8f39c9 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -231,8 +231,8 @@ github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyC github.com/valyala/fasttemplate v1.2.1/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= github.com/valyala/fasttemplate v1.2.2 h1:lxLXG0uE3Qnshl9QyaK6XJxMXlQZELvChBOCmQD0Loo= github.com/valyala/fasttemplate v1.2.2/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= -github.com/wailsapp/go-webview2 v1.0.7 h1:s95+7irJsAsTy1RsjJ6N0cYX7tZ4gP7Uzawds0L2urs= -github.com/wailsapp/go-webview2 v1.0.7/go.mod h1:Uk2BePfCRzttBBjFrBmqKGJd41P6QIHeV9kTgIeOZNo= +github.com/wailsapp/go-webview2 v1.0.10 h1:PP5Hug6pnQEAhfRzLCoOh2jJaPdrqeRgJKZhyYyDV/w= +github.com/wailsapp/go-webview2 v1.0.10/go.mod h1:Uk2BePfCRzttBBjFrBmqKGJd41P6QIHeV9kTgIeOZNo= github.com/wailsapp/mimetype v1.4.1 h1:pQN9ycO7uo4vsUUuPeHEYoUkLVkaRntMnHJxVwYhwHs= github.com/wailsapp/mimetype v1.4.1/go.mod h1:9aV5k31bBOv5z6u+QP8TltzvNGJPmNJD4XlAL3U+j3o= github.com/wzshiming/ctc v1.2.3 h1:q+hW3IQNsjIlOFBTGZZZeIXTElFM4grF4spW/errh/c= diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 96c1145e98a..732b39bc521 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -34,10 +34,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 From 3c39b674d3f7b3002acf080ee7e9f61756f312df Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sun, 29 Oct 2023 18:53:59 +1100 Subject: [PATCH 28/87] chore: update sponsors.svg (#3005) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 29b0c226336..1003e5419e1 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -17,8 +17,12 @@ text { } Bronze Sponsors - Cody Bentley - + Cody Bentley + + + + Kazuya Gokita + Covering Costs Nick From 2693de6822d38599027aacd9c14259a0195b9714 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=A4=9A=E5=95=A6b=E6=A2=A6?= <1715109585@qq.com> Date: Sun, 29 Oct 2023 16:16:33 +0800 Subject: [PATCH 29/87] I developed a window designer software that I hope can be added to the template window designer free open source designed specifically for wails (#3006) --- .../current/community/templates.mdx | 1 + 1 file changed, 1 insertion(+) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx index fb487e87c79..4cd7af805de 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -28,6 +28,7 @@ sidebar_position: 1 - [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - 使用 JavaScript + Quasar V2(Vue 3, Vite, Sass, Pinia, ESLint, Prettier)的模板 - [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - 使用 TypeScript + Quasar V2(Vue 3、Vite、Sass、Pinia、ESLint、Prettier、带 <script setup> 的Composition API)的模板 - [wails-template-naive](https://github.com/tk103331/wails-template-naive) - 基于 Naive UI(一款 Vue 3 组件库)的 Wails 模板 +- [wails-template-vue-go-easy](https://github.com/duolabmeng6/wails-template-vue-go-easy) - 基于 GoEasyDesigner(一款用于 vue 可视化窗口设计软件)的 Wails 模板 ## Angular From ac560a6526283feb31aa30aff23467820670078b Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Mon, 30 Oct 2023 19:49:25 +1100 Subject: [PATCH 30/87] chore: update sponsors.svg (#3010) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 26 +++++++++++++++----------- 1 file changed, 15 insertions(+), 11 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 1003e5419e1..a5ab64e0ade 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -129,46 +129,50 @@ text { Helpers - + - + - + - + - + - + - + - + - + - + - + + + + + From b328f6017259f20e0d1fb0c809fcfcacc1922512 Mon Sep 17 00:00:00 2001 From: biuaxia Date: Mon, 30 Oct 2023 20:53:21 +0800 Subject: [PATCH 31/87] Update changelog.mdx (#3014) --- website/src/pages/changelog.mdx | 1 + 1 file changed, 1 insertion(+) diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 732b39bc521..7cfa072aad7 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -16,6 +16,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Added +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) - Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) From 0533d2e2088caf079a5a7be633a27581befc9460 Mon Sep 17 00:00:00 2001 From: biuaxia Date: Mon, 30 Oct 2023 20:54:03 +0800 Subject: [PATCH 32/87] Update options.mdx (#3013) Change the ZoomFactor and IsZoomControlEnabled in the document to the location of the new version --- .../version-v2.6.0/reference/options.mdx | 36 +++++++++---------- 1 file changed, 18 insertions(+), 18 deletions(-) diff --git a/website/versioned_docs/version-v2.6.0/reference/options.mdx b/website/versioned_docs/version-v2.6.0/reference/options.mdx index d24da9810f6..abeb8a8bba3 100644 --- a/website/versioned_docs/version-v2.6.0/reference/options.mdx +++ b/website/versioned_docs/version-v2.6.0/reference/options.mdx @@ -53,8 +53,6 @@ func main() { CSSDragValue: "drag", EnableDefaultContextMenu: false, EnableFraudulentWebsiteDetection: false, - ZoomFactor: 1.0, - IsZoomControlEnabled: false, Bind: []interface{}{ app, }, @@ -82,7 +80,9 @@ func main() { OnSuspend func() // OnResume is called when Windows resumes from low power mode OnResume func(), - WebviewGpuDisabled: false, + WebviewGpuDisabled: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, }, Mac: &mac.Options{ TitleBar: &mac.TitleBar{ @@ -457,21 +457,6 @@ services of Apple and Microsoft. Name: EnableFraudulentWebsiteDetection
Type: `bool` -### ZoomFactor - -Name: ZoomFactor
-Type: `float64` - -This defines the zoom factor for the WebView2. This is the option matching the Edge user activated zoom in or out. - -### IsZoomControlEnabled - -Name: IsZoomControlEnabled
-Type: `bool` - -This enables the zoom factor to be changed by the user. Please note that the zoom factor can be set in the options while -disallowing the user to change it at runtime (f.e. for a kiosk application or similar). - ### Bind A slice of struct instances defining methods that need to be bound to the frontend. @@ -687,6 +672,21 @@ Setting this to `true` will disable GPU hardware acceleration for the webview. Name: WebviewGpuIsDisabled
Type: `bool` +#### ZoomFactor + +Name: ZoomFactor
+Type: `float64` + +This defines the zoom factor for the WebView2. This is the option matching the Edge user activated zoom in or out. + +#### IsZoomControlEnabled + +Name: IsZoomControlEnabled
+Type: `bool` + +This enables the zoom factor to be changed by the user. Please note that the zoom factor can be set in the options while +disallowing the user to change it at runtime (f.e. for a kiosk application or similar). + ### Mac This defines [Mac specific options](#mac). From d96ec5d332bdc27969e1e968c9a8c6090da3a80a Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Tue, 31 Oct 2023 20:55:47 +1100 Subject: [PATCH 33/87] chore: update sponsors.svg (#3016) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 28 ++++++++++++++++------------ 1 file changed, 16 insertions(+), 12 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index a5ab64e0ade..70df0bd98f5 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -129,50 +129,54 @@ text { Helpers - + - + - + - + - + - + - + - + - + - + - + - + + + + + From 93335d766106e0173c42724f9a7becfdbfc0d08e Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Wed, 1 Nov 2023 08:05:17 +1100 Subject: [PATCH 34/87] update docs --- website/src/pages/credits.mdx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/website/src/pages/credits.mdx b/website/src/pages/credits.mdx index 09ca8aecb4a..695900a0454 100644 --- a/website/src/pages/credits.mdx +++ b/website/src/pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - Project owner, lead developer - [Stffabi](https://github.com/stffabi) - Technical lead, developer and maintainer -- [Misite Bao](https://github.com/misitebao) - Documentation wizard, Chinese translation, Windows testing, Bug finder general - [Travis McLane](https://github.com/tmclane) - Cross-compilation work, MacOS testing - [Lyimmi](https://github.com/Lyimmi) - All things Linux +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## Sponsors From e960798e857602dd600e811d6a7e16b14804837d Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 1 Nov 2023 20:02:59 +1100 Subject: [PATCH 35/87] chore: update sponsors.svg (#3019) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 70df0bd98f5..cbf1cced2fe 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -17,12 +17,16 @@ text { } Bronze Sponsors - Cody Bentley - + Cody Bentley + - Kazuya Gokita - + Kazuya Gokita + + + + Simon Thomas + Covering Costs Nick From ae688aa07d279fb2222123e9623a7d04d3f5436e Mon Sep 17 00:00:00 2001 From: Andrey Pshenkin Date: Thu, 2 Nov 2023 09:26:05 +0000 Subject: [PATCH 36/87] Implement custom protocol association support (#3000) * implement MacOS openFile/openFiles events * wip: windows file association * fix macro import * add file icon copy * try copy icon * keep only required part of scripts * update config schema * fix json * set fileAssociation for mac via config * proper iconName handling * add fileAssociation icon generator * fix file association icons bundle * don't break compatibility * remove mimeType as not supported linux for now * add documentation * adjust config schema * restore formatting * try implement single instance lock with params passing * fix focusing * fix focusing * formatting * use channel buffer for second instance events * handle errors * add comment * remove unused option in file association * wip: linux single instance lock * wip: linux single instance * some experiments with making window active * try to use unminimise * remove unused * try present for window * try present for window * fix build * cleanup * cleanup * implement single instance lock on mac os * implement proper show for windows * proper unmimimise * get rid of openFiles mac os. change configuration structure * remove unused channel * remove unused function * add documentation for single instance lock * add PR link * wip mac os deeplinks * put custom url listner on top to catch link on app opening * put custom url listner on top to catch link on app opening * try add custom url windows * adjust custom url * add docs * merge master * update documentation * add comment for darwin * add PR link * change naming * change naming * change naming * change naming * fix formatting * fix naming * Fix typo --------- Co-authored-by: Lea Anthony --- .../frontend/desktop/darwin/CustomProtocol.h | 14 ++ .../frontend/desktop/darwin/CustomProtocol.m | 20 ++ .../frontend/desktop/darwin/frontend.go | 24 +++ v2/internal/project/project.go | 7 + .../buildassets/build/darwin/Info.dev.plist | 17 ++ v2/pkg/buildassets/build/darwin/Info.plist | 17 ++ .../build/windows/installer/project.nsi | 2 + .../build/windows/installer/wails_tools.nsh | 29 +++ v2/pkg/options/mac/mac.go | 1 + .../docs/guides/custom-protocol-schemes.mdx | 189 ++++++++++++++++++ website/src/pages/changelog.mdx | 1 + website/static/schemas/config.v2.json | 25 +++ 12 files changed, 346 insertions(+) create mode 100644 v2/internal/frontend/desktop/darwin/CustomProtocol.h create mode 100644 v2/internal/frontend/desktop/darwin/CustomProtocol.m create mode 100644 website/docs/guides/custom-protocol-schemes.mdx diff --git a/v2/internal/frontend/desktop/darwin/CustomProtocol.h b/v2/internal/frontend/desktop/darwin/CustomProtocol.h new file mode 100644 index 00000000000..da0e7079f1d --- /dev/null +++ b/v2/internal/frontend/desktop/darwin/CustomProtocol.h @@ -0,0 +1,14 @@ +#ifndef CustomProtocol_h +#define CustomProtocol_h + +#import + +extern void HandleCustomProtocol(char*); + +@interface CustomProtocolSchemeHandler : NSObject ++ (void)handleGetURLEvent:(NSAppleEventDescriptor *)event withReplyEvent:(NSAppleEventDescriptor *)replyEvent; +@end + +void StartCustomProtocolHandler(void); + +#endif /* CustomProtocol_h */ diff --git a/v2/internal/frontend/desktop/darwin/CustomProtocol.m b/v2/internal/frontend/desktop/darwin/CustomProtocol.m new file mode 100644 index 00000000000..7365e4f5042 --- /dev/null +++ b/v2/internal/frontend/desktop/darwin/CustomProtocol.m @@ -0,0 +1,20 @@ +#include "CustomProtocol.h" + +@implementation CustomProtocolSchemeHandler ++ (void)handleGetURLEvent:(NSAppleEventDescriptor *)event withReplyEvent:(NSAppleEventDescriptor *)replyEvent { + [event paramDescriptorForKeyword:keyDirectObject]; + + NSString *urlStr = [[event paramDescriptorForKeyword:keyDirectObject] stringValue]; + + HandleCustomProtocol((char*)[[[event paramDescriptorForKeyword:keyDirectObject] stringValue] UTF8String]); +} +@end + +void StartCustomProtocolHandler(void) { + NSAppleEventManager *appleEventManager = [NSAppleEventManager sharedAppleEventManager]; + + [appleEventManager setEventHandler:[CustomProtocolSchemeHandler class] + andSelector:@selector(handleGetURLEvent:withReplyEvent:) + forEventClass:kInternetEventClass + andEventID: kAEGetURL]; +} diff --git a/v2/internal/frontend/desktop/darwin/frontend.go b/v2/internal/frontend/desktop/darwin/frontend.go index 7a4905eeb1d..c792ae9c88e 100644 --- a/v2/internal/frontend/desktop/darwin/frontend.go +++ b/v2/internal/frontend/desktop/darwin/frontend.go @@ -8,6 +8,7 @@ package darwin #cgo LDFLAGS: -framework Foundation -framework Cocoa -framework WebKit #import #import "Application.h" +#import "CustomProtocol.h" #import "WailsContext.h" #include @@ -39,6 +40,7 @@ var messageBuffer = make(chan string, 100) var requestBuffer = make(chan webview.Request, 100) var callbackBuffer = make(chan uint, 10) var openFilepathBuffer = make(chan string, 100) +var openUrlBuffer = make(chan string, 100) var secondInstanceBuffer = make(chan options.SecondInstanceData, 1) type Frontend struct { @@ -79,6 +81,9 @@ func NewFrontend(ctx context.Context, appoptions *options.App, myLogger *logger. } result.startURL, _ = url.Parse(startURL) + // this should be initialized as early as possible to handle first instance launch + C.StartCustomProtocolHandler() + if _starturl, _ := ctx.Value("starturl").(*url.URL); _starturl != nil { result.startURL = _starturl } else { @@ -110,6 +115,7 @@ func NewFrontend(ctx context.Context, appoptions *options.App, myLogger *logger. go result.startMessageProcessor() go result.startCallbackProcessor() go result.startFileOpenProcessor() + go result.startUrlOpenProcessor() go result.startSecondInstanceProcessor() return result @@ -121,6 +127,12 @@ func (f *Frontend) startFileOpenProcessor() { } } +func (f *Frontend) startUrlOpenProcessor() { + for url := range openUrlBuffer { + f.ProcessOpenUrlEvent(url) + } +} + func (f *Frontend) startSecondInstanceProcessor() { for secondInstanceData := range secondInstanceBuffer { if f.frontendOptions.SingleInstanceLock != nil && @@ -385,6 +397,12 @@ func (f *Frontend) ProcessOpenFileEvent(filePath string) { } } +func (f *Frontend) ProcessOpenUrlEvent(url string) { + if f.frontendOptions.Mac != nil && f.frontendOptions.Mac.OnUrlOpen != nil { + f.frontendOptions.Mac.OnUrlOpen(url) + } +} + func (f *Frontend) Callback(message string) { escaped, err := json.Marshal(message) if err != nil { @@ -434,3 +452,9 @@ func HandleOpenFile(filePath *C.char) { goFilepath := C.GoString(filePath) openFilepathBuffer <- goFilepath } + +//export HandleCustomProtocol +func HandleCustomProtocol(url *C.char) { + goUrl := C.GoString(url) + openUrlBuffer <- goUrl +} diff --git a/v2/internal/project/project.go b/v2/internal/project/project.go index a0bf518a9c9..d42977fee51 100644 --- a/v2/internal/project/project.go +++ b/v2/internal/project/project.go @@ -222,6 +222,7 @@ type Info struct { Copyright *string `json:"copyright"` Comments *string `json:"comments"` FileAssociations []FileAssociation `json:"fileAssociations"` + Protocols []Protocol `json:"protocols"` } type FileAssociation struct { @@ -232,6 +233,12 @@ type FileAssociation struct { Role string `json:"role"` } +type Protocol struct { + Scheme string `json:"scheme"` + Description string `json:"description"` + Role string `json:"role"` +} + type Bindings struct { TsGeneration TsGeneration `json:"ts_generation"` } diff --git a/v2/pkg/buildassets/build/darwin/Info.dev.plist b/v2/pkg/buildassets/build/darwin/Info.dev.plist index 9d4096f4fb2..04727c23f95 100644 --- a/v2/pkg/buildassets/build/darwin/Info.dev.plist +++ b/v2/pkg/buildassets/build/darwin/Info.dev.plist @@ -42,6 +42,23 @@ {{end}} {{end}} + {{if .Info.Protocols}} + CFBundleURLTypes + + {{range .Info.Protocols}} + + CFBundleURLName + com.wails.{{.Scheme}} + CFBundleURLSchemes + + {{.Scheme}} + + CFBundleTypeRole + {{.Role}} + + {{end}} + + {{end}} NSAppTransportSecurity NSAllowsLocalNetworking diff --git a/v2/pkg/buildassets/build/darwin/Info.plist b/v2/pkg/buildassets/build/darwin/Info.plist index 079bdafa962..19cc9370cb8 100644 --- a/v2/pkg/buildassets/build/darwin/Info.plist +++ b/v2/pkg/buildassets/build/darwin/Info.plist @@ -42,5 +42,22 @@ {{end}} {{end}} + {{if .Info.Protocols}} + CFBundleURLTypes + + {{range .Info.Protocols}} + + CFBundleURLName + com.wails.{{.Scheme}} + CFBundleURLSchemes + + {{.Scheme}} + + CFBundleTypeRole + {{.Role}} + + {{end}} + + {{end}} diff --git a/v2/pkg/buildassets/build/windows/installer/project.nsi b/v2/pkg/buildassets/build/windows/installer/project.nsi index f18f2b5dfb9..654ae2e4957 100644 --- a/v2/pkg/buildassets/build/windows/installer/project.nsi +++ b/v2/pkg/buildassets/build/windows/installer/project.nsi @@ -92,6 +92,7 @@ Section CreateShortCut "$DESKTOP\${INFO_PRODUCTNAME}.lnk" "$INSTDIR\${PRODUCT_EXECUTABLE}" !insertmacro wails.associateFiles + !insertmacro wails.associateCustomProtocols !insertmacro wails.writeUninstaller SectionEnd @@ -107,6 +108,7 @@ Section "uninstall" Delete "$DESKTOP\${INFO_PRODUCTNAME}.lnk" !insertmacro wails.unassociateFiles + !insertmacro wails.unassociateCustomProtocols !insertmacro wails.deleteUninstaller SectionEnd diff --git a/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh b/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh index bc79fb395ac..f9c0f885225 100644 --- a/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh +++ b/v2/pkg/buildassets/build/windows/installer/wails_tools.nsh @@ -218,3 +218,32 @@ RequestExecutionLevel "${REQUEST_EXECUTION_LEVEL}" Delete "$INSTDIR\{{.IconName}}.ico" {{end}} !macroend + +!macro CUSTOM_PROTOCOL_ASSOCIATE PROTOCOL DESCRIPTION ICON COMMAND + DeleteRegKey SHELL_CONTEXT "Software\Classes\${PROTOCOL}" + WriteRegStr SHELL_CONTEXT "Software\Classes\${PROTOCOL}" "" "${DESCRIPTION}" + WriteRegStr SHELL_CONTEXT "Software\Classes\${PROTOCOL}" "URL Protocol" "" + WriteRegStr SHELL_CONTEXT "Software\Classes\${PROTOCOL}\DefaultIcon" "" "${ICON}" + WriteRegStr SHELL_CONTEXT "Software\Classes\${PROTOCOL}\shell" "" "" + WriteRegStr SHELL_CONTEXT "Software\Classes\${PROTOCOL}\shell\open" "" "" + WriteRegStr SHELL_CONTEXT "Software\Classes\${PROTOCOL}\shell\open\command" "" "${COMMAND}" +!macroend + +!macro CUSTOM_PROTOCOL_UNASSOCIATE PROTOCOL + DeleteRegKey SHELL_CONTEXT "Software\Classes\${PROTOCOL}" +!macroend + +!macro wails.associateCustomProtocols + ; Create custom protocols associations + {{range .Info.Protocols}} + !insertmacro CUSTOM_PROTOCOL_ASSOCIATE "{{.Scheme}}" "{{.Description}}" "$INSTDIR\${PRODUCT_EXECUTABLE},0" "$INSTDIR\${PRODUCT_EXECUTABLE} $\"%1$\"" + + {{end}} +!macroend + +!macro wails.unassociateCustomProtocols + ; Delete app custom protocol associations + {{range .Info.Protocols}} + !insertmacro CUSTOM_PROTOCOL_UNASSOCIATE "{{.Scheme}}" + {{end}} +!macroend diff --git a/v2/pkg/options/mac/mac.go b/v2/pkg/options/mac/mac.go index ecd2923fbbc..5758befd011 100644 --- a/v2/pkg/options/mac/mac.go +++ b/v2/pkg/options/mac/mac.go @@ -24,5 +24,6 @@ type Options struct { //ActivationPolicy ActivationPolicy About *AboutInfo OnFileOpen func(filePath string) `json:"-"` + OnUrlOpen func(filePath string) `json:"-"` //URLHandlers map[string]func(string) } diff --git a/website/docs/guides/custom-protocol-schemes.mdx b/website/docs/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..c56634f0eb0 --- /dev/null +++ b/website/docs/guides/custom-protocol-schemes.mdx @@ -0,0 +1,189 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + + +## Set Up Custom Protocol Schemes Association: +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +|:------------|:--------------------------------------------------------------------------------------| +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + + +### Windows +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` +3. Configure nfpm to use your scripts and files. Example: +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` +6. Build your .deb package using nfpm: +```sh +nfpm pkg --packager deb --target . +``` +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, +new instance of app is launched and file path is passed as argument to your app. +To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 7cfa072aad7..8b65b549057 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -28,6 +28,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c - The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Changed diff --git a/website/static/schemas/config.v2.json b/website/static/schemas/config.v2.json index 7ab40d95fbd..f215415e641 100644 --- a/website/static/schemas/config.v2.json +++ b/website/static/schemas/config.v2.json @@ -200,6 +200,31 @@ } } } + }, + "protocols": { + "type": "array", + "description": "Custom URI protocols that should be opened by the application", + "items": { + "type": "object", + "properties": { + "scheme": { + "type": "string", + "description": "protocol scheme. e.g. myapp" + }, + "description": { + "type": "string", + "description": "Windows-only. The description. It is displayed on the `Type` column on Windows Explorer." + }, + "role": { + "description": "macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole.", + "allOf": [ + { + "$ref": "#/definitions/BundleTypeRole" + } + ] + } + } + } } } }, From 0a63215cde292c38fa1e08fa7ed23a570f373b05 Mon Sep 17 00:00:00 2001 From: Bruno Borges Date: Fri, 3 Nov 2023 12:55:03 -0700 Subject: [PATCH 37/87] Resource efficient will often mean CPU/memory. Here, it saves download size. (#3028) --- website/docs/introduction.mdx | 2 +- .../ko/docusaurus-plugin-content-docs/current/introduction.mdx | 2 +- .../vi/docusaurus-plugin-content-docs/current/introduction.mdx | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/website/docs/introduction.mdx b/website/docs/introduction.mdx index 1bfa099e80f..98f42806fa9 100644 --- a/website/docs/introduction.mdx +++ b/website/docs/introduction.mdx @@ -48,7 +48,7 @@ and TypeScript versions for each template. Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. -**It does not embed a browser**, so it is resource efficient. Instead, it uses the native rendering engine for the +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. ### Go & JavaScript Interoperability diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/introduction.mdx index 222a0ae9846..9fd585e11f8 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Wails comes with a number of pre-configured templates that allow you to get your Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. -**It does not embed a browser**, so it is resource efficient. Instead, it uses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. ### Go & JavaScript Interoperability diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx index a8bb301fb23..98d1dd2f807 100644 --- a/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Wails comes with a number of pre-configured templates that allow you to get your Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. -**It does not embed a browser**, so it is resource efficient. Instead, it uses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. ### Go & JavaScript Interoperability From 08e12de2a00a04b79cc98c586337c66ff359ae79 Mon Sep 17 00:00:00 2001 From: Denis Bernard Date: Sat, 4 Nov 2023 08:03:12 +0100 Subject: [PATCH 38/87] linux: workaround for #2977 (#3027) In NewWindow, set options.Linux.WebviewGpuPolicy to WebviewGpuPolicyNever if options.Linux is nil, disabling GPU acceleration by default on linux until the upstream bugs https://bugs.webkit.org/show_bug.cgi?id=228268 and https://bugs.webkit.org/show_bug.cgi?id=261874 are fixed. --- v2/internal/frontend/desktop/linux/window.go | 4 ++++ v2/pkg/options/linux/linux.go | 5 +++++ 2 files changed, 9 insertions(+) diff --git a/v2/internal/frontend/desktop/linux/window.go b/v2/internal/frontend/desktop/linux/window.go index 82030f439cd..fbe8d7b5c76 100644 --- a/v2/internal/frontend/desktop/linux/window.go +++ b/v2/internal/frontend/desktop/linux/window.go @@ -25,6 +25,7 @@ import ( "github.com/wailsapp/wails/v2/internal/frontend" "github.com/wailsapp/wails/v2/pkg/menu" "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/linux" ) func gtkBool(input bool) C.gboolean { @@ -90,6 +91,9 @@ func NewWindow(appoptions *options.App, debug bool, devtoolsEnabled bool) *Windo var webviewGpuPolicy int if appoptions.Linux != nil { webviewGpuPolicy = int(appoptions.Linux.WebviewGpuPolicy) + } else { + // workaround for https://github.com/wailsapp/wails/issues/2977 + webviewGpuPolicy = int(linux.WebviewGpuPolicyNever) } webview := C.SetupWebview( diff --git a/v2/pkg/options/linux/linux.go b/v2/pkg/options/linux/linux.go index 062119cb714..797450c2725 100644 --- a/v2/pkg/options/linux/linux.go +++ b/v2/pkg/options/linux/linux.go @@ -28,6 +28,11 @@ type Options struct { // - WebviewGpuPolicyAlways // - WebviewGpuPolicyOnDemand // - WebviewGpuPolicyNever + // + // Due to https://github.com/wailsapp/wails/issues/2977, if options.Linux is nil + // in the call to wails.Run(), WebviewGpuPolicy is set by default to WebviewGpuPolicyNever. + // Client code may override this behavior by passing a non-nil Options and set + // WebviewGpuPolicy as needed. WebviewGpuPolicy WebviewGpuPolicy // ProgramName is used to set the program's name for the window manager via GTK's g_set_prgname(). From 426a569c8995f032757b1a587eacf54f831e7f72 Mon Sep 17 00:00:00 2001 From: Andrey Pshenkin Date: Sun, 5 Nov 2023 03:10:01 +0000 Subject: [PATCH 39/87] Fix single instance lock for macOS sandbox app (#3029) * fix single instance lock for sandbox app * fix single instance lock for sandbox app --- .../frontend/desktop/darwin/AppDelegate.h | 2 ++ .../frontend/desktop/darwin/AppDelegate.m | 17 +++++++++++++---- .../desktop/darwin/single_instance.go | 19 ++++++++++++++++++- 3 files changed, 33 insertions(+), 5 deletions(-) diff --git a/v2/internal/frontend/desktop/darwin/AppDelegate.h b/v2/internal/frontend/desktop/darwin/AppDelegate.h index 4fa32233e8b..a8d10f647d9 100644 --- a/v2/internal/frontend/desktop/darwin/AppDelegate.h +++ b/v2/internal/frontend/desktop/darwin/AppDelegate.h @@ -28,4 +28,6 @@ extern void HandleSecondInstanceData(char * message); void SendDataToFirstInstance(char * singleInstanceUniqueId, char * text); +char* GetMacOsNativeTempDir(); + #endif /* AppDelegate_h */ diff --git a/v2/internal/frontend/desktop/darwin/AppDelegate.m b/v2/internal/frontend/desktop/darwin/AppDelegate.m index 55bc2b4a817..18ecfbcff63 100644 --- a/v2/internal/frontend/desktop/darwin/AppDelegate.m +++ b/v2/internal/frontend/desktop/darwin/AppDelegate.m @@ -48,17 +48,26 @@ - (void)applicationDidFinishLaunching:(NSNotification *)aNotification { } void SendDataToFirstInstance(char * singleInstanceUniqueId, char * message) { + // we pass message in object because otherwise sandboxing will prevent us from sending it https://developer.apple.com/forums/thread/129437 + NSString * myString = [NSString stringWithUTF8String:message]; [[NSDistributedNotificationCenter defaultCenter] postNotificationName:[NSString stringWithUTF8String:singleInstanceUniqueId] - object:nil - userInfo:@{@"message": [NSString stringWithUTF8String:message]} + object:(__bridge const void *)(myString) + userInfo:nil deliverImmediately:YES]; } +char* GetMacOsNativeTempDir() { + NSString *tempDir = NSTemporaryDirectory(); + char *copy = strdup([tempDir UTF8String]); + + return copy; +} + - (void)handleSecondInstanceNotification:(NSNotification *)note; { - if (note.userInfo[@"message"] != nil) { - NSString *message = note.userInfo[@"message"]; + if (note.object != nil) { + NSString * message = (__bridge NSString *)note.object; const char* utf8Message = message.UTF8String; HandleSecondInstanceData((char*)utf8Message); } diff --git a/v2/internal/frontend/desktop/darwin/single_instance.go b/v2/internal/frontend/desktop/darwin/single_instance.go index 6f61ed1734f..012baef0280 100644 --- a/v2/internal/frontend/desktop/darwin/single_instance.go +++ b/v2/internal/frontend/desktop/darwin/single_instance.go @@ -12,13 +12,16 @@ package darwin import "C" import ( "encoding/json" + "fmt" "github.com/wailsapp/wails/v2/pkg/options" "os" + "strings" "syscall" + "unsafe" ) func SetupSingleInstance(uniqueID string) { - lockFilePath := os.TempDir() + lockFilePath := getTempDir() lockFileName := uniqueID + ".lock" _, err := createLockFile(lockFilePath + "/" + lockFileName) @@ -62,14 +65,28 @@ func HandleSecondInstanceData(secondInstanceMessage *C.char) { func createLockFile(filename string) (*os.File, error) { file, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0600) if err != nil { + fmt.Printf("Failed to open lockfile %s: %s", filename, err) return nil, err } err = syscall.Flock(int(file.Fd()), syscall.LOCK_EX|syscall.LOCK_NB) if err != nil { + // Flock failed for some other reason than other instance already lock it. Print it in logs for possible debugging. + if !strings.Contains(err.Error(), "resource temporarily unavailable") { + fmt.Printf("Failed to lock lockfile %s: %s", filename, err) + } file.Close() return nil, err } return file, nil } + +// If app is sandboxed, golang os.TempDir() will return path that will not be accessible. So use native macOS temp dir function. +func getTempDir() string { + cstring := C.GetMacOsNativeTempDir() + path := C.GoString(cstring) + C.free(unsafe.Pointer(cstring)) + + return path +} From 57c04ba740aecce7eb276eec87741aa281046226 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Mon, 6 Nov 2023 20:46:14 +1100 Subject: [PATCH 40/87] chore: update sponsors.svg (#3034) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 18 +++++++----------- 1 file changed, 7 insertions(+), 11 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index cbf1cced2fe..728c3665029 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -113,24 +113,20 @@ text { - - - - - - - - + - + + + + - + - + Helpers From e32c2b05e47013aa46c484738f98c9d6e9653684 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 8 Nov 2023 20:00:03 +1100 Subject: [PATCH 41/87] chore: update sponsors.svg (#3038) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 32 ++++++++++++++------------------ 1 file changed, 14 insertions(+), 18 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 728c3665029..c349d8013ac 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -47,36 +47,32 @@ text { Iain Buying Breakfast - tc-hib - + tc-hib + - Tai Groot - + Tai Groot + - Michael - + Michael + - Tom Wu - + Tom Wu + - Bironou - + Bironou + - Arden - + Arden + - igops - - - - Prashanth - + igops + Buying Coffee From 03545e33d98a16b87132af743d43bbac58f02579 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 8 Nov 2023 21:06:11 +1100 Subject: [PATCH 42/87] docs: sync translations (#3001) Co-authored-by: leaanthony Co-authored-by: Lea Anthony --- .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../current/introduction.mdx | 2 +- .../changelog.mdx | 6 + .../credits.mdx | 2 +- .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../current/introduction.mdx | 2 +- .../changelog.mdx | 6 + .../credits.mdx | 2 +- .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../changelog.mdx | 6 + .../credits.mdx | 2 +- .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../current/introduction.mdx | 2 +- .../changelog.mdx | 6 + .../credits.mdx | 2 +- .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../current/introduction.mdx | 2 +- .../changelog.mdx | 6 + .../credits.mdx | 2 +- .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../changelog.mdx | 6 + .../credits.mdx | 2 +- .../current/community/templates.mdx | 1 - .../guides/custom-protocol-schemes.mdx | 204 ++++++++++++++++++ .../current/guides/file-association.mdx | 51 ++++- .../current/guides/single-instance-lock.mdx | 81 +++++++ .../current/introduction.mdx | 2 +- .../changelog.mdx | 6 + .../credits.mdx | 2 +- 41 files changed, 2385 insertions(+), 34 deletions(-) create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/introduction.mdx index 724c24436b4..680c0bba22a 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Wails est livré avec un certain nombre de modèles préconfigurés qui vous per Wails utilise une bibliothèque conçue pour gérer les éléments natifs tels que les fenêtres, menus, boîtes de dialogues, etc, pour que vous puissiez construire des applications de bureau riches en fonctionnalités. -**Il n'embarque pas de navigateur**, il est donc efficace sur les ressources. Au lieu de cela, il utilise le moteur de rendu natif de la plate-forme. Sous Windows, c'est la nouvelle bibliothèque Microsoft Webview2, construite sur Chromium. +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. Sous Windows, c'est la nouvelle bibliothèque Microsoft Webview2, construite sur Chromium. ### Interopérabilité Go & Javascript diff --git a/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx index b1b51b2277c..1e38ce80727 100644 --- a/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ Le format est basé sur [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) ### Ajouts +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ Le format est basé sur [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Changements @@ -30,10 +34,12 @@ Le format est basé sur [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Corrections - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/fr/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/fr/docusaurus-plugin-content-pages/credits.mdx index 1fa097ee349..9c9adddf01d 100644 --- a/website/i18n/fr/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - Propriétaire du projet, développeur principal - [Stffabi](https://github.com/stffabi) - Chef technique, développeur et mainteneur -- [Misite Bao](https://github.com/misitebao) - Sorcier de la documentation, traducteur chinois, testeur pour Windows, découvreur principal de bugs - [Travis McLane](https://github.com/tmclane) - Travaux de compilation sur plusieurs plateformes, testeur MacOS - [Lyimmi](https://github.com/Lyimmi) - Tout ce qui est Linux +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## Partenaires diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/introduction.mdx index 338c58e6775..174ffb4f88f 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Wailsには、アプリの開発をすばやく始められるように、多数 Wailsは、ウィンドウ、メニュー、ダイアログなどのネイティブ要素を処理する専用ライブラリを使用するため、見栄えが良く、リッチな機能を備えたデスクトップアプリを構築できます。 -**ブラウザを埋め込まないため**、無駄なリソースを割きません。 ブラウザを埋め込まない代わりに、OSプラットフォームのネイティブなレンダリングエンジンを使用します。 例えばWindowsの場合、Chromium上でビルトされているMicrosoft Webview2ライブラリを使用します。 +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. 例えばWindowsの場合、Chromium上でビルトされているMicrosoft Webview2ライブラリを使用します。 ### GoとJavascriptのやり取り diff --git a/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx index 5828a733f3a..072e48e709d 100644 --- a/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), ### Added +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Changed @@ -30,10 +34,12 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/ja/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/ja/docusaurus-plugin-content-pages/credits.mdx index db7fdbfb144..5c1c2bb22f7 100644 --- a/website/i18n/ja/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - プロジェクトオーナー、開発リーダー - [Stffabi](https://github.com/stffabi) - テクニカルリーダー、開発担当、保守担当 -- [Misite Bao](https://github.com/misitebao) - ドキュメント専門担当、翻訳担当(中国語)、Windowsテスター、不具合収集担当 - [Travis McLane](https://github.com/tmclane) - クロスコンパイル担当、MacOSテスター - [Lyimmi](https://github.com/Lyimmi) - Linux関連全般 +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## スポンサー diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx index c4f2e844564..3d777c7e54f 100644 --- a/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), ### Added +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Changed @@ -30,10 +34,12 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/ko/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/ko/docusaurus-plugin-content-pages/credits.mdx index af5cd28c530..db6f2cb340a 100644 --- a/website/i18n/ko/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - Project owner, lead developer - [Stffabi](https://github.com/stffabi) - Technical lead, developer and maintainer -- [Misite Bao](https://github.com/misitebao) - Documentation wizard, Chinese translation, Windows testing, Bug finder general - [Travis McLane](https://github.com/tmclane) - Cross-compilation work, MacOS testing - [Lyimmi](https://github.com/Lyimmi) - All things Linux +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## Sponsors diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/introduction.mdx index e03250e3b3a..f25eaba30c2 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Há uma série de modelos pré-configurados que permitem que o aplicativo entre Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. -**Ele não incorpora um navegador**, então é eficiente em recursos. Ao invés disso, ele usa o mecanismo de renderização nativo para a plataforma. No Windows, esta é a nova biblioteca da Microsoft Webview2, construída no Chromium. +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. No Windows, esta é a nova biblioteca da Microsoft Webview2, construída no Chromium. ### Go & JavaScript Interoperability diff --git a/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx index 7f4f545d3fb..7083edba6dc 100644 --- a/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ O formato é baseado em [Manter um Log de Alterações](https://keepachangelog.c ### Adicionado +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ O formato é baseado em [Manter um Log de Alterações](https://keepachangelog.c - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Alterado @@ -30,10 +34,12 @@ O formato é baseado em [Manter um Log de Alterações](https://keepachangelog.c - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Corrigido - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/pt/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/pt/docusaurus-plugin-content-pages/credits.mdx index af5cd28c530..db6f2cb340a 100644 --- a/website/i18n/pt/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - Project owner, lead developer - [Stffabi](https://github.com/stffabi) - Technical lead, developer and maintainer -- [Misite Bao](https://github.com/misitebao) - Documentation wizard, Chinese translation, Windows testing, Bug finder general - [Travis McLane](https://github.com/tmclane) - Cross-compilation work, MacOS testing - [Lyimmi](https://github.com/Lyimmi) - All things Linux +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## Sponsors diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/introduction.mdx index 1e246304506..9afebec4d31 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Wails поставляется с рядом предварительно нас Wails использует специально созданную библиотеку для обработки системных элементов, таких как окна, меню, диалоги и так далее, чтобы вы могли создавать хорошо выглядящие, богатые функционалом приложения. -**В нем нет встроенного браузера**, таким образом, это экономично с точки зрения ресурсов. Вместо этого он использует нативный движок отрисовки для необходимой платформы. На Windows это новая библиотека Microsoft Webview2, основанная на Chromium. +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. На Windows это новая библиотека Microsoft Webview2, основанная на Chromium. ### Go & JavaScript Interoperability diff --git a/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx index 5828a733f3a..072e48e709d 100644 --- a/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), ### Added +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Changed @@ -30,10 +34,12 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/ru/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/ru/docusaurus-plugin-content-pages/credits.mdx index 8292fbf1a64..8a135da4c42 100644 --- a/website/i18n/ru/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - Project owner, lead developer - [Stffabi](https://github.com/stffabi) - Technical lead, developer and maintainer -- [Misite Bao](https://github.com/misitebao) - Мастер документации, перевод на китайский язык, тестирование Windows, общий поиск ошибок - [Travis McLane](https://github.com/tmclane) - кросс-компиляция, тестирование MacOS - [Lyimmi](https://github.com/Lyimmi) - All things Linux +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## Sponsors diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx index 4a332a43083..7e37449752b 100644 --- a/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/vi/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ Tất cả những thay đổi đáng chú ý sẽ được thêm vào tài li ### Added +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ Tất cả những thay đổi đáng chú ý sẽ được thêm vào tài li - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### Changed @@ -30,10 +34,12 @@ Tất cả những thay đổi đáng chú ý sẽ được thêm vào tài li - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx index af5cd28c530..db6f2cb340a 100644 --- a/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/vi/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - Project owner, lead developer - [Stffabi](https://github.com/stffabi) - Technical lead, developer and maintainer -- [Misite Bao](https://github.com/misitebao) - Documentation wizard, Chinese translation, Windows testing, Bug finder general - [Travis McLane](https://github.com/tmclane) - Cross-compilation work, MacOS testing - [Lyimmi](https://github.com/Lyimmi) - All things Linux +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## Sponsors diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx index 4cd7af805de..fb487e87c79 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -28,7 +28,6 @@ sidebar_position: 1 - [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - 使用 JavaScript + Quasar V2(Vue 3, Vite, Sass, Pinia, ESLint, Prettier)的模板 - [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - 使用 TypeScript + Quasar V2(Vue 3、Vite、Sass、Pinia、ESLint、Prettier、带 <script setup> 的Composition API)的模板 - [wails-template-naive](https://github.com/tk103331/wails-template-naive) - 基于 Naive UI(一款 Vue 3 组件库)的 Wails 模板 -- [wails-template-vue-go-easy](https://github.com/duolabmeng6/wails-template-vue-go-easy) - 基于 GoEasyDesigner(一款用于 vue 可视化窗口设计软件)的 Wails 模板 ## Angular diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx index b10c66bb373..167955b12d7 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/file-association.mdx @@ -89,6 +89,31 @@ func main() { } ``` +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + ### Linux Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. @@ -193,7 +218,27 @@ func main() { } ``` -## Limitations: +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: -On Windows and Linux when associated file is opened, new instance of your app is launched. -Currently, Wails doesn't support opening files in already running app. There is plugin for single instance support for v3 in development. +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/introduction.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/introduction.mdx index f54b918d22a..6637fd6641c 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/introduction.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/introduction.mdx @@ -42,7 +42,7 @@ Wails 带有许多预配置的模板,可让您快速启动和运行应用程 Wails 使用专门构建的库来处理窗口、菜单、对话框等原生元素,因此您可以构建美观、功能丰富的桌面应用程序。 -**它不嵌入浏览器**,因此性能高。 相反,它使用平台的原生渲染引擎。 在 Windows 上,是基于 Chromium 构建的新 Microsoft Webview2 库。 +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. 在 Windows 上,是基于 Chromium 构建的新 Microsoft Webview2 库。 ### Go 和 JavaScript 互操作 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx index 4b2142d8c3c..239d2b2a668 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-pages/changelog.mdx @@ -15,6 +15,8 @@ ### 新增 +- Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) +- Added Single Instance Lock support with passing arguments to first instance. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2951) - Added support for enabling/disabling swipe gestures for Windows WebView2. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2878) - When building with `-devtools` flag, CMD/CTRL+SHIFT+F12 can be used to open the devtools. Added by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/2915) – Added file association support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/2918) - Added support for setting some of the Webview preferences, `textInteractionEnabled` and `tabFocusesLinks` on Mac. Added by @fkhadra in [PR](https://github.com/wailsapp/wails/pull/2937) @@ -23,6 +25,8 @@ - New task created for linting v2 `task v2:lint`. Workflow updated to run the task. Added by @mikeee in [PR](https://github.com/wailsapp/wails/pull/2957) - Added new community template wails-htmx-templ-chi-tailwind. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2984) - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c +- The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) +- Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) ### 变更 @@ -30,10 +34,12 @@ - AssetServer requests are now processed concurrently by spawning a goroutine per request. Changed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2926) - Now building with `-devtools` flag doesn't enable the default context-menu. Changed by @mmghv in [PR](https://github.com/wailsapp/wails/pull/2923) - Change Window Level. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/2944) + #### 修复 - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) ## v2.6.0 - 2023-09-06 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-pages/credits.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-pages/credits.mdx index 552e000e04b..68e0e58865a 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-pages/credits.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-pages/credits.mdx @@ -2,9 +2,9 @@ - [Lea Anthony](https://github.com/leaanthony) - 项目所有者,首席开发人员 - [Stffabi](https://github.com/stffabi) - 技术领导、 开发者和维护者 -- [Misite Bao](https://github.com/misitebao) - 文档向导,中文翻译,Windows 测试,常规 Bug 发现 - [Travis McLane](https://github.com/tmclane) - 处理交叉编译相关工作, MacOS 平台的测试 - [Lyimmi](https://github.com/Lyimmi) - Linux 的所有事物 +- [Simon Thomas](mailto:enquiries@wails.io) - Growth Hacker ## 赞助商 From 70b7aa1ec25eeee8788ff7926d632ed88d8b2caa Mon Sep 17 00:00:00 2001 From: Lykin <137850705+tiny-craft@users.noreply.github.com> Date: Thu, 9 Nov 2023 19:08:02 +0800 Subject: [PATCH 43/87] add Tiny RDM to showcase (#3030) --- website/docs/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ .../current/community/showcase/tinyrdm.mdx | 10 ++++++++++ website/static/img/showcase/tiny-rdm.webp | Bin 0 -> 104598 bytes 9 files changed, 80 insertions(+) create mode 100644 website/docs/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx create mode 100644 website/static/img/showcase/tiny-rdm.webp diff --git a/website/docs/community/showcase/tinyrdm.mdx b/website/docs/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/docs/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..8d9fa8a74a1 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +[Tiny RDM](https://redis.tinycraft.cc/) est une interface graphique Redis légère, moderne et open source. Elle dispose d'une belle interface utilisateur, d'une gestion intuitive de la base de données Redis et est compatible avec Windows, Mac et Linux. Elle offre des opérations visuelles de données clé-valeur, prend en charge diverses options de décodage et de visualisation des données, intègre une console pour exécuter des commandes, des requêtes de journal lent, et bien plus encore. \ No newline at end of file diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..87b505b4f96 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +[Tiny RDM](https://redis.tinycraft.cc/)は、オープンソースで、モダンで軽量なRedis GUIです。美しいUI、直感的なRedisデータベースの管理を備え、Windows、Mac、Linuxと互換性があります。ビジュアルなキー-値データ操作を提供し、さまざまなデータのデコードと表示オプションをサポートし、コマンドを実行するための組み込みコンソール、遅いクエリの検索などをサポートしています。 \ No newline at end of file diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..7cfc95177fb --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +[Tiny RDM](https://redis.tinycraft.cc/) 는 오픈 소스이며 현대적이고 가벼운 Redis GUI입니다. 아름다운 UI, 직관적인 Redis 데이터베이스 관리를 제공하며 Windows, Mac 및 Linux과 호환됩니다. 시각적인 키-값 데이터 작업을 지원하며 다양한 데이터 디코딩 및 표시 옵션을 제공하며 명령 실행을 위한 내장 콘솔, 느린 로그 쿼리 등을 지원합니다. \ No newline at end of file diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..8a7f4f6ac4b --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +O [Tiny RDM](https://redis.tinycraft.cc/) é uma GUI Redis moderna e leve de código aberto. Ele possui uma bela interface de usuário, gerenciamento intuitivo de banco de dados Redis e é compatível com Windows, Mac e Linux. Ele oferece operações visuais de dados chave-valor, suporta várias opções de decodificação e visualização de dados, um console embutido para executar comandos, consultas de log lento e muito mais. \ No newline at end of file diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..4e021d3a07d --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +[Tiny RDM](https://redis.tinycraft.cc/) - это современный легкий Redis GUI с открытым исходным кодом. У него красивый пользовательский интерфейс, интуитивное управление базой данных Redis и поддержка Windows, Mac и Linux. Он предоставляет визуальные операции с данными ключ-значение, поддерживает различные варианты декодирования и просмотра данных, а также встроенную консоль для выполнения команд, запросы медленных записей и многое другое. \ No newline at end of file diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..c710e181421 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +[Tiny RDM](https://redis.tinycraft.cc/)是一款开源的现代化轻量级Redis GUI。它拥有美观的操作界面,直观的Redis数据库管理,可在Windows、Mac和Linux上运行。它提供可视化键值数据操作,支持多种数据解码和查看方式,同时内置控制台以执行命令,支持慢查询日志。 diff --git a/website/static/img/showcase/tiny-rdm.webp b/website/static/img/showcase/tiny-rdm.webp new file mode 100644 index 0000000000000000000000000000000000000000..29339520ef782222efcd31895671e2c3d6ff035e GIT binary patch literal 104598 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ℑ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 Pd6) Date: Fri, 10 Nov 2023 10:27:33 +1100 Subject: [PATCH 44/87] Bump golang.org/x/image from 0.5.0 to 0.10.0 in /v2 (#3023) Bumps [golang.org/x/image](https://github.com/golang/image) from 0.5.0 to 0.10.0. - [Commits](https://github.com/golang/image/compare/v0.5.0...v0.10.0) --- updated-dependencies: - dependency-name: golang.org/x/image dependency-type: indirect ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- v2/go.mod | 2 +- v2/go.sum | 12 +++++++----- 2 files changed, 8 insertions(+), 6 deletions(-) diff --git a/v2/go.mod b/v2/go.mod index 4cb6c48ae4e..88bd7c75cda 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -94,7 +94,7 @@ require ( github.com/yuin/goldmark-emoji v1.0.1 // indirect golang.org/x/crypto v0.14.0 // indirect golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 // indirect - golang.org/x/image v0.5.0 // indirect + golang.org/x/image v0.10.0 // indirect golang.org/x/term v0.13.0 // indirect golang.org/x/text v0.13.0 // indirect gopkg.in/warnings.v0 v0.1.2 // indirect diff --git a/v2/go.sum b/v2/go.sum index 1d3da8f39c9..7e4691d0b95 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -12,7 +12,6 @@ github.com/MarvinJWendt/testza v0.2.12/go.mod h1:JOIegYyV7rX+7VZ9r77L/eH6CfJHHzX github.com/MarvinJWendt/testza v0.3.0/go.mod h1:eFcL4I0idjtIx8P9C6KkAuLgATNKpX4/2oUqKc6bF2c= github.com/MarvinJWendt/testza v0.4.2/go.mod h1:mSdhXiKH8sg/gQehJ63bINcCKp7RtYewEjXsvsVUPbE= github.com/MarvinJWendt/testza v0.4.3 h1:u2XaM4IqGp9dsdUmML8/Z791fu4yjQYzOiufOtJwTII= -github.com/MarvinJWendt/testza v0.4.3/go.mod h1:CpXaOfceNEYnLDtNIyTrPPcCpDJYqzZnu2aiA2Wp33U= github.com/Masterminds/semver v1.5.0 h1:H65muMkzWKEuNDnfl9d70GUjFniHKHRbFPGBuZ3QEww= github.com/Masterminds/semver v1.5.0/go.mod h1:MB6lktGJrhw8PrUyiEoblNEGEQ+RzHPF078ddwwvV3Y= github.com/Microsoft/go-winio v0.4.14/go.mod h1:qXqCSQ3Xa7+6tgxaGTIe4Kpcdsi+P8jBhyzoq1bpyYA= @@ -77,7 +76,6 @@ github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMyw github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= -github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4= github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ= github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= @@ -108,7 +106,6 @@ github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa02 github.com/klauspost/cpuid/v2 v2.0.10/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= github.com/klauspost/cpuid/v2 v2.0.12/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= github.com/klauspost/cpuid/v2 v2.1.0 h1:eyi1Ad2aNJMW95zcSbmGg7Cg6cq3ADwLpMAP96d8rF0= -github.com/klauspost/cpuid/v2 v2.1.0/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= @@ -259,9 +256,10 @@ golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 h1:k/i9J1pBpvlfR+9QsetwPyERs golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w= golang.org/x/image v0.0.0-20200430140353-33d19683fad8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= golang.org/x/image v0.0.0-20201208152932-35266b937fa6/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= -golang.org/x/image v0.5.0 h1:5JMiNunQeQw++mMOz48/ISeNu3Iweh/JaZU8ZLqHRrI= -golang.org/x/image v0.5.0/go.mod h1:FVC7BI/5Ym8R25iw5OLsgshdUBbT1h5jZTpA+mvAdZ4= +golang.org/x/image v0.10.0 h1:gXjUUtwtx5yOE0VKWq1CH4IJAClq4UGgUA3i+rpON9M= +golang.org/x/image v0.10.0/go.mod h1:jtrku+n79PfroUbvDdeUWMAI+heR786BofxrbiSF+J0= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/mod v0.12.0 h1:rmsUpXtvNzj340zd98LZ4KntptpfRHwpFOHG188oHXc= golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= @@ -270,6 +268,7 @@ golang.org/x/net v0.0.0-20210326060303-6b1517762897/go.mod h1:uSPa2vr4CLtc/ILN5o golang.org/x/net v0.0.0-20210505024714-0287a6fb4125/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210614182718-04defd469f4e/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -301,6 +300,7 @@ golang.org/x/sys v0.0.0-20220319134239-a9b59b0215f8/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE= golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -308,6 +308,7 @@ golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9sn golang.org/x/term v0.0.0-20210220032956-6a3ed077a48d/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= golang.org/x/term v0.13.0 h1:bb+I9cTfFazGW51MZqBVmZy7+JEJMouUHTUSKVQLBek= golang.org/x/term v0.13.0/go.mod h1:LTmsnFJwVN6bCy1rVCoS+qHT1HhALEFxKncY3WNNh4U= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= @@ -315,6 +316,7 @@ golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= From c76ced7daa9b29fb7a4d2edb164541497ffeed9b Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Fri, 10 Nov 2023 14:23:47 +1100 Subject: [PATCH 45/87] docs: sync translations (#3041) Co-authored-by: leaanthony --- .../current/community/showcase/tinyrdm.mdx | 2 +- .../current/community/templates.mdx | 4 ++-- .../current/guides/application-development.mdx | 8 ++++---- .../current/community/showcase/tinyrdm.mdx | 2 +- .../current/community/showcase/tinyrdm.mdx | 2 +- .../current/community/showcase/tinyrdm.mdx | 2 +- .../current/community/showcase/tinyrdm.mdx | 2 +- .../current/community/showcase/tinyrdm.mdx | 2 +- 8 files changed, 12 insertions(+), 12 deletions(-) diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx index 8d9fa8a74a1..cd9cec8b309 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -7,4 +7,4 @@

``` -[Tiny RDM](https://redis.tinycraft.cc/) est une interface graphique Redis légère, moderne et open source. Elle dispose d'une belle interface utilisateur, d'une gestion intuitive de la base de données Redis et est compatible avec Windows, Mac et Linux. Elle offre des opérations visuelles de données clé-valeur, prend en charge diverses options de décodage et de visualisation des données, intègre une console pour exécuter des commandes, des requêtes de journal lent, et bien plus encore. \ No newline at end of file +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx index d1b9913b1ec..6baf0941b8c 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/templates.mdx @@ -47,7 +47,7 @@ Si vous n'êtes pas sûr d'un modèle, inspectez `package.json` et `wails.json` - [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - Un modèle utilisant Svelte - [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - Un modèle utilisant Svelte et Vite - [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - Un modèle utilisant Svelte et Vite avec TailwindCSS v3 -- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - Un modèle mis à jour en utilisant Svelte v4.2.0 et Vite avec TailwindCSS v3.3.3 - [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - Un modèle utilisant SvelteKit ## Solid @@ -62,7 +62,7 @@ Si vous n'êtes pas sûr d'un modèle, inspectez `package.json` et `wails.json` ## HTMX -- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Utilisez une combinaison unique de htmx pour interactivité, et de templ pour créer des composants et des formes ## Pure JavaScript (Vanilla) diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/application-development.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/application-development.mdx index 764f2240a53..411853fd6f1 100644 --- a/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/application-development.mdx +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/guides/application-development.mdx @@ -189,16 +189,16 @@ Le serveur de développement utilise une technique appelée "debouncing", ce qui Certains frameworks sont fournis avec leur propre serveur de rechargement en direct, cependant ils ne seront pas en mesure de tirer parti des liaisons Wails Go. Dans ce scénario, il est préférable d'exécuter un script qui va surveiller le projet dans dossier build, dossier que Wails surveille aussi. Pour un exemple, voir le modèle svelte par défaut qui utilise [rollup](https://rollupjs.org/guide/en/). -### Create React App +### Créer une application React -The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: +Le processus pour créer un projet Reactest un peu plus compliqué. Afin de prendre en charge le rechargement du frontend en direct, la configuration suivante doit être ajoutée à votre `wails.json`: ```json "frontend:dev:watcher": "yarn start", "frontend:dev:serverUrl": "http://localhost:3000", ``` -The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: +La commande `frontend:dev:watcher` démarrera le serveur de développement React (hébergé sur le port `3000` typiquement). La commande `frontend:dev:serverUrl` demande ensuite à Wails d'exposer les ressources depuis le serveur de développement lors du chargement du frontend, plutôt que depuis le dossier de construction. En plus de ce qui précède, le fichier `index.html` doit être mis à jour avec les éléments suivants : ```html @@ -208,7 +208,7 @@ The `frontend:dev:watcher` command will start the Create-React-App development s ``` -This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. +Ceci est nécessaire, car la commande watcher qui reconstruit le frontend empêche Wails de les injecter. Ça contourne le problème en assurant les scripts sont toujours injectés. Avec cette configuration, `wails dev` peut être exécuté, ce qui construira le frontend et le backend de manière appropriée avec le rechargement à chaud activé. De plus, lorsque vous accédez à l'application à partir d'un navigateur, les outils de développement de React peuvent maintenant être utilisés sur une version non minifiée de l'application pour le débogage. Enfin, pour des compilations plus rapides, `wails dev -s` peut être exécuté pour passer la construction par défaut du frontend par Wails car c'est une étape inutile. ## Module Go diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx index 87b505b4f96..cd9cec8b309 100644 --- a/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -7,4 +7,4 @@

``` -[Tiny RDM](https://redis.tinycraft.cc/)は、オープンソースで、モダンで軽量なRedis GUIです。美しいUI、直感的なRedisデータベースの管理を備え、Windows、Mac、Linuxと互換性があります。ビジュアルなキー-値データ操作を提供し、さまざまなデータのデコードと表示オプションをサポートし、コマンドを実行するための組み込みコンソール、遅いクエリの検索などをサポートしています。 \ No newline at end of file +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx index 7cfc95177fb..cd9cec8b309 100644 --- a/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -7,4 +7,4 @@

``` -[Tiny RDM](https://redis.tinycraft.cc/) 는 오픈 소스이며 현대적이고 가벼운 Redis GUI입니다. 아름다운 UI, 직관적인 Redis 데이터베이스 관리를 제공하며 Windows, Mac 및 Linux과 호환됩니다. 시각적인 키-값 데이터 작업을 지원하며 다양한 데이터 디코딩 및 표시 옵션을 제공하며 명령 실행을 위한 내장 콘솔, 느린 로그 쿼리 등을 지원합니다. \ No newline at end of file +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx index 8a7f4f6ac4b..cd9cec8b309 100644 --- a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -7,4 +7,4 @@

``` -O [Tiny RDM](https://redis.tinycraft.cc/) é uma GUI Redis moderna e leve de código aberto. Ele possui uma bela interface de usuário, gerenciamento intuitivo de banco de dados Redis e é compatível com Windows, Mac e Linux. Ele oferece operações visuais de dados chave-valor, suporta várias opções de decodificação e visualização de dados, um console embutido para executar comandos, consultas de log lento e muito mais. \ No newline at end of file +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx index 4e021d3a07d..cd9cec8b309 100644 --- a/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -7,4 +7,4 @@

``` -[Tiny RDM](https://redis.tinycraft.cc/) - это современный легкий Redis GUI с открытым исходным кодом. У него красивый пользовательский интерфейс, интуитивное управление базой данных Redis и поддержка Windows, Mac и Linux. Он предоставляет визуальные операции с данными ключ-значение, поддерживает различные варианты декодирования и просмотра данных, а также встроенную консоль для выполнения команд, запросы медленных записей и многое другое. \ No newline at end of file +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx index c710e181421..cd9cec8b309 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/tinyrdm.mdx @@ -7,4 +7,4 @@

``` -[Tiny RDM](https://redis.tinycraft.cc/)是一款开源的现代化轻量级Redis GUI。它拥有美观的操作界面,直观的Redis数据库管理,可在Windows、Mac和Linux上运行。它提供可视化键值数据操作,支持多种数据解码和查看方式,同时内置控制台以执行命令,支持慢查询日志。 +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. From d88a741ce70395aa77c86bb3e30a4fdad408be09 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sun, 12 Nov 2023 11:45:55 +1100 Subject: [PATCH 46/87] chore: update sponsors.svg (#3053) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index c349d8013ac..cf3aeef58ba 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -1,5 +1,5 @@ - + From 9ce0ddb4f8f9bc70a7ebe4b5647ef0752055c346 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 12 Nov 2023 12:30:49 +1100 Subject: [PATCH 47/87] [chore] lint fixes --- v2/cmd/wails/build.go | 2 - v2/cmd/wails/dev.go | 5 +- v2/cmd/wails/doctor.go | 4 +- v2/cmd/wails/flags/dev.go | 2 - v2/cmd/wails/flags/init.go | 1 - v2/cmd/wails/generate.go | 13 +++--- v2/cmd/wails/init.go | 16 +++---- v2/cmd/wails/internal/dev/dev.go | 5 +- v2/cmd/wails/internal/dev/watcher.go | 1 - v2/cmd/wails/internal/gomod/gomod.go | 2 +- v2/cmd/wails/main.go | 1 - v2/cmd/wails/update.go | 4 +- v2/examples/customlayout/go.mod | 6 ++- v2/examples/customlayout/go.sum | 1 + v2/go.mod | 4 +- v2/internal/app/app_devtools_not.go | 17 +++---- v2/internal/binding/binding.go | 4 +- v2/internal/binding/boundMethod.go | 1 - v2/internal/binding/db.go | 8 +--- v2/internal/binding/generate.go | 24 +++++----- v2/internal/binding/reflect.go | 2 - v2/internal/frontend/calls.go | 2 +- .../frontend/desktop/darwin/callbacks.go | 2 +- .../frontend/desktop/darwin/clipboard.go | 1 - v2/internal/frontend/desktop/darwin/dialog.go | 17 ++++--- .../frontend/desktop/darwin/frontend.go | 35 ++++++++------ .../frontend/desktop/darwin/inspector.go | 1 - v2/internal/frontend/desktop/darwin/menu.go | 2 +- .../frontend/desktop/darwin/menuitem.go | 11 +++-- v2/internal/frontend/desktop/darwin/screen.go | 1 + .../desktop/darwin/single_instance.go | 7 +-- v2/internal/frontend/desktop/darwin/window.go | 4 +- .../frontend/desktop/windows/dialog.go | 8 ++-- v2/internal/frontend/dispatcher/calls.go | 1 - v2/internal/frontend/dispatcher/events.go | 1 + .../frontend/dispatcher/securecalls.go | 2 +- .../frontend/dispatcher/systemcalls.go | 5 +- v2/internal/frontend/frontend.go | 4 +- v2/internal/frontend/runtime/events.go | 2 +- v2/internal/fs/fs.go | 14 +++--- v2/internal/github/github.go | 6 +-- .../cfd/CommonFileDialog_nonWindows.go | 10 ++-- .../go-common-file-dialog/cfd/errors.go | 4 +- .../cfd/iFileOpenDialog.go | 2 +- .../cfd/iShellItemArray.go | 2 +- .../cfd/vtblCommonFunc.go | 2 +- .../go-common-file-dialog/cfdutil/CFDUtil.go | 16 +++++-- v2/internal/gomod/gomod.go | 1 + v2/internal/gomod/gomod_data_unix.go | 4 ++ v2/internal/logger/custom_logger.go | 1 - v2/internal/logger/default_logger.go | 2 - v2/internal/menumanager/applicationmenu.go | 2 - v2/internal/menumanager/contextmenu.go | 2 - v2/internal/menumanager/menuitemmap.go | 7 +-- v2/internal/menumanager/menumanager.go | 5 +- v2/internal/menumanager/processedMenu.go | 28 +++++------ v2/internal/menumanager/traymenu.go | 8 ++-- v2/internal/process/process.go | 1 - v2/internal/project/project.go | 4 +- v2/internal/s/s.go | 16 +++---- v2/internal/shell/shell.go | 6 +-- v2/internal/signal/signal.go | 19 ++++---- v2/internal/staticanalysis/staticanalysis.go | 3 +- v2/internal/system/packagemanager/pm.go | 8 ++-- v2/internal/system/system.go | 13 ++---- v2/internal/typescriptify/typescriptify.go | 10 ++-- .../webview2runtime/webview2installer.go | 2 +- v2/pkg/application/application.go | 4 +- v2/pkg/assetserver/assethandler.go | 5 +- v2/pkg/assetserver/assetserver.go | 6 +-- v2/pkg/assetserver/assetserver_webview.go | 3 +- v2/pkg/assetserver/common.go | 4 +- v2/pkg/assetserver/webview/request_darwin.go | 5 +- v2/pkg/buildassets/buildassets.go | 4 +- v2/pkg/commands/bindings/bindings.go | 1 - v2/pkg/commands/build/base.go | 15 ++---- v2/pkg/commands/build/build.go | 7 +-- v2/pkg/commands/build/builder.go | 6 +-- v2/pkg/commands/build/nsis_installer.go | 4 +- v2/pkg/commands/build/packager.go | 27 +++++------ v2/pkg/logger/filelogger.go | 2 +- v2/pkg/mac/login_darwin.go | 2 +- v2/pkg/mac/notification_darwin.go | 2 +- v2/pkg/menu/callback.go | 2 +- v2/pkg/menu/colours/colours.go | 6 ++- v2/pkg/menu/keys/keys.go | 2 +- v2/pkg/menu/keys/parser.go | 3 -- v2/pkg/menu/keys/stringify.go | 9 ++-- v2/pkg/menu/menu.go | 3 +- v2/pkg/menu/menuitem.go | 14 ++---- v2/pkg/menu/menuroles.go | 46 +++++++++---------- v2/pkg/menu/styledlabel.go | 7 +++ v2/pkg/menu/tray.go | 3 +- v2/pkg/options/mac/mac.go | 4 +- v2/pkg/options/mac/preferences.go | 6 ++- v2/pkg/options/mac/titlebar.go | 2 - v2/pkg/options/options.go | 3 +- v2/pkg/options/windows/windows.go | 2 +- v2/pkg/runtime/dialog.go | 1 + v2/pkg/runtime/events.go | 4 +- v2/pkg/runtime/log.go | 1 + v2/pkg/runtime/menu.go | 1 + v2/pkg/runtime/runtime.go | 1 + v2/pkg/runtime/screen.go | 7 ++- v2/pkg/templates/generate/generate.go | 1 - v2/pkg/templates/templates.go | 7 --- v2/tools/release/release.go | 8 ++-- 107 files changed, 312 insertions(+), 355 deletions(-) mode change 100755 => 100644 v2/internal/binding/binding.go mode change 100755 => 100644 v2/internal/binding/reflect.go diff --git a/v2/cmd/wails/build.go b/v2/cmd/wails/build.go index 1c6b791ec85..858ecb4c9e2 100644 --- a/v2/cmd/wails/build.go +++ b/v2/cmd/wails/build.go @@ -18,7 +18,6 @@ import ( ) func buildApplication(f *flags.Build) error { - if f.NoColour { pterm.DisableColor() colour.ColourEnabled = false @@ -255,5 +254,4 @@ func buildApplication(f *flags.Build) error { } return nil - } diff --git a/v2/cmd/wails/dev.go b/v2/cmd/wails/dev.go index dbb2cf5d8fc..30213a68e13 100644 --- a/v2/cmd/wails/dev.go +++ b/v2/cmd/wails/dev.go @@ -1,16 +1,16 @@ package main import ( + "os" + "github.com/pterm/pterm" "github.com/wailsapp/wails/v2/cmd/wails/flags" "github.com/wailsapp/wails/v2/cmd/wails/internal/dev" "github.com/wailsapp/wails/v2/internal/colour" "github.com/wailsapp/wails/v2/pkg/clilogger" - "os" ) func devApplication(f *flags.Dev) error { - if f.NoColour { pterm.DisableColor() colour.ColourEnabled = false @@ -34,5 +34,4 @@ func devApplication(f *flags.Dev) error { } return dev.Application(f, logger) - } diff --git a/v2/cmd/wails/doctor.go b/v2/cmd/wails/doctor.go index ef73419ecc8..74f9ec6cc88 100644 --- a/v2/cmd/wails/doctor.go +++ b/v2/cmd/wails/doctor.go @@ -128,8 +128,8 @@ func diagnoseEnvironment(f *flags.Doctor) error { // Output Dependencies Status var dependenciesMissing []string var externalPackages []*packagemanager.Dependency - var dependenciesAvailableRequired = 0 - var dependenciesAvailableOptional = 0 + dependenciesAvailableRequired := 0 + dependenciesAvailableOptional := 0 dependenciesTableData := pterm.TableData{ {"Dependency", "Package Name", "Status", "Version"}, diff --git a/v2/cmd/wails/flags/dev.go b/v2/cmd/wails/flags/dev.go index 7e5e6239c74..501450a982e 100644 --- a/v2/cmd/wails/flags/dev.go +++ b/v2/cmd/wails/flags/dev.go @@ -47,7 +47,6 @@ func (*Dev) Default() *Dev { } func (d *Dev) Process() error { - var err error err = d.loadAndMergeProjectConfig() if err != nil { @@ -113,7 +112,6 @@ func (d *Dev) loadAndMergeProjectConfig() error { } return nil - } // GenerateBuildOptions creates a build.Options using the flags diff --git a/v2/cmd/wails/flags/init.go b/v2/cmd/wails/flags/init.go index 6e642ec9a2f..16d56a207d9 100644 --- a/v2/cmd/wails/flags/init.go +++ b/v2/cmd/wails/flags/init.go @@ -14,7 +14,6 @@ type Init struct { } func (i *Init) Default() *Init { - result := &Init{ TemplateName: "vanilla", } diff --git a/v2/cmd/wails/generate.go b/v2/cmd/wails/generate.go index a7b059ecfdd..159df90a1b5 100644 --- a/v2/cmd/wails/generate.go +++ b/v2/cmd/wails/generate.go @@ -2,6 +2,9 @@ package main import ( "fmt" + "os" + "path/filepath" + "github.com/leaanthony/debme" "github.com/leaanthony/gosod" "github.com/pterm/pterm" @@ -14,12 +17,9 @@ import ( "github.com/wailsapp/wails/v2/pkg/clilogger" "github.com/wailsapp/wails/v2/pkg/commands/bindings" "github.com/wailsapp/wails/v2/pkg/commands/buildtags" - "os" - "path/filepath" ) func generateModule(f *flags.GenerateModule) error { - if f.NoColour { pterm.DisableColor() colour.ColourEnabled = false @@ -55,7 +55,6 @@ func generateModule(f *flags.GenerateModule) error { } func generateTemplate(f *flags.GenerateTemplate) error { - if f.NoColour { pterm.DisableColor() colour.ColourEnabled = false @@ -77,7 +76,7 @@ func generateTemplate(f *flags.GenerateTemplate) error { } templateDir := filepath.Join(cwd, f.Name) if !fs.DirExists(templateDir) { - err := os.MkdirAll(templateDir, 0755) + err := os.MkdirAll(templateDir, 0o755) if err != nil { return err } @@ -200,7 +199,7 @@ func processPackageJSON(frontendDir string) error { json, _ = sjson.SetBytes(json, "name", "{{.ProjectName}}") json, _ = sjson.SetBytes(json, "author", "{{.AuthorName}}") - err = os.WriteFile(packageJSON, json, 0644) + err = os.WriteFile(packageJSON, json, 0o644) if err != nil { return err } @@ -231,7 +230,7 @@ func processPackageLockJSON(frontendDir string) error { printBulletPoint("Updating package-lock.json data...") json, _ = sjson.Set(json, "name", "{{.ProjectName}}") - err = os.WriteFile(filename, []byte(json), 0644) + err = os.WriteFile(filename, []byte(json), 0o644) if err != nil { return err } diff --git a/v2/cmd/wails/init.go b/v2/cmd/wails/init.go index f9a9c6b3f00..709905c0f17 100644 --- a/v2/cmd/wails/init.go +++ b/v2/cmd/wails/init.go @@ -3,6 +3,12 @@ package main import ( "bufio" "fmt" + "os" + "os/exec" + "path/filepath" + "strings" + "time" + "github.com/flytam/filenamify" "github.com/leaanthony/slicer" "github.com/pkg/errors" @@ -13,15 +19,9 @@ import ( "github.com/wailsapp/wails/v2/pkg/clilogger" "github.com/wailsapp/wails/v2/pkg/git" "github.com/wailsapp/wails/v2/pkg/templates" - "os" - "os/exec" - "path/filepath" - "strings" - "time" ) func initProject(f *flags.Init) error { - if f.NoColour { pterm.DisableColor() colour.ColourEnabled = false @@ -215,7 +215,7 @@ func initGit(options *templates.Options) error { "frontend/dist", "frontend/node_modules", } - err = os.WriteFile(filepath.Join(options.TargetDir, ".gitignore"), []byte(strings.Join(ignore, "\n")), 0644) + err = os.WriteFile(filepath.Join(options.TargetDir, ".gitignore"), []byte(strings.Join(ignore, "\n")), 0o644) if err != nil { return errors.Wrap(err, "Unable to create gitignore") } @@ -271,7 +271,7 @@ func updateReplaceLine(targetPath string) { } } - err = os.WriteFile("go.mod", []byte(strings.Join(lines, "\n")), 0644) + err = os.WriteFile("go.mod", []byte(strings.Join(lines, "\n")), 0o644) if err != nil { fatal(err.Error()) } diff --git a/v2/cmd/wails/internal/dev/dev.go b/v2/cmd/wails/internal/dev/dev.go index 80d7a6d875c..1f9b95c8994 100644 --- a/v2/cmd/wails/internal/dev/dev.go +++ b/v2/cmd/wails/internal/dev/dev.go @@ -52,7 +52,6 @@ func sliceToMap(input []string) map[string]struct{} { // Application runs the application in dev mode func Application(f *flags.Dev, logger *clilogger.CLILogger) error { - cwd := lo.Must(os.Getwd()) // Update go.mod to use current wails version @@ -271,7 +270,6 @@ func runFrontendDevWatcherCommand(frontendDirectory string, devCommand string, d // restartApp does the actual rebuilding of the application when files change func restartApp(buildOptions *build.Options, debugBinaryProcess *process.Process, f *flags.Dev, exitCodeChannel chan int, legacyUseDevServerInsteadofCustomScheme bool) (*process.Process, string, error) { - appBinary, err := build.Build(buildOptions) println() if err != nil { @@ -298,7 +296,6 @@ func restartApp(buildOptions *build.Options, debugBinaryProcess *process.Process // parse appargs if any args, err := shlex.Split(f.AppArgs) - if err != nil { buildOptions.Logger.Fatal("Unable to parse appargs: %s", err.Error()) } @@ -345,7 +342,7 @@ func doWatcherLoop(cwd string, buildOptions *build.Options, debugBinaryProcess * logutils.LogGreen("Watching (sub)/directory: %s", cwd) // Main Loop - var extensionsThatTriggerARebuild = sliceToMap(strings.Split(f.Extensions, ",")) + extensionsThatTriggerARebuild := sliceToMap(strings.Split(f.Extensions, ",")) var dirsThatTriggerAReload []string for _, dir := range strings.Split(f.ReloadDirs, ",") { if dir == "" { diff --git a/v2/cmd/wails/internal/dev/watcher.go b/v2/cmd/wails/internal/dev/watcher.go index 499b76dfd38..4e7457ef942 100644 --- a/v2/cmd/wails/internal/dev/watcher.go +++ b/v2/cmd/wails/internal/dev/watcher.go @@ -18,7 +18,6 @@ type Watcher interface { // initialiseWatcher creates the project directory watcher that will trigger recompile func initialiseWatcher(cwd string) (*fsnotify.Watcher, error) { - // Ignore dot files, node_modules and build directories by default ignoreDirs := getIgnoreDirs(cwd) diff --git a/v2/cmd/wails/internal/gomod/gomod.go b/v2/cmd/wails/internal/gomod/gomod.go index 52e56344b33..5da14a5ff96 100644 --- a/v2/cmd/wails/internal/gomod/gomod.go +++ b/v2/cmd/wails/internal/gomod/gomod.go @@ -56,7 +56,7 @@ func SyncGoMod(logger *clilogger.CLILogger, updateWailsVersion bool) error { } if updated { - return os.WriteFile(gomodFilename, gomodData, 0755) + return os.WriteFile(gomodFilename, gomodData, 0o755) } return nil diff --git a/v2/cmd/wails/main.go b/v2/cmd/wails/main.go index a0a7d4a319a..ccf1576e9c1 100644 --- a/v2/cmd/wails/main.go +++ b/v2/cmd/wails/main.go @@ -66,7 +66,6 @@ func bool2Str(b bool) string { var app *clir.Cli func main() { - var err error app = clir.NewCli("Wails", "Go/HTML Appkit", internal.Version) diff --git a/v2/cmd/wails/update.go b/v2/cmd/wails/update.go index ac0e7375af2..9f8b6e60462 100644 --- a/v2/cmd/wails/update.go +++ b/v2/cmd/wails/update.go @@ -15,7 +15,6 @@ import ( // AddSubcommand adds the `init` command for the Wails application func update(f *flags.Update) error { - if f.NoColour { colour.ColourEnabled = false pterm.DisableColor() @@ -73,8 +72,7 @@ func update(f *flags.Update) error { } func updateToVersion(targetVersion *github.SemanticVersion, force bool, currentVersion string) error { - - var targetVersionString = "v" + targetVersion.String() + targetVersionString := "v" + targetVersion.String() if targetVersionString == currentVersion { pterm.Println("\nLooks like you're up to date!") diff --git a/v2/examples/customlayout/go.mod b/v2/examples/customlayout/go.mod index ccec36a7604..8947007a7b0 100644 --- a/v2/examples/customlayout/go.mod +++ b/v2/examples/customlayout/go.mod @@ -1,6 +1,8 @@ module changeme -go 1.18 +go 1.21 + +toolchain go1.21.0 require github.com/wailsapp/wails/v2 v2.1.0 @@ -24,7 +26,7 @@ require ( github.com/tkrajina/go-reflector v0.5.6 // indirect github.com/valyala/bytebufferpool v1.0.0 // indirect github.com/valyala/fasttemplate v1.2.2 // indirect - github.com/wailsapp/go-webview2 v1.0.7 // indirect + github.com/wailsapp/go-webview2 v1.0.10 // indirect github.com/wailsapp/mimetype v1.4.1 // indirect golang.org/x/crypto v0.14.0 // indirect golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 // indirect diff --git a/v2/examples/customlayout/go.sum b/v2/examples/customlayout/go.sum index fb0946bc54c..4cdbdac671c 100644 --- a/v2/examples/customlayout/go.sum +++ b/v2/examples/customlayout/go.sum @@ -56,6 +56,7 @@ github.com/valyala/fasttemplate v1.2.2 h1:lxLXG0uE3Qnshl9QyaK6XJxMXlQZELvChBOCmQ github.com/valyala/fasttemplate v1.2.2/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= github.com/wailsapp/go-webview2 v1.0.7 h1:s95+7irJsAsTy1RsjJ6N0cYX7tZ4gP7Uzawds0L2urs= github.com/wailsapp/go-webview2 v1.0.7/go.mod h1:Uk2BePfCRzttBBjFrBmqKGJd41P6QIHeV9kTgIeOZNo= +github.com/wailsapp/go-webview2 v1.0.10/go.mod h1:Uk2BePfCRzttBBjFrBmqKGJd41P6QIHeV9kTgIeOZNo= github.com/wailsapp/mimetype v1.4.1 h1:pQN9ycO7uo4vsUUuPeHEYoUkLVkaRntMnHJxVwYhwHs= github.com/wailsapp/mimetype v1.4.1/go.mod h1:9aV5k31bBOv5z6u+QP8TltzvNGJPmNJD4XlAL3U+j3o= golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= diff --git a/v2/go.mod b/v2/go.mod index 88bd7c75cda..b166a30b2ef 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -1,6 +1,8 @@ module github.com/wailsapp/wails/v2 -go 1.18 +go 1.21 + +toolchain go1.21.0 require ( github.com/Masterminds/semver v1.5.0 diff --git a/v2/internal/app/app_devtools_not.go b/v2/internal/app/app_devtools_not.go index d37ee3124ac..91267204864 100644 --- a/v2/internal/app/app_devtools_not.go +++ b/v2/internal/app/app_devtools_not.go @@ -1,8 +1,9 @@ -//go:build !devtools - -package app - -// Note: devtools flag is also added in debug builds -func IsDevtoolsEnabled() bool { - return false -} +//go:build !devtools + +package app + +// IsDevtoolsEnabled returns true if devtools should be enabled +// Note: devtools flag is also added in debug builds +func IsDevtoolsEnabled() bool { + return false +} diff --git a/v2/internal/binding/binding.go b/v2/internal/binding/binding.go old mode 100755 new mode 100644 index 75b821f29ca..d911e12a3d7 --- a/v2/internal/binding/binding.go +++ b/v2/internal/binding/binding.go @@ -60,7 +60,6 @@ func NewBindings(logger *logger.Logger, structPointersToBind []interface{}, exem // Add the given struct methods to the Bindings func (b *Bindings) Add(structPtr interface{}) error { - methods, err := b.getMethods(structPtr) if err != nil { return fmt.Errorf("cannot bind value to app: %s", err.Error()) @@ -146,7 +145,6 @@ func (b *Bindings) GenerateModels() ([]byte, error) { } func (b *Bindings) WriteModels(modelsDir string) error { - modelsData, err := b.GenerateModels() if err != nil { return err @@ -157,7 +155,7 @@ func (b *Bindings) WriteModels(modelsDir string) error { } filename := filepath.Join(modelsDir, "models.ts") - err = os.WriteFile(filename, modelsData, 0755) + err = os.WriteFile(filename, modelsData, 0o755) if err != nil { return err } diff --git a/v2/internal/binding/boundMethod.go b/v2/internal/binding/boundMethod.go index f6ffdb600bc..c13e2ff37cc 100644 --- a/v2/internal/binding/boundMethod.go +++ b/v2/internal/binding/boundMethod.go @@ -28,7 +28,6 @@ func (b *BoundMethod) OutputCount() int { // ParseArgs method converts the input json into the types expected by the method func (b *BoundMethod) ParseArgs(args []json.RawMessage) ([]interface{}, error) { - result := make([]interface{}, b.InputCount()) if len(args) != b.InputCount() { return nil, fmt.Errorf("received %d arguments to method '%s', expected %d", len(args), b.Name, b.InputCount()) diff --git a/v2/internal/binding/db.go b/v2/internal/binding/db.go index f22798680ba..1fc7e8c662e 100644 --- a/v2/internal/binding/db.go +++ b/v2/internal/binding/db.go @@ -34,7 +34,6 @@ func newDB() *DB { // GetMethodFromStore returns the method for the given package/struct/method names // nil is returned if any one of those does not exist func (d *DB) GetMethodFromStore(packageName string, structName string, methodName string) *BoundMethod { - // Lock the db whilst processing and unlock on return d.lock.RLock() defer d.lock.RUnlock() @@ -53,7 +52,6 @@ func (d *DB) GetMethodFromStore(packageName string, structName string, methodNam // GetMethod returns the method for the given qualified method name // qualifiedMethodName is "packagename.structname.methodname" func (d *DB) GetMethod(qualifiedMethodName string) *BoundMethod { - // Lock the db whilst processing and unlock on return d.lock.RLock() defer d.lock.RUnlock() @@ -72,7 +70,6 @@ func (d *DB) GetObfuscatedMethod(id int) *BoundMethod { // AddMethod adds the given method definition to the db using the given qualified path: packageName.structName.methodName func (d *DB) AddMethod(packageName string, structName string, methodName string, methodDefinition *BoundMethod) { - // Lock the db whilst processing and unlock on return d.lock.Lock() defer d.lock.Unlock() @@ -99,12 +96,10 @@ func (d *DB) AddMethod(packageName string, structName string, methodName string, // Store in the methodMap key := packageName + "." + structName + "." + methodName d.methodMap[key] = methodDefinition - } // ToJSON converts the method map to JSON func (d *DB) ToJSON() (string, error) { - // Lock the db whilst processing and unlock on return d.lock.RLock() defer d.lock.RUnlock() @@ -120,8 +115,7 @@ func (d *DB) ToJSON() (string, error) { // UpdateObfuscatedCallMap sets up the secure call mappings func (d *DB) UpdateObfuscatedCallMap() map[string]int { - - var mappings = make(map[string]int) + mappings := make(map[string]int) // Iterate map keys and sort them keys := make([]string, 0, len(d.methodMap)) diff --git a/v2/internal/binding/generate.go b/v2/internal/binding/generate.go index 8416aade1ca..02a0bd29272 100644 --- a/v2/internal/binding/generate.go +++ b/v2/internal/binding/generate.go @@ -15,12 +15,14 @@ import ( "github.com/leaanthony/slicer" ) -var mapRegex *regexp.Regexp -var keyPackageIndex int -var keyTypeIndex int -var valueArrayIndex int -var valuePackageIndex int -var valueTypeIndex int +var ( + mapRegex *regexp.Regexp + keyPackageIndex int + keyTypeIndex int + valueArrayIndex int + valuePackageIndex int + valueTypeIndex int +) func init() { mapRegex = regexp.MustCompile(`(?:map\[(?:(?P\w+)\.)?(?P\w+)])?(?P\[])?(?:\*?(?P\w+)\.)?(?P.+)`) @@ -81,9 +83,7 @@ func (b *Bindings) GenerateGoBindings(baseDir string) error { } else { jsoutput.WriteString(fmt.Sprintf(" return window['go']['%s']['%s']['%s'](%s);", packageName, structName, methodName, argsString)) } - jsoutput.WriteString("\n") - jsoutput.WriteString(fmt.Sprintf("}")) - jsoutput.WriteString("\n") + jsoutput.WriteString("\n}\n") // Generate TS tsBody.WriteString(fmt.Sprintf("\nexport function %s(", methodName)) @@ -127,12 +127,12 @@ func (b *Bindings) GenerateGoBindings(baseDir string) error { tsContent.WriteString(tsBody.String()) jsfilename := filepath.Join(packageDir, structName+".js") - err = os.WriteFile(jsfilename, jsoutput.Bytes(), 0755) + err = os.WriteFile(jsfilename, jsoutput.Bytes(), 0o755) if err != nil { return err } tsfilename := filepath.Join(packageDir, structName+".d.ts") - err = os.WriteFile(tsfilename, tsContent.Bytes(), 0755) + err = os.WriteFile(tsfilename, tsContent.Bytes(), 0o755) if err != nil { return err } @@ -186,7 +186,7 @@ func goTypeToJSDocType(input string, importNamespaces *slicer.StringSlicer) stri valueArray := matches[valueArrayIndex] valuePackage := matches[valuePackageIndex] valueType := matches[valueTypeIndex] - //fmt.Printf("input=%s, keyPackage=%s, keyType=%s, valueArray=%s, valuePackage=%s, valueType=%s\n", + // fmt.Printf("input=%s, keyPackage=%s, keyType=%s, valueArray=%s, valuePackage=%s, valueType=%s\n", // input, // keyPackage, // keyType, diff --git a/v2/internal/binding/reflect.go b/v2/internal/binding/reflect.go old mode 100755 new mode 100644 index 66a9cf7bdea..263af29d1e9 --- a/v2/internal/binding/reflect.go +++ b/v2/internal/binding/reflect.go @@ -25,7 +25,6 @@ func isStruct(value interface{}) bool { } func (b *Bindings) getMethods(value interface{}) ([]*BoundMethod, error) { - // Create result placeholder var result []*BoundMethod @@ -169,7 +168,6 @@ func getPackageName(in string) string { func getSplitReturn(in string) (string, string) { result := strings.Split(in, ".") return result[0], result[1] - } func hasElements(typ reflect.Type) bool { diff --git a/v2/internal/frontend/calls.go b/v2/internal/frontend/calls.go index 3983c24bfcd..5401106bc5f 100644 --- a/v2/internal/frontend/calls.go +++ b/v2/internal/frontend/calls.go @@ -1,5 +1,5 @@ package frontend type Calls interface { - Callback(string) + Callback(message string) } diff --git a/v2/internal/frontend/desktop/darwin/callbacks.go b/v2/internal/frontend/desktop/darwin/callbacks.go index 7d930a2f9e5..ab0d18e47ab 100644 --- a/v2/internal/frontend/desktop/darwin/callbacks.go +++ b/v2/internal/frontend/desktop/darwin/callbacks.go @@ -12,6 +12,7 @@ package darwin #include */ import "C" + import ( "errors" "strconv" @@ -20,7 +21,6 @@ import ( ) func (f *Frontend) handleCallback(menuItemID uint) error { - menuItem := getMenuItemForID(menuItemID) if menuItem == nil { return errors.New("unknown menuItem ID: " + strconv.Itoa(int(menuItemID))) diff --git a/v2/internal/frontend/desktop/darwin/clipboard.go b/v2/internal/frontend/desktop/darwin/clipboard.go index eea6c79aedb..c40ba877175 100644 --- a/v2/internal/frontend/desktop/darwin/clipboard.go +++ b/v2/internal/frontend/desktop/darwin/clipboard.go @@ -16,7 +16,6 @@ func (f *Frontend) ClipboardGetText() (string, error) { } func (f *Frontend) ClipboardSetText(text string) error { - copyCmd := exec.Command("pbcopy") in, err := copyCmd.StdinPipe() if err != nil { diff --git a/v2/internal/frontend/desktop/darwin/dialog.go b/v2/internal/frontend/desktop/darwin/dialog.go index c6be559cbec..66bb2f13a6c 100644 --- a/v2/internal/frontend/desktop/darwin/dialog.go +++ b/v2/internal/frontend/desktop/darwin/dialog.go @@ -11,6 +11,7 @@ package darwin #import "WailsContext.h" */ import "C" + import ( "encoding/json" "fmt" @@ -23,10 +24,12 @@ import ( ) // Obj-C dialog methods send the response to this channel -var messageDialogResponse = make(chan int) -var openFileDialogResponse = make(chan string) -var saveFileDialogResponse = make(chan string) -var dialogLock sync.Mutex +var ( + messageDialogResponse = make(chan int) + openFileDialogResponse = make(chan string) + saveFileDialogResponse = make(chan string) + dialogLock sync.Mutex +) // OpenDirectoryDialog prompts the user to select a directory func (f *Frontend) OpenDirectoryDialog(options frontend.OpenDialogOptions) (string, error) { @@ -74,7 +77,7 @@ func (f *Frontend) openDialog(options *frontend.OpenDialogOptions, multiple bool filters := filterStrings.Join(";") C.OpenFileDialog(f.mainWindow.context, title, defaultFilename, defaultDirectory, allowDirectories, allowFiles, canCreateDirectories, treatPackagesAsDirectories, resolveAliases, showHiddenFiles, allowMultipleFileSelection, c.String(filters)) - var result = <-openFileDialogResponse + result := <-openFileDialogResponse var parsedResults []string err := json.Unmarshal([]byte(result), &parsedResults) @@ -130,7 +133,7 @@ func (f *Frontend) SaveFileDialog(options frontend.SaveDialogOptions) (string, e filters := filterStrings.Join(";") C.SaveFileDialog(f.mainWindow.context, title, defaultFilename, defaultDirectory, canCreateDirectories, treatPackagesAsDirectories, showHiddenFiles, c.String(filters)) - var result = <-saveFileDialogResponse + result := <-saveFileDialogResponse return result, nil } @@ -165,7 +168,7 @@ func (f *Frontend) MessageDialog(options frontend.MessageDialogOptions) (string, C.MessageDialog(f.mainWindow.context, dialogType, title, message, buttons[0], buttons[1], buttons[2], buttons[3], defaultButton, cancelButton, iconData, iconDataLength) - var result = <-messageDialogResponse + result := <-messageDialogResponse selectedC := buttons[result] var selected string diff --git a/v2/internal/frontend/desktop/darwin/frontend.go b/v2/internal/frontend/desktop/darwin/frontend.go index c792ae9c88e..bc8a4c64c34 100644 --- a/v2/internal/frontend/desktop/darwin/frontend.go +++ b/v2/internal/frontend/desktop/darwin/frontend.go @@ -14,6 +14,7 @@ package darwin #include */ import "C" + import ( "context" "encoding/json" @@ -36,15 +37,16 @@ import ( const startURL = "wails://wails/" -var messageBuffer = make(chan string, 100) -var requestBuffer = make(chan webview.Request, 100) -var callbackBuffer = make(chan uint, 10) -var openFilepathBuffer = make(chan string, 100) -var openUrlBuffer = make(chan string, 100) -var secondInstanceBuffer = make(chan options.SecondInstanceData, 1) +var ( + messageBuffer = make(chan string, 100) + requestBuffer = make(chan webview.Request, 100) + callbackBuffer = make(chan uint, 10) + openFilepathBuffer = make(chan string, 100) + openUrlBuffer = make(chan string, 100) + secondInstanceBuffer = make(chan options.SecondInstanceData, 1) +) type Frontend struct { - // Context ctx context.Context @@ -153,6 +155,7 @@ func (f *Frontend) startRequestProcessor() { f.assets.ServeWebViewRequest(request) } } + func (f *Frontend) startCallbackProcessor() { for callback := range callbackBuffer { err := f.handleCallback(callback) @@ -171,15 +174,12 @@ func (f *Frontend) WindowReloadApp() { } func (f *Frontend) WindowSetSystemDefaultTheme() { - return } func (f *Frontend) WindowSetLightTheme() { - return } func (f *Frontend) WindowSetDarkTheme() { - return } func (f *Frontend) Run(ctx context.Context) error { @@ -189,8 +189,8 @@ func (f *Frontend) Run(ctx context.Context) error { SetupSingleInstance(f.frontendOptions.SingleInstanceLock.UniqueId) } - var _debug = ctx.Value("debug") - var _devtoolsEnabled = ctx.Value("devtoolsEnabled") + _debug := ctx.Value("debug") + _devtoolsEnabled := ctx.Value("devtoolsEnabled") if _debug != nil { f.debug = _debug.(bool) @@ -215,6 +215,7 @@ func (f *Frontend) Run(ctx context.Context) error { func (f *Frontend) WindowCenter() { f.mainWindow.Center() } + func (f *Frontend) WindowSetAlwaysOnTop(onTop bool) { f.mainWindow.SetAlwaysOnTop(onTop) } @@ -222,6 +223,7 @@ func (f *Frontend) WindowSetAlwaysOnTop(onTop bool) { func (f *Frontend) WindowSetPosition(x, y int) { f.mainWindow.SetPosition(x, y) } + func (f *Frontend) WindowGetPosition() (int, int) { return f.mainWindow.GetPosition() } @@ -253,6 +255,7 @@ func (f *Frontend) WindowShow() { func (f *Frontend) WindowHide() { f.mainWindow.Hide() } + func (f *Frontend) Show() { f.mainWindow.ShowApplication() } @@ -260,18 +263,23 @@ func (f *Frontend) Show() { func (f *Frontend) Hide() { f.mainWindow.HideApplication() } + func (f *Frontend) WindowMaximise() { f.mainWindow.Maximise() } + func (f *Frontend) WindowToggleMaximise() { f.mainWindow.ToggleMaximise() } + func (f *Frontend) WindowUnmaximise() { f.mainWindow.UnMaximise() } + func (f *Frontend) WindowMinimise() { f.mainWindow.Minimise() } + func (f *Frontend) WindowUnminimise() { f.mainWindow.UnMinimise() } @@ -279,6 +287,7 @@ func (f *Frontend) WindowUnminimise() { func (f *Frontend) WindowSetMinSize(width int, height int) { f.mainWindow.SetMinSize(width, height) } + func (f *Frontend) WindowSetMaxSize(width int, height int) { f.mainWindow.SetMaxSize(width, height) } @@ -345,7 +354,6 @@ func (f *Frontend) Notify(name string, data ...interface{}) { } func (f *Frontend) processMessage(message string) { - if message == "DomReady" { if f.frontendOptions.OnDomReady != nil { f.frontendOptions.OnDomReady(f.ctx) @@ -388,7 +396,6 @@ func (f *Frontend) processMessage(message string) { f.logger.Info("Unknown message returned from dispatcher: %+v", result) } }() - } func (f *Frontend) ProcessOpenFileEvent(filePath string) { diff --git a/v2/internal/frontend/desktop/darwin/inspector.go b/v2/internal/frontend/desktop/darwin/inspector.go index 8499745d0fe..dc3f08969a0 100644 --- a/v2/internal/frontend/desktop/darwin/inspector.go +++ b/v2/internal/frontend/desktop/darwin/inspector.go @@ -7,5 +7,4 @@ import ( ) func showInspector(_ unsafe.Pointer) { - } diff --git a/v2/internal/frontend/desktop/darwin/menu.go b/v2/internal/frontend/desktop/darwin/menu.go index 08090f89a2c..24dbe32017d 100644 --- a/v2/internal/frontend/desktop/darwin/menu.go +++ b/v2/internal/frontend/desktop/darwin/menu.go @@ -13,6 +13,7 @@ package darwin #include */ import "C" + import ( "unsafe" @@ -122,7 +123,6 @@ func processMenuItem(parent *NSMenu, menuItem *menu.MenuItem) *MenuItem { } return parent.AddMenuItem(menuItem) - } func (f *Frontend) MenuSetApplicationMenu(menu *menu.Menu) { diff --git a/v2/internal/frontend/desktop/darwin/menuitem.go b/v2/internal/frontend/desktop/darwin/menuitem.go index 00ad57aa37f..64aab84a9a7 100644 --- a/v2/internal/frontend/desktop/darwin/menuitem.go +++ b/v2/internal/frontend/desktop/darwin/menuitem.go @@ -13,16 +13,19 @@ package darwin #include */ import "C" + import ( "log" "math" "sync" ) -var menuItemToID = make(map[*MenuItem]uint) -var idToMenuItem = make(map[uint]*MenuItem) -var menuItemLock sync.Mutex -var menuItemIDCounter uint = 0 +var ( + menuItemToID = make(map[*MenuItem]uint) + idToMenuItem = make(map[uint]*MenuItem) + menuItemLock sync.Mutex + menuItemIDCounter uint = 0 +) func createMenuItemID(item *MenuItem) uint { menuItemLock.Lock() diff --git a/v2/internal/frontend/desktop/darwin/screen.go b/v2/internal/frontend/desktop/darwin/screen.go index a96a8efa743..bd64a31f9eb 100644 --- a/v2/internal/frontend/desktop/darwin/screen.go +++ b/v2/internal/frontend/desktop/darwin/screen.go @@ -82,6 +82,7 @@ Screen GetNthScreen(int nth, void *inctx){ */ import "C" + import ( "unsafe" diff --git a/v2/internal/frontend/desktop/darwin/single_instance.go b/v2/internal/frontend/desktop/darwin/single_instance.go index 012baef0280..970f1f431f5 100644 --- a/v2/internal/frontend/desktop/darwin/single_instance.go +++ b/v2/internal/frontend/desktop/darwin/single_instance.go @@ -10,21 +10,22 @@ package darwin */ import "C" + import ( "encoding/json" "fmt" - "github.com/wailsapp/wails/v2/pkg/options" "os" "strings" "syscall" "unsafe" + + "github.com/wailsapp/wails/v2/pkg/options" ) func SetupSingleInstance(uniqueID string) { lockFilePath := getTempDir() lockFileName := uniqueID + ".lock" _, err := createLockFile(lockFilePath + "/" + lockFileName) - // if lockFile exist – send notification to second instance if err != nil { c := NewCalloc() @@ -63,7 +64,7 @@ func HandleSecondInstanceData(secondInstanceMessage *C.char) { // exclusive lock on it. If the file already exists AND is still locked, it will // fail. func createLockFile(filename string) (*os.File, error) { - file, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0600) + file, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0o600) if err != nil { fmt.Printf("Failed to open lockfile %s: %s", filename, err) return nil, err diff --git a/v2/internal/frontend/desktop/darwin/window.go b/v2/internal/frontend/desktop/darwin/window.go index 740848df144..236026ee207 100644 --- a/v2/internal/frontend/desktop/darwin/window.go +++ b/v2/internal/frontend/desktop/darwin/window.go @@ -13,6 +13,7 @@ package darwin #include */ import "C" + import ( "log" "runtime" @@ -46,7 +47,6 @@ func bool2CboolPtr(value bool) *C.bool { } func NewWindow(frontendOptions *options.App, debug bool, devtools bool) *Window { - c := NewCalloc() defer c.Free() @@ -196,6 +196,7 @@ func (w *Window) SetTitle(title string) { func (w *Window) Maximise() { C.Maximise(w.context) } + func (w *Window) ToggleMaximise() { C.ToggleMaximise(w.context) } @@ -251,6 +252,7 @@ func (w *Window) Show() { func (w *Window) Hide() { C.Hide(w.context) } + func (w *Window) ShowApplication() { C.ShowApplication(w.context) } diff --git a/v2/internal/frontend/desktop/windows/dialog.go b/v2/internal/frontend/desktop/windows/dialog.go index 1ca422b7199..7aca4289438 100644 --- a/v2/internal/frontend/desktop/windows/dialog.go +++ b/v2/internal/frontend/desktop/windows/dialog.go @@ -47,7 +47,7 @@ func (f *Frontend) OpenDirectoryDialog(options frontend.OpenDialogOptions) (stri return cfd.NewSelectFolderDialog(config) }, false) - if err != nil && err != cfd.ErrorCancelled { + if err != nil && err != cfd.ErrCancelled { return "", err } return result.(string), nil @@ -72,7 +72,7 @@ func (f *Frontend) OpenFileDialog(options frontend.OpenDialogOptions) (string, e return cfd.NewOpenFileDialog(config) }, false) - if err != nil && err != cfd.ErrorCancelled { + if err != nil && err != cfd.ErrCancelled { return "", err } return result.(string), nil @@ -99,7 +99,7 @@ func (f *Frontend) OpenMultipleFilesDialog(options frontend.OpenDialogOptions) ( return cfd.NewOpenMultipleFilesDialog(config) }, true) - if err != nil && err != cfd.ErrorCancelled { + if err != nil && err != cfd.ErrCancelled { return nil, err } return result.([]string), nil @@ -126,7 +126,7 @@ func (f *Frontend) SaveFileDialog(options frontend.SaveDialogOptions) (string, e return cfd.NewSaveFileDialog(config) }, false) - if err != nil && err != cfd.ErrorCancelled { + if err != nil && err != cfd.ErrCancelled { return "", err } return result.(string), nil diff --git a/v2/internal/frontend/dispatcher/calls.go b/v2/internal/frontend/dispatcher/calls.go index 491d17fe2bd..ba10629137f 100644 --- a/v2/internal/frontend/dispatcher/calls.go +++ b/v2/internal/frontend/dispatcher/calls.go @@ -15,7 +15,6 @@ type callMessage struct { } func (d *Dispatcher) processCallMessage(message string, sender frontend.Frontend) (string, error) { - var payload callMessage err := json.Unmarshal([]byte(message[1:]), &payload) if err != nil { diff --git a/v2/internal/frontend/dispatcher/events.go b/v2/internal/frontend/dispatcher/events.go index 11680651bb1..12fe7b89e04 100644 --- a/v2/internal/frontend/dispatcher/events.go +++ b/v2/internal/frontend/dispatcher/events.go @@ -3,6 +3,7 @@ package dispatcher import ( "encoding/json" "errors" + "github.com/wailsapp/wails/v2/internal/frontend" ) diff --git a/v2/internal/frontend/dispatcher/securecalls.go b/v2/internal/frontend/dispatcher/securecalls.go index 40accaa26e6..8cdcdfb8507 100644 --- a/v2/internal/frontend/dispatcher/securecalls.go +++ b/v2/internal/frontend/dispatcher/securecalls.go @@ -3,6 +3,7 @@ package dispatcher import ( "encoding/json" "fmt" + "github.com/wailsapp/wails/v2/internal/frontend" ) @@ -13,7 +14,6 @@ type secureCallMessage struct { } func (d *Dispatcher) processSecureCallMessage(message string, sender frontend.Frontend) (string, error) { - var payload secureCallMessage err := json.Unmarshal([]byte(message[1:]), &payload) if err != nil { diff --git a/v2/internal/frontend/dispatcher/systemcalls.go b/v2/internal/frontend/dispatcher/systemcalls.go index b810d9beaf1..b090a416ef6 100644 --- a/v2/internal/frontend/dispatcher/systemcalls.go +++ b/v2/internal/frontend/dispatcher/systemcalls.go @@ -4,9 +4,10 @@ import ( "encoding/json" "errors" "fmt" - "github.com/wailsapp/wails/v2/pkg/runtime" "strings" + "github.com/wailsapp/wails/v2/pkg/runtime" + "github.com/wailsapp/wails/v2/internal/frontend" ) @@ -23,7 +24,6 @@ type size struct { } func (d *Dispatcher) processSystemCall(payload callMessage, sender frontend.Frontend) (interface{}, error) { - // Strip prefix name := strings.TrimPrefix(payload.Name, systemCallPrefix) @@ -64,5 +64,4 @@ func (d *Dispatcher) processSystemCall(payload callMessage, sender frontend.Fron default: return nil, fmt.Errorf("unknown systemcall message: %s", payload.Name) } - } diff --git a/v2/internal/frontend/frontend.go b/v2/internal/frontend/frontend.go index 27eb4a2e755..6b2ccbcae41 100644 --- a/v2/internal/frontend/frontend.go +++ b/v2/internal/frontend/frontend.go @@ -77,7 +77,7 @@ type MessageDialogOptions struct { } type Frontend interface { - Run(context.Context) error + Run(ctx context.Context) error RunMainLoop() ExecJS(js string) Hide() @@ -123,7 +123,7 @@ type Frontend interface { WindowClose() WindowPrint() - //Screen + // Screen ScreenGetAll() ([]Screen, error) // Menus diff --git a/v2/internal/frontend/runtime/events.go b/v2/internal/frontend/runtime/events.go index ac9d6299c3f..1f2e0a6e4b7 100644 --- a/v2/internal/frontend/runtime/events.go +++ b/v2/internal/frontend/runtime/events.go @@ -143,7 +143,7 @@ func (e *Events) notifyBackend(eventName string, data ...interface{}) { } // Do we have items to delete? - if itemsToDelete == true { + if itemsToDelete { // Create a new Listeners slice var newListeners []*eventListener diff --git a/v2/internal/fs/fs.go b/v2/internal/fs/fs.go index 5bfa71b1c0f..5662c020c01 100644 --- a/v2/internal/fs/fs.go +++ b/v2/internal/fs/fs.go @@ -2,6 +2,7 @@ package fs import ( "crypto/md5" + "encoding/hex" "fmt" "io" "io/fs" @@ -27,14 +28,14 @@ func RelativeToCwd(relativePath string) (string, error) { // Mkdir will create the given directory func Mkdir(dirname string) error { - return os.Mkdir(dirname, 0755) + return os.Mkdir(dirname, 0o755) } // MkDirs creates the given nested directories. // Returns error on failure func MkDirs(fullPath string, mode ...os.FileMode) error { var perms os.FileMode - perms = 0755 + perms = 0o755 if len(mode) == 1 { perms = mode[0] } @@ -111,7 +112,7 @@ func RelativePath(relativepath string, optionalpaths ...string) string { // I'm allowing this for 1 reason only: It's fatal if the path // supplied is wrong as it's only used internally in Wails. If we get // that path wrong, we should know about it immediately. The other reason is - // that it cuts down a ton of unnecassary error handling. + // that it cuts down a ton of unnecessary error handling. panic(err) } return result @@ -141,7 +142,7 @@ func MD5File(filename string) (string, error) { return "", err } - return fmt.Sprintf("%x", h.Sum(nil)), nil + return hex.EncodeToString(h.Sum(nil)), nil } // MustMD5File will call MD5File and abort the program on error @@ -157,7 +158,7 @@ func MustMD5File(filename string) string { // MustWriteString will attempt to write the given data to the given filename // It will abort the program in the event of a failure func MustWriteString(filename string, data string) { - err := os.WriteFile(filename, []byte(data), 0755) + err := os.WriteFile(filename, []byte(data), 0o755) if err != nil { fatal("Unable to write file", filename, ":", err.Error()) os.Exit(1) @@ -194,7 +195,6 @@ func GetSubdirectories(rootDir string) (*slicer.StringSlicer, error) { } func DirIsEmpty(dir string) (bool, error) { - // CREDIT: https://stackoverflow.com/a/30708914/8325411 f, err := os.Open(dir) if err != nil { @@ -284,7 +284,6 @@ func SetPermissions(dir string, perm os.FileMode) error { // Symlinks are ignored and skipped. // Credit: https://gist.github.com/r0l1/92462b38df26839a3ca324697c8cba04 func CopyDirExtended(src string, dst string, ignore []string) (err error) { - ignoreList := slicer.String(ignore) src = filepath.Clean(src) dst = filepath.Clean(dst) @@ -382,7 +381,6 @@ func FindPathToFile(fsys fs.FS, file string) (string, error) { // FindFileInParents searches for a file in the current directory and all parent directories. // Returns the absolute path to the file if found, otherwise an empty string func FindFileInParents(path string, filename string) string { - // Check for bad paths if _, err := os.Stat(path); err != nil { return "" diff --git a/v2/internal/github/github.go b/v2/internal/github/github.go index db3b70c5850..c16e1d9dda4 100644 --- a/v2/internal/github/github.go +++ b/v2/internal/github/github.go @@ -3,13 +3,14 @@ package github import ( "encoding/json" "fmt" - "github.com/charmbracelet/glamour" "io" "net/http" "net/url" "runtime" "sort" "strings" + + "github.com/charmbracelet/glamour" ) func GetReleaseNotes(tagVersion string, noColour bool) string { @@ -57,7 +58,6 @@ func GetReleaseNotes(tagVersion string, noColour bool) string { // GetVersionTags gets the list of tags on the Wails repo // It returns a list of sorted tags in descending order func GetVersionTags() ([]*SemanticVersion, error) { - result := []*SemanticVersion{} var err error @@ -97,7 +97,6 @@ func GetVersionTags() ([]*SemanticVersion, error) { // GetLatestStableRelease gets the latest stable release on GitHub func GetLatestStableRelease() (result *SemanticVersion, err error) { - tags, err := GetVersionTags() if err != nil { return nil, err @@ -114,7 +113,6 @@ func GetLatestStableRelease() (result *SemanticVersion, err error) { // GetLatestPreRelease gets the latest prerelease on GitHub func GetLatestPreRelease() (result *SemanticVersion, err error) { - tags, err := GetVersionTags() if err != nil { return nil, err diff --git a/v2/internal/go-common-file-dialog/cfd/CommonFileDialog_nonWindows.go b/v2/internal/go-common-file-dialog/cfd/CommonFileDialog_nonWindows.go index 3ab96985059..04c7cbcfef2 100644 --- a/v2/internal/go-common-file-dialog/cfd/CommonFileDialog_nonWindows.go +++ b/v2/internal/go-common-file-dialog/cfd/CommonFileDialog_nonWindows.go @@ -5,24 +5,24 @@ package cfd import "fmt" -var unsupportedError = fmt.Errorf("common file dialogs are only available on windows") +var errUnsupported = fmt.Errorf("common file dialogs are only available on windows") // TODO doc func NewOpenFileDialog(config DialogConfig) (OpenFileDialog, error) { - return nil, unsupportedError + return nil, errUnsupported } // TODO doc func NewOpenMultipleFilesDialog(config DialogConfig) (OpenMultipleFilesDialog, error) { - return nil, unsupportedError + return nil, errUnsupported } // TODO doc func NewSelectFolderDialog(config DialogConfig) (SelectFolderDialog, error) { - return nil, unsupportedError + return nil, errUnsupported } // TODO doc func NewSaveFileDialog(config DialogConfig) (SaveFileDialog, error) { - return nil, unsupportedError + return nil, errUnsupported } diff --git a/v2/internal/go-common-file-dialog/cfd/errors.go b/v2/internal/go-common-file-dialog/cfd/errors.go index c097c8eb2f4..6f21fedbfb1 100644 --- a/v2/internal/go-common-file-dialog/cfd/errors.go +++ b/v2/internal/go-common-file-dialog/cfd/errors.go @@ -2,6 +2,4 @@ package cfd import "errors" -var ( - ErrorCancelled = errors.New("cancelled by user") -) +var ErrCancelled = errors.New("cancelled by user") diff --git a/v2/internal/go-common-file-dialog/cfd/iFileOpenDialog.go b/v2/internal/go-common-file-dialog/cfd/iFileOpenDialog.go index 8c82cda2c6d..4c080c916d4 100644 --- a/v2/internal/go-common-file-dialog/cfd/iFileOpenDialog.go +++ b/v2/internal/go-common-file-dialog/cfd/iFileOpenDialog.go @@ -178,7 +178,7 @@ func (vtbl *iFileOpenDialogVtbl) getResultsStrings(objPtr unsafe.Pointer) ([]str return nil, err } if shellItemArray == nil { - return nil, ErrorCancelled + return nil, ErrCancelled } defer shellItemArray.vtbl.release(unsafe.Pointer(shellItemArray)) count, err := shellItemArray.vtbl.getCount(unsafe.Pointer(shellItemArray)) diff --git a/v2/internal/go-common-file-dialog/cfd/iShellItemArray.go b/v2/internal/go-common-file-dialog/cfd/iShellItemArray.go index 84f26fa2008..d904e72b21c 100644 --- a/v2/internal/go-common-file-dialog/cfd/iShellItemArray.go +++ b/v2/internal/go-common-file-dialog/cfd/iShellItemArray.go @@ -60,7 +60,7 @@ func (vtbl *iShellItemArrayVtbl) getItemAt(objPtr unsafe.Pointer, index uintptr) return "", err } if shellItem == nil { - return "", ErrorCancelled + return "", ErrCancelled } defer shellItem.vtbl.release(unsafe.Pointer(shellItem)) return shellItem.vtbl.getDisplayName(unsafe.Pointer(shellItem)) diff --git a/v2/internal/go-common-file-dialog/cfd/vtblCommonFunc.go b/v2/internal/go-common-file-dialog/cfd/vtblCommonFunc.go index a9210001017..929d5a2b7ac 100644 --- a/v2/internal/go-common-file-dialog/cfd/vtblCommonFunc.go +++ b/v2/internal/go-common-file-dialog/cfd/vtblCommonFunc.go @@ -179,7 +179,7 @@ func (vtbl *iFileDialogVtbl) getResultString(objPtr unsafe.Pointer) (string, err return "", err } if shellItem == nil { - return "", ErrorCancelled + return "", ErrCancelled } defer shellItem.vtbl.release(unsafe.Pointer(shellItem)) return shellItem.vtbl.getDisplayName(unsafe.Pointer(shellItem)) diff --git a/v2/internal/go-common-file-dialog/cfdutil/CFDUtil.go b/v2/internal/go-common-file-dialog/cfdutil/CFDUtil.go index bde52d7439a..655266bc3fb 100644 --- a/v2/internal/go-common-file-dialog/cfdutil/CFDUtil.go +++ b/v2/internal/go-common-file-dialog/cfdutil/CFDUtil.go @@ -10,7 +10,9 @@ func ShowOpenFileDialog(config cfd.DialogConfig) (string, error) { if err != nil { return "", err } - defer dialog.Release() + defer func() { + _ = dialog.Release() + }() return dialog.ShowAndGetResult() } @@ -20,7 +22,9 @@ func ShowOpenMultipleFilesDialog(config cfd.DialogConfig) ([]string, error) { if err != nil { return nil, err } - defer dialog.Release() + defer func() { + _ = dialog.Release() + }() return dialog.ShowAndGetResults() } @@ -30,7 +34,9 @@ func ShowPickFolderDialog(config cfd.DialogConfig) (string, error) { if err != nil { return "", err } - defer dialog.Release() + defer func() { + _ = dialog.Release() + }() return dialog.ShowAndGetResult() } @@ -40,6 +46,8 @@ func ShowSaveFileDialog(config cfd.DialogConfig) (string, error) { if err != nil { return "", err } - defer dialog.Release() + defer func() { + _ = dialog.Release() + }() return dialog.ShowAndGetResult() } diff --git a/v2/internal/gomod/gomod.go b/v2/internal/gomod/gomod.go index 8ab7e0b6654..c38e60f0bac 100644 --- a/v2/internal/gomod/gomod.go +++ b/v2/internal/gomod/gomod.go @@ -2,6 +2,7 @@ package gomod import ( "fmt" + "github.com/Masterminds/semver" "golang.org/x/mod/modfile" ) diff --git a/v2/internal/gomod/gomod_data_unix.go b/v2/internal/gomod/gomod_data_unix.go index f3a5e04c347..c6004f4862f 100644 --- a/v2/internal/gomod/gomod_data_unix.go +++ b/v2/internal/gomod/gomod_data_unix.go @@ -10,6 +10,7 @@ require github.com/wailsapp/wails/v2 v2.0.0-beta.7 //replace github.com/wailsapp/wails/v2 v2.0.0-beta.7 => /home/lea/wails/v2 ` + const basicUpdated string = `module changeme go 1.17 @@ -29,6 +30,7 @@ require ( //replace github.com/wailsapp/wails/v2 v2.0.0-beta.7 => /home/lea/wails/v2 ` + const multilineReplace = `module changeme go 1.17 @@ -98,6 +100,7 @@ require ( replace github.com/wailsapp/wails/v2 v2.0.0-beta.20 => /home/lea/wails/v2 ` + const multilineReplaceNoVersionUpdated = `module changeme go 1.17 @@ -108,6 +111,7 @@ require ( replace github.com/wailsapp/wails/v2 => /home/lea/wails/v2 ` + const multilineReplaceNoVersionBlockUpdated = `module changeme go 1.17 diff --git a/v2/internal/logger/custom_logger.go b/v2/internal/logger/custom_logger.go index 5e24aa0938d..51e07c0fc3d 100644 --- a/v2/internal/logger/custom_logger.go +++ b/v2/internal/logger/custom_logger.go @@ -86,7 +86,6 @@ func (l *customLogger) Warning(format string, args ...interface{}) { func (l *customLogger) Error(format string, args ...interface{}) { format = fmt.Sprintf("%s | %s", l.name, format) l.logger.Error(format, args...) - } // Fatal level logging. Works like Sprintf. diff --git a/v2/internal/logger/default_logger.go b/v2/internal/logger/default_logger.go index fe5c0538793..5c72ae20988 100644 --- a/v2/internal/logger/default_logger.go +++ b/v2/internal/logger/default_logger.go @@ -84,7 +84,6 @@ func (l *Logger) Info(format string, args ...interface{}) { if l.logLevel <= logger.INFO { l.output.Info(fmt.Sprintf(format, args...)) } - } // Warning level logging. Works like Sprintf. @@ -99,7 +98,6 @@ func (l *Logger) Error(format string, args ...interface{}) { if l.logLevel <= logger.ERROR { l.output.Error(fmt.Sprintf(format, args...)) } - } // Fatal level logging. Works like Sprintf. diff --git a/v2/internal/menumanager/applicationmenu.go b/v2/internal/menumanager/applicationmenu.go index 424834e5351..4446a00cbf7 100644 --- a/v2/internal/menumanager/applicationmenu.go +++ b/v2/internal/menumanager/applicationmenu.go @@ -3,7 +3,6 @@ package menumanager import "github.com/wailsapp/wails/v2/pkg/menu" func (m *Manager) SetApplicationMenu(applicationMenu *menu.Menu) error { - if applicationMenu == nil { return nil } @@ -38,7 +37,6 @@ func (m *Manager) UpdateApplicationMenu() (string, error) { } func (m *Manager) processApplicationMenu() error { - // Process the menu m.processedApplicationMenu = NewWailsMenu(m.applicationMenuItemMap, m.applicationMenu) m.processRadioGroups(m.processedApplicationMenu, m.applicationMenuItemMap) diff --git a/v2/internal/menumanager/contextmenu.go b/v2/internal/menumanager/contextmenu.go index 77c47891c00..f05bcdc49b8 100644 --- a/v2/internal/menumanager/contextmenu.go +++ b/v2/internal/menumanager/contextmenu.go @@ -23,7 +23,6 @@ func (t *ContextMenu) AsJSON() (string, error) { } func NewContextMenu(contextMenu *menu.ContextMenu) *ContextMenu { - result := &ContextMenu{ ID: contextMenu.ID, menu: contextMenu.Menu, @@ -37,7 +36,6 @@ func NewContextMenu(contextMenu *menu.ContextMenu) *ContextMenu { } func (m *Manager) AddContextMenu(contextMenu *menu.ContextMenu) { - newContextMenu := NewContextMenu(contextMenu) // Save the references diff --git a/v2/internal/menumanager/menuitemmap.go b/v2/internal/menumanager/menuitemmap.go index 790d5d06d6c..e4e291be6bb 100644 --- a/v2/internal/menumanager/menuitemmap.go +++ b/v2/internal/menumanager/menuitemmap.go @@ -2,8 +2,10 @@ package menumanager import ( "fmt" - "github.com/wailsapp/wails/v2/pkg/menu" + "strconv" "sync" + + "github.com/wailsapp/wails/v2/pkg/menu" ) // MenuItemMap holds a mapping between menuIDs and menu items @@ -48,14 +50,13 @@ func (m *MenuItemMap) Dump() { // GenerateMenuID returns a unique string ID for a menu item func (m *MenuItemMap) generateMenuID() string { m.menuIDCounterMutex.Lock() - result := fmt.Sprintf("%d", m.menuIDCounter) + result := strconv.FormatInt(m.menuIDCounter, 10) m.menuIDCounter++ m.menuIDCounterMutex.Unlock() return result } func (m *MenuItemMap) processMenuItem(item *menu.MenuItem) { - if item.SubMenu != nil { for _, submenuitem := range item.SubMenu.Items { m.processMenuItem(submenuitem) diff --git a/v2/internal/menumanager/menumanager.go b/v2/internal/menumanager/menumanager.go index ea793941591..0c6be0df234 100644 --- a/v2/internal/menumanager/menumanager.go +++ b/v2/internal/menumanager/menumanager.go @@ -2,11 +2,11 @@ package menumanager import ( "fmt" + "github.com/wailsapp/wails/v2/pkg/menu" ) type Manager struct { - // The application menu. applicationMenu *menu.Menu applicationMenuJSON string @@ -43,7 +43,6 @@ func (m *Manager) getMenuItemByID(menuMap *MenuItemMap, menuId string) *menu.Men } func (m *Manager) ProcessClick(menuID string, data string, menuType string, parentID string) error { - var menuItemMap *MenuItemMap switch menuType { @@ -93,7 +92,7 @@ func (m *Manager) ProcessClick(menuID string, data string, menuType string, pare // Create new Callback struct callbackData := &menu.CallbackData{ MenuItem: menuItem, - //ContextData: data, + // ContextData: data, } // Call back! diff --git a/v2/internal/menumanager/processedMenu.go b/v2/internal/menumanager/processedMenu.go index 8d886e19e89..0f2351846ab 100644 --- a/v2/internal/menumanager/processedMenu.go +++ b/v2/internal/menumanager/processedMenu.go @@ -2,6 +2,7 @@ package menumanager import ( "encoding/json" + "github.com/wailsapp/wails/v2/pkg/menu" "github.com/wailsapp/wails/v2/pkg/menu/keys" ) @@ -11,7 +12,7 @@ type ProcessedMenuItem struct { // Label is what appears as the menu text Label string `json:",omitempty"` // Role is a predefined menu type - //Role menu.Role `json:",omitempty"` + // Role menu.Role `json:",omitempty"` // Accelerator holds a representation of a key binding Accelerator *keys.Accelerator `json:",omitempty"` // Type of MenuItem, EG: Checkbox, Text, Separator, Radio, Submenu @@ -23,7 +24,7 @@ type ProcessedMenuItem struct { // Checked indicates if the item is selected (used by Checkbox and Radio types only) Checked bool `json:",omitempty"` // Submenu contains a list of menu items that will be shown as a submenu - //SubMenu []*MenuItem `json:"SubMenu,omitempty"` + // SubMenu []*MenuItem `json:"SubMenu,omitempty"` SubMenu *ProcessedMenu `json:",omitempty"` /* // Colour @@ -47,7 +48,6 @@ type ProcessedMenuItem struct { } func NewProcessedMenuItem(menuItemMap *MenuItemMap, menuItem *menu.MenuItem) *ProcessedMenuItem { - ID := menuItemMap.menuItemToIDMap[menuItem] // Parse ANSI text @@ -63,21 +63,21 @@ func NewProcessedMenuItem(menuItemMap *MenuItemMap, menuItem *menu.MenuItem) *Pr result := &ProcessedMenuItem{ ID: ID, Label: menuItem.Label, - //Role: menuItem.Role, + // Role: menuItem.Role, Accelerator: menuItem.Accelerator, Type: menuItem.Type, Disabled: menuItem.Disabled, Hidden: menuItem.Hidden, Checked: menuItem.Checked, SubMenu: nil, - //BackgroundColour: menuItem.BackgroundColour, - //FontSize: menuItem.FontSize, - //FontName: menuItem.FontName, - //Image: menuItem.Image, - //MacTemplateImage: menuItem.MacTemplateImage, - //MacAlternate: menuItem.MacAlternate, - //Tooltip: menuItem.Tooltip, - //StyledLabel: styledLabel, + // BackgroundColour: menuItem.BackgroundColour, + // FontSize: menuItem.FontSize, + // FontName: menuItem.FontName, + // Image: menuItem.Image, + // MacTemplateImage: menuItem.MacTemplateImage, + // MacAlternate: menuItem.MacAlternate, + // Tooltip: menuItem.Tooltip, + // StyledLabel: styledLabel, } if menuItem.SubMenu != nil { @@ -92,7 +92,6 @@ type ProcessedMenu struct { } func NewProcessedMenu(menuItemMap *MenuItemMap, menu *menu.Menu) *ProcessedMenu { - result := &ProcessedMenu{} if menu != nil { for _, item := range menu.Items { @@ -131,7 +130,6 @@ func NewWailsMenu(menuItemMap *MenuItemMap, menu *menu.Menu) *WailsMenu { } func (w *WailsMenu) AsJSON() (string, error) { - menuAsJSON, err := json.Marshal(w) if err != nil { return "", err @@ -150,7 +148,6 @@ func (w *WailsMenu) processRadioGroups() { } func (w *WailsMenu) processMenuItem(item *ProcessedMenuItem) { - switch item.Type { // We need to recurse submenus @@ -172,7 +169,6 @@ func (w *WailsMenu) processMenuItem(item *ProcessedMenuItem) { } func (w *WailsMenu) finaliseRadioGroup() { - // If we were processing a radio group, fix up the references if len(w.currentRadioGroup) > 0 { diff --git a/v2/internal/menumanager/traymenu.go b/v2/internal/menumanager/traymenu.go index aed5b05ace6..5efc4a86192 100644 --- a/v2/internal/menumanager/traymenu.go +++ b/v2/internal/menumanager/traymenu.go @@ -13,8 +13,10 @@ import ( "github.com/wailsapp/wails/v2/pkg/menu" ) -var trayMenuID int -var trayMenuIDMutex sync.Mutex +var ( + trayMenuID int + trayMenuIDMutex sync.Mutex +) func generateTrayID() string { var idStr string @@ -51,7 +53,6 @@ func (t *TrayMenu) AsJSON() (string, error) { } func NewTrayMenu(trayMenu *menu.TrayMenu) *TrayMenu { - // Parse ANSI text var styledLabel []*ansi.StyledText tempLabel := trayMenu.Label @@ -205,7 +206,6 @@ func (m *Manager) UpdateTrayMenuLabel(trayMenu *menu.TrayMenu) (string, error) { } return string(data), nil - } func (m *Manager) GetContextMenus() ([]string, error) { diff --git a/v2/internal/process/process.go b/v2/internal/process/process.go index 6d497ed8e67..18c9f45da36 100644 --- a/v2/internal/process/process.go +++ b/v2/internal/process/process.go @@ -25,7 +25,6 @@ func NewProcess(cmd string, args ...string) *Process { // Start the process func (p *Process) Start(exitCodeChannel chan int) error { - err := p.cmd.Start() if err != nil { return err diff --git a/v2/internal/project/project.go b/v2/internal/project/project.go index d42977fee51..34cbe88da59 100644 --- a/v2/internal/project/project.go +++ b/v2/internal/project/project.go @@ -12,7 +12,6 @@ import ( // Project holds the data related to a Wails project type Project struct { - /*** Application Data ***/ Name string `json:"name"` AssetDirectory string `json:"assetdir,omitempty"` @@ -144,11 +143,10 @@ func (p *Project) Save() error { if err != nil { return err } - return os.WriteFile(p.filename, data, 0755) + return os.WriteFile(p.filename, data, 0o755) } func (p *Project) setDefaults() { - if p.Path == "" { p.Path = lo.Must(os.Getwd()) } diff --git a/v2/internal/s/s.go b/v2/internal/s/s.go index 86536e24c0c..f363d6541fe 100644 --- a/v2/internal/s/s.go +++ b/v2/internal/s/s.go @@ -2,6 +2,7 @@ package s import ( "crypto/md5" + "encoding/hex" "fmt" "io" "io/ioutil" @@ -74,9 +75,10 @@ func CD(dir string) { checkError(err) log("CD %s [%s]", dir, CWD()) } + func MKDIR(path string, mode ...os.FileMode) { var perms os.FileMode - perms = 0755 + perms = 0o755 if len(mode) == 1 { perms = mode[0] } @@ -88,7 +90,7 @@ func MKDIR(path string, mode ...os.FileMode) { // ENDIR ensures that the path gets created if it doesn't exist func ENDIR(path string, mode ...os.FileMode) { var perms os.FileMode - perms = 0755 + perms = 0o755 if len(mode) == 1 { perms = mode[0] } @@ -210,17 +212,13 @@ func ISDIR(path string) bool { // ISDIREMPTY returns true if the given directory is empty func ISDIREMPTY(dir string) bool { - // CREDIT: https://stackoverflow.com/a/30708914/8325411 f, err := os.Open(dir) checkError(err) defer closefile(f) _, err = f.Readdirnames(1) // Or f.Readdir(1) - if err == io.EOF { - return true - } - return false + return err == io.EOF } // ISFILE returns true if the given file exists @@ -270,7 +268,7 @@ func LOADSTRING(filename string) string { // SAVEBYTES will create a file with the given string func SAVEBYTES(filename string, data []byte) { log("SAVEBYTES %s", filename) - err := os.WriteFile(filename, data, 0755) + err := os.WriteFile(filename, data, 0o755) checkError(err) } @@ -297,7 +295,7 @@ func MD5FILE(filename string) string { _, err = io.Copy(h, f) checkError(err) - return fmt.Sprintf("%x", h.Sum(nil)) + return hex.EncodeToString(h.Sum(nil)) } // Sub is the substitution type diff --git a/v2/internal/shell/shell.go b/v2/internal/shell/shell.go index badea2b3941..349e27bff7e 100644 --- a/v2/internal/shell/shell.go +++ b/v2/internal/shell/shell.go @@ -42,6 +42,7 @@ func (c *Command) Run() error { func (c *Command) Stdout() string { return c.stdo.String() } + func (c *Command) Stderr() string { return c.stde.String() } @@ -93,8 +94,5 @@ func RunCommandVerbose(directory string, command string, args ...string) error { // CommandExists returns true if the given command can be found on the shell func CommandExists(name string) bool { _, err := exec.LookPath(name) - if err != nil { - return false - } - return true + return err == nil } diff --git a/v2/internal/signal/signal.go b/v2/internal/signal/signal.go index 96e10bee683..fa797453f85 100644 --- a/v2/internal/signal/signal.go +++ b/v2/internal/signal/signal.go @@ -9,8 +9,10 @@ import ( var signalChannel = make(chan os.Signal, 2) -var callbacks []func() -var lock sync.Mutex +var ( + callbacks []func() + lock sync.Mutex +) func OnShutdown(callback func()) { lock.Lock() @@ -20,20 +22,17 @@ func OnShutdown(callback func()) { // Start the Signal Manager func Start() { - // Hook into interrupts gosignal.Notify(signalChannel, os.Interrupt, syscall.SIGTERM, syscall.SIGINT) // Spin off signal listener and wait for either a cancellation // or signal go func() { - select { - case <-signalChannel: - println("") - println("Ctrl+C detected. Shutting down...") - for _, callback := range callbacks { - callback() - } + <-signalChannel + println("") + println("Ctrl+C detected. Shutting down...") + for _, callback := range callbacks { + callback() } }() } diff --git a/v2/internal/staticanalysis/staticanalysis.go b/v2/internal/staticanalysis/staticanalysis.go index 0d8fb92d3f0..d5cc8c6e522 100644 --- a/v2/internal/staticanalysis/staticanalysis.go +++ b/v2/internal/staticanalysis/staticanalysis.go @@ -2,9 +2,10 @@ package staticanalysis import ( "go/ast" - "golang.org/x/tools/go/packages" "path/filepath" "strings" + + "golang.org/x/tools/go/packages" ) type EmbedDetails struct { diff --git a/v2/internal/system/packagemanager/pm.go b/v2/internal/system/packagemanager/pm.go index dfd39429984..bba45cd05ea 100644 --- a/v2/internal/system/packagemanager/pm.go +++ b/v2/internal/system/packagemanager/pm.go @@ -16,9 +16,9 @@ type packagemap = map[string][]*Package type PackageManager interface { Name() string Packages() packagemap - PackageInstalled(*Package) (bool, error) - PackageAvailable(*Package) (bool, error) - InstallCommand(*Package) string + PackageInstalled(pkg *Package) (bool, error) + PackageAvailable(pkg *Package) (bool, error) + InstallCommand(pkg *Package) string } // Dependency represents a system package that we require @@ -37,7 +37,6 @@ type DependencyList []*Dependency // InstallAllRequiredCommand returns the command you need to use to install all required dependencies func (d DependencyList) InstallAllRequiredCommand() string { - result := "" for _, dependency := range d { if !dependency.Installed && !dependency.Optional { @@ -50,7 +49,6 @@ func (d DependencyList) InstallAllRequiredCommand() string { // InstallAllOptionalCommand returns the command you need to use to install all optional dependencies func (d DependencyList) InstallAllOptionalCommand() string { - result := "" for _, dependency := range d { if !dependency.Installed && dependency.Optional { diff --git a/v2/internal/system/system.go b/v2/internal/system/system.go index a633989efc5..67453538fc0 100644 --- a/v2/internal/system/system.go +++ b/v2/internal/system/system.go @@ -9,12 +9,10 @@ import ( "github.com/wailsapp/wails/v2/internal/system/packagemanager" ) -var ( - IsAppleSilicon bool -) +var IsAppleSilicon bool // Info holds information about the current operating system, -// package manager and required dependancies +// package manager and required dependencies type Info struct { OS *operatingsystem.OS PM packagemanager.PackageManager @@ -23,7 +21,7 @@ type Info struct { // GetInfo scans the system for operating system details, // the system package manager and the status of required -// dependancies. +// dependencies. func GetInfo() (*Info, error) { var result Info err := result.discover() @@ -34,7 +32,6 @@ func GetInfo() (*Info, error) { } func checkNodejs() *packagemanager.Dependency { - // Check for Nodejs output, err := exec.Command("node", "-v").Output() installed := true @@ -58,7 +55,6 @@ func checkNodejs() *packagemanager.Dependency { } func checkNPM() *packagemanager.Dependency { - // Check for npm output, err := exec.Command("npm", "-version").Output() installed := true @@ -80,7 +76,6 @@ func checkNPM() *packagemanager.Dependency { } func checkUPX() *packagemanager.Dependency { - // Check for npm output, err := exec.Command("upx", "-V").Output() installed := true @@ -102,7 +97,6 @@ func checkUPX() *packagemanager.Dependency { } func checkNSIS() *packagemanager.Dependency { - // Check for nsis installer output, err := exec.Command("makensis", "-VERSION").Output() installed := true @@ -141,7 +135,6 @@ func checkLibrary(name string) func() *packagemanager.Dependency { } func checkDocker() *packagemanager.Dependency { - // Check for npm output, err := exec.Command("docker", "version").Output() installed := true diff --git a/v2/internal/typescriptify/typescriptify.go b/v2/internal/typescriptify/typescriptify.go index b14059a51af..bb72e6fb8b5 100644 --- a/v2/internal/typescriptify/typescriptify.go +++ b/v2/internal/typescriptify/typescriptify.go @@ -156,10 +156,10 @@ func (t *TypeScriptify) deepFields(typeOf reflect.Type) []reflect.StructField { kind := f.Type.Kind() isPointer := kind == reflect.Ptr && f.Type.Elem().Kind() == reflect.Struct if f.Anonymous && kind == reflect.Struct { - //fmt.Println(v.Interface()) + // fmt.Println(v.Interface()) fields = append(fields, t.deepFields(f.Type)...) } else if f.Anonymous && isPointer { - //fmt.Println(v.Interface()) + // fmt.Println(v.Interface()) fields = append(fields, t.deepFields(f.Type.Elem())...) } else { // Check we have a json tag @@ -439,7 +439,7 @@ func (t TypeScriptify) backup(fileName string) error { backupFn = path.Join(t.BackupDir, backupFn) } - return ioutil.WriteFile(backupFn, bytes, os.FileMode(0700)) + return ioutil.WriteFile(backupFn, bytes, os.FileMode(0o700)) } func (t TypeScriptify) ConvertToFile(fileName string, packageName string) error { @@ -697,7 +697,7 @@ func (t *TypeScriptify) convertType(depth int, typeOf reflect.Type, customCode m builder.AddMapField(jsonFieldName, field) } else if field.Type.Kind() == reflect.Slice || field.Type.Kind() == reflect.Array { // Slice: - if field.Type.Elem().Kind() == reflect.Ptr { //extract ptr type + if field.Type.Elem().Kind() == reflect.Ptr { // extract ptr type field.Type = field.Type.Elem() } @@ -935,6 +935,6 @@ func typeClashWithReservedKeyword(input string) bool { func warnAboutTypesClash(entity string) { // TODO: Refactor logging l := log.New(os.Stderr, "", 0) - l.Println(fmt.Sprintf("Usage of reserved keyword found and not supported: %s", entity)) + l.Printf("Usage of reserved keyword found and not supported: %s", entity) log.Println("Please rename returned type or consider adding bindings config to your wails.json") } diff --git a/v2/internal/webview2runtime/webview2installer.go b/v2/internal/webview2runtime/webview2installer.go index a2a2922dc20..3645dae02c3 100644 --- a/v2/internal/webview2runtime/webview2installer.go +++ b/v2/internal/webview2runtime/webview2installer.go @@ -11,7 +11,7 @@ var setupexe []byte // WriteInstallerToFile writes the installer file to the given file. func WriteInstallerToFile(targetFile string) error { - return os.WriteFile(targetFile, setupexe, 0755) + return os.WriteFile(targetFile, setupexe, 0o755) } // WriteInstaller writes the installer exe file to the given directory and returns the path to it. diff --git a/v2/pkg/application/application.go b/v2/pkg/application/application.go index 8d8d72ef6ab..8ba5869699a 100644 --- a/v2/pkg/application/application.go +++ b/v2/pkg/application/application.go @@ -2,11 +2,12 @@ package application import ( "context" + "sync" + "github.com/wailsapp/wails/v2/internal/app" "github.com/wailsapp/wails/v2/internal/signal" "github.com/wailsapp/wails/v2/pkg/menu" "github.com/wailsapp/wails/v2/pkg/options" - "sync" ) // Application is the main Wails application @@ -86,7 +87,6 @@ func (a *Application) Bind(boundStruct any) { } func (a *Application) On(eventType EventType, callback func()) { - c := func(ctx context.Context) { callback() } diff --git a/v2/pkg/assetserver/assethandler.go b/v2/pkg/assetserver/assethandler.go index c85bf81e65d..76d41147816 100644 --- a/v2/pkg/assetserver/assethandler.go +++ b/v2/pkg/assetserver/assethandler.go @@ -10,6 +10,7 @@ import ( "net/http" "os" "path" + "strconv" "strings" "github.com/wailsapp/wails/v2/pkg/options/assetserver" @@ -37,7 +38,6 @@ type assetHandler struct { } func NewAssetHandler(options assetserver.Options, log Logger) (http.Handler, error) { - vfs := options.Assets if vfs != nil { if _, err := vfs.Open("."); err != nil { @@ -178,7 +178,8 @@ func (d *assetHandler) serveFSFile(rw http.ResponseWriter, req *http.Request, fi return nil } - rw.Header().Set(HeaderContentLength, fmt.Sprintf("%d", statInfo.Size())) + size := strconv.FormatInt(statInfo.Size(), 10) + rw.Header().Set(HeaderContentLength, size) // Write the first 512 bytes used for MimeType sniffing _, err = io.Copy(rw, bytes.NewReader(buf[:n])) diff --git a/v2/pkg/assetserver/assetserver.go b/v2/pkg/assetserver/assetserver.go index 7fd6508ea10..54fe5d0636f 100644 --- a/v2/pkg/assetserver/assetserver.go +++ b/v2/pkg/assetserver/assetserver.go @@ -123,10 +123,8 @@ func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) { path := req.URL.Path if path == runtimeJSPath { d.writeBlob(rw, path, d.runtimeJS) - } else if path == runtimePath && d.runtimeHandler != nil { d.runtimeHandler.HandleRuntimeCall(rw, req) - } else if path == ipcJSPath { content := d.runtime.DesktopIPC() if d.ipcJS != nil { @@ -136,7 +134,6 @@ func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) { } else if script, ok := d.pluginScripts[path]; ok { d.writeBlob(rw, path, []byte(script)) - } else if d.isRuntimeInjectionMatch(path) { recorder := &bodyRecorder{ ResponseWriter: rw, @@ -150,7 +147,8 @@ func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) { } return strings.Contains(h.Get(HeaderContentType), "text/html") - }} + }, + } handler.ServeHTTP(recorder, req) diff --git a/v2/pkg/assetserver/assetserver_webview.go b/v2/pkg/assetserver/assetserver_webview.go index 575c81bb1d1..dfab1f2d78d 100644 --- a/v2/pkg/assetserver/assetserver_webview.go +++ b/v2/pkg/assetserver/assetserver_webview.go @@ -133,7 +133,8 @@ func (d *AssetServer) processWebViewRequestInternal(r webview.Request) { if req.ContentLength == 0 { req.ContentLength, _ = strconv.ParseInt(req.Header.Get(HeaderContentLength), 10, 64) } else { - req.Header.Set(HeaderContentLength, fmt.Sprintf("%d", req.ContentLength)) + size := strconv.FormatInt(req.ContentLength, 10) + req.Header.Set(HeaderContentLength, size) } if host := req.Header.Get(HeaderHost); host != "" { diff --git a/v2/pkg/assetserver/common.go b/v2/pkg/assetserver/common.go index 01e51f2be82..57934e08e14 100644 --- a/v2/pkg/assetserver/common.go +++ b/v2/pkg/assetserver/common.go @@ -3,9 +3,9 @@ package assetserver import ( "bytes" "errors" - "fmt" "io" "net/http" + "strconv" "strings" "github.com/wailsapp/wails/v2/pkg/options" @@ -44,7 +44,7 @@ const ( func serveFile(rw http.ResponseWriter, filename string, blob []byte) error { header := rw.Header() - header.Set(HeaderContentLength, fmt.Sprintf("%d", len(blob))) + header.Set(HeaderContentLength, strconv.Itoa(len(blob))) if mimeType := header.Get(HeaderContentType); mimeType == "" { mimeType = GetMimetype(filename, blob) header.Set(HeaderContentType, mimeType) diff --git a/v2/pkg/assetserver/webview/request_darwin.go b/v2/pkg/assetserver/webview/request_darwin.go index f0e85780b3d..c44e5f1969d 100644 --- a/v2/pkg/assetserver/webview/request_darwin.go +++ b/v2/pkg/assetserver/webview/request_darwin.go @@ -197,7 +197,10 @@ func (r *request) Close() error { if r.body != nil { err = r.body.Close() } - r.Response().Finish() + err = r.Response().Finish() + if err != nil { + return err + } C.URLSchemeTaskRelease(r.task) return err } diff --git a/v2/pkg/buildassets/buildassets.go b/v2/pkg/buildassets/buildassets.go index 26401745db3..778d97fbfd5 100644 --- a/v2/pkg/buildassets/buildassets.go +++ b/v2/pkg/buildassets/buildassets.go @@ -128,12 +128,12 @@ func writeFileSystemFile(projectData *project.Project, file string, content []by targetPath := GetLocalPath(projectData, file) if dir := filepath.Dir(targetPath); !fs.DirExists(dir) { - if err := fs.MkDirs(dir, 0755); err != nil { + if err := fs.MkDirs(dir, 0o755); err != nil { return fmt.Errorf("Unable to create directory: %w", err) } } - if err := os.WriteFile(targetPath, content, 0644); err != nil { + if err := os.WriteFile(targetPath, content, 0o644); err != nil { return err } return nil diff --git a/v2/pkg/commands/bindings/bindings.go b/v2/pkg/commands/bindings/bindings.go index 71c1747b736..1432acee11c 100644 --- a/v2/pkg/commands/bindings/bindings.go +++ b/v2/pkg/commands/bindings/bindings.go @@ -26,7 +26,6 @@ type Options struct { // GenerateBindings generates bindings for the Wails project in the given ProjectDirectory. // If no project directory is given then the current working directory is used. func GenerateBindings(options Options) (string, error) { - filename, _ := lo.Coalesce(options.Filename, "wailsbindings") if runtime.GOOS == "windows" { filename += ".exe" diff --git a/v2/pkg/commands/build/base.go b/v2/pkg/commands/build/base.go index abfbafff51c..6595aff0f6b 100644 --- a/v2/pkg/commands/build/base.go +++ b/v2/pkg/commands/build/base.go @@ -74,7 +74,6 @@ func (b *BaseBuilder) convertFileToIntegerString(filename string) (string, error } func (b *BaseBuilder) convertByteSliceToIntegerString(data []byte) string { - // Create string builder var result strings.Builder @@ -85,8 +84,7 @@ func (b *BaseBuilder) convertByteSliceToIntegerString(data []byte) string { result.WriteString(fmt.Sprintf("%v,", data[i])) } - result.WriteString(fmt.Sprintf("%v", data[len(data)-1])) - + result.WriteString(strconv.FormatUint(uint64(data[len(data)-1]), 10)) } return result.String() @@ -94,10 +92,8 @@ func (b *BaseBuilder) convertByteSliceToIntegerString(data []byte) string { // CleanUp does post-build housekeeping func (b *BaseBuilder) CleanUp() { - // Delete all the files b.filesToDelete.Each(func(filename string) { - // if file doesn't exist, ignore if !b.fileExists(filename) { return @@ -106,7 +102,6 @@ func (b *BaseBuilder) CleanUp() { // Delete file. We ignore errors because these files will be overwritten // by the next build anyway. _ = os.Remove(filename) - }) } @@ -159,7 +154,6 @@ func (b *BaseBuilder) OutputFilename(options *Options) string { // CompileProject compiles the project func (b *BaseBuilder) CompileProject(options *Options) error { - // Check if the runtime wrapper exists err := generateRuntimeWrapper(options) if err != nil { @@ -402,7 +396,7 @@ Please reinstall by doing the following: return nil } - var args = []string{"--best", "--no-color", "--no-progress", options.CompiledBinary} + args := []string{"--best", "--no-color", "--no-progress", options.CompiledBinary} if options.CompressFlags != "" { args = strings.Split(options.CompressFlags, " ") @@ -426,7 +420,6 @@ Please reinstall by doing the following: } func generateRuntimeWrapper(options *Options) error { - if options.WailsJSDir == "" { cwd, err := os.Getwd() if err != nil { @@ -452,7 +445,6 @@ func (b *BaseBuilder) NpmInstall(sourceDir string, verbose bool) error { // NpmInstallUsingCommand runs the given install command in the specified npm project directory func (b *BaseBuilder) NpmInstallUsingCommand(sourceDir string, installCommand string, verbose bool) error { - packageJSON := filepath.Join(sourceDir, "package.json") // Check package.json exists @@ -492,7 +484,7 @@ func (b *BaseBuilder) NpmInstallUsingCommand(sourceDir string, installCommand st } // Shortcut installation - if install == false { + if !install { if verbose { pterm.Println("Skipping npm install") } @@ -549,7 +541,6 @@ func (b *BaseBuilder) NpmRunWithEnvironment(projectDir, buildTarget string, verb // BuildFrontend executes the `npm build` command for the frontend directory func (b *BaseBuilder) BuildFrontend(outputLogger *clilogger.CLILogger) error { - verbose := b.options.Verbosity == VERBOSE frontendDir := b.projectData.GetFrontendDir() diff --git a/v2/pkg/commands/build/build.go b/v2/pkg/commands/build/build.go index a29840b1bde..2223bf575f0 100644 --- a/v2/pkg/commands/build/build.go +++ b/v2/pkg/commands/build/build.go @@ -73,7 +73,6 @@ type Options struct { // Build the project! func Build(options *Options) (string, error) { - // Extract logger outputLogger := options.Logger @@ -171,7 +170,7 @@ func CreateEmbedDirectories(cwd string, buildOptions *Options) error { for _, embedDetail := range embedDetails { fullPath := embedDetail.GetFullPath() if _, err := os.Stat(fullPath); os.IsNotExist(err) { - err := os.MkdirAll(fullPath, 0755) + err := os.MkdirAll(fullPath, 0o755) if err != nil { return err } @@ -184,7 +183,6 @@ func CreateEmbedDirectories(cwd string, buildOptions *Options) error { } return nil - } func fatal(message string) { @@ -213,7 +211,6 @@ func printBulletPoint(text string, args ...any) { } func GenerateBindings(buildOptions *Options) error { - obfuscated := buildOptions.Obfuscated if obfuscated { printBulletPoint("Generating obfuscated bindings: ") @@ -371,7 +368,6 @@ func execPostBuildHook(outputLogger *clilogger.CLILogger, options *Options, hook } return executeBuildHook(outputLogger, options, hookIdentifier, argReplacements, postBuildHook, "post") - } func executeBuildHook(_ *clilogger.CLILogger, options *Options, hookIdentifier string, argReplacements map[string]string, buildHook string, hookName string) error { @@ -407,7 +403,6 @@ func executeBuildHook(_ *clilogger.CLILogger, options *Options, hookIdentifier s if options.Verbosity == VERBOSE { pterm.Info.Println(strings.Join(args, " ")) - } if !fs.DirExists(options.BinDirectory) { diff --git a/v2/pkg/commands/build/builder.go b/v2/pkg/commands/build/builder.go index 4840341c031..6a220c530bd 100644 --- a/v2/pkg/commands/build/builder.go +++ b/v2/pkg/commands/build/builder.go @@ -8,8 +8,8 @@ import ( // Builder defines a builder that can build Wails applications type Builder interface { SetProjectData(projectData *project.Project) - BuildFrontend(*clilogger.CLILogger) error - CompileProject(*Options) error - OutputFilename(*Options) string + BuildFrontend(logger *clilogger.CLILogger) error + CompileProject(options *Options) error + OutputFilename(options *Options) string CleanUp() } diff --git a/v2/pkg/commands/build/nsis_installer.go b/v2/pkg/commands/build/nsis_installer.go index 11f1407a388..820df2d1d36 100644 --- a/v2/pkg/commands/build/nsis_installer.go +++ b/v2/pkg/commands/build/nsis_installer.go @@ -41,7 +41,7 @@ func GenerateNSISInstaller(options *Options, amd64Binary string, arm64Binary str // Write the WebView2 SetupFile webviewSetup := buildassets.GetLocalPath(options.ProjectData, path.Join(nsisFolder, nsisWebView2SetupFile)) if dir := filepath.Dir(webviewSetup); !fs.DirExists(dir) { - if err := fs.MkDirs(dir, 0755); err != nil { + if err := fs.MkDirs(dir, 0o755); err != nil { return err } } @@ -92,7 +92,7 @@ func makeNSIS(options *Options, installerKind string, amd64Binary string, arm64B outputLogger := options.Logger outputLogger.Print(" - Building '%s' installer: ", installerKind) - var args = []string{} + args := []string{} if amd64Binary != "" { args = append(args, "-DARG_WAILS_AMD64_BINARY="+amd64Binary) } diff --git a/v2/pkg/commands/build/packager.go b/v2/pkg/commands/build/packager.go index beca47c5bea..54aea2e674e 100644 --- a/v2/pkg/commands/build/packager.go +++ b/v2/pkg/commands/build/packager.go @@ -3,13 +3,14 @@ package build import ( "bytes" "fmt" + "image" + "os" + "path/filepath" + "github.com/leaanthony/winicon" "github.com/tc-hib/winres" "github.com/tc-hib/winres/version" "github.com/wailsapp/wails/v2/internal/project" - "image" - "os" - "path/filepath" "github.com/jackmordaunt/icns" "github.com/pkg/errors" @@ -20,7 +21,6 @@ import ( // PackageProject packages the application func packageProject(options *Options, platform string) error { - var err error switch platform { case "darwin": @@ -42,7 +42,6 @@ func packageProject(options *Options, platform string) error { // cleanBinDirectory will remove an existing bin directory and recreate it func cleanBinDirectory(options *Options) error { - buildDirectory := options.BinDirectory // Clear out old builds @@ -54,7 +53,7 @@ func cleanBinDirectory(options *Options) error { } // Create clean directory - err := os.MkdirAll(buildDirectory, 0700) + err := os.MkdirAll(buildDirectory, 0o700) if err != nil { return err } @@ -63,7 +62,6 @@ func cleanBinDirectory(options *Options) error { } func packageApplicationForDarwin(options *Options) error { - var err error // Create directory structure @@ -74,12 +72,12 @@ func packageApplicationForDarwin(options *Options) error { contentsDirectory := filepath.Join(options.BinDirectory, bundlename, "/Contents") exeDir := filepath.Join(contentsDirectory, "/MacOS") - err = fs.MkDirs(exeDir, 0755) + err = fs.MkDirs(exeDir, 0o755) if err != nil { return err } resourceDir := filepath.Join(contentsDirectory, "/Resources") - err = fs.MkDirs(resourceDir, 0755) + err = fs.MkDirs(resourceDir, 0o755) if err != nil { return err } @@ -116,7 +114,6 @@ func packageApplicationForDarwin(options *Options) error { } func processPList(options *Options, contentsDirectory string) error { - sourcePList := "Info.plist" if options.Mode == Dev { // Use Info.dev.plist if using build mode @@ -130,7 +127,7 @@ func processPList(options *Options, contentsDirectory string) error { } targetFile := filepath.Join(contentsDirectory, "Info.plist") - return os.WriteFile(targetFile, content, 0644) + return os.WriteFile(targetFile, content, 0o644) } func processDarwinIcon(projectData *project.Project, iconName string, resourceDir string, destIconName string) (err error) { @@ -152,7 +149,6 @@ func processDarwinIcon(projectData *project.Project, iconName string, resourceDi dest, err := os.Create(tgtBundle) if err != nil { return err - } defer func() { err = dest.Close() @@ -206,12 +202,12 @@ func generateIcoFile(options *Options, iconName string, destIconName string) err icoFile := buildassets.GetLocalPath(options.ProjectData, "windows/"+destIconName+".ico") if !fs.FileExists(icoFile) { if dir := filepath.Dir(icoFile); !fs.DirExists(dir) { - if err := fs.MkDirs(dir, 0755); err != nil { + if err := fs.MkDirs(dir, 0o755); err != nil { return err } } - output, err := os.OpenFile(icoFile, os.O_CREATE|os.O_WRONLY, 0644) + output, err := os.OpenFile(icoFile, os.O_CREATE|os.O_WRONLY, 0o644) if err != nil { return err } @@ -226,13 +222,12 @@ func generateIcoFile(options *Options, iconName string, destIconName string) err } func compileResources(options *Options) error { - currentDir, err := os.Getwd() if err != nil { return err } defer func() { - os.Chdir(currentDir) + _ = os.Chdir(currentDir) }() windowsDir := filepath.Join(options.ProjectData.GetBuildDir(), "windows") err = os.Chdir(windowsDir) diff --git a/v2/pkg/logger/filelogger.go b/v2/pkg/logger/filelogger.go index 5cf68fc1a55..954c46f599b 100644 --- a/v2/pkg/logger/filelogger.go +++ b/v2/pkg/logger/filelogger.go @@ -19,7 +19,7 @@ func NewFileLogger(filename string) Logger { // Print works like Sprintf. func (l *FileLogger) Print(message string) { - f, err := os.OpenFile(l.filename, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644) + f, err := os.OpenFile(l.filename, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0o644) if err != nil { log.Fatal(err) } diff --git a/v2/pkg/mac/login_darwin.go b/v2/pkg/mac/login_darwin.go index 2ff49be83b5..b2390e3059d 100644 --- a/v2/pkg/mac/login_darwin.go +++ b/v2/pkg/mac/login_darwin.go @@ -33,7 +33,7 @@ func StartAtLogin(enabled bool) error { } _, stde, err := shell.RunCommand("/tmp", "osascript", "-e", command) if err != nil { - errors.Wrap(err, stde) + return errors.Wrap(err, stde) } return nil } diff --git a/v2/pkg/mac/notification_darwin.go b/v2/pkg/mac/notification_darwin.go index a06ecb53ad7..b7743df97d4 100644 --- a/v2/pkg/mac/notification_darwin.go +++ b/v2/pkg/mac/notification_darwin.go @@ -24,7 +24,7 @@ func ShowNotification(title string, subtitle string, message string, sound strin } _, stde, err := shell.RunCommand("/tmp", "osascript", "-e", command) if err != nil { - errors.Wrap(err, stde) + return errors.Wrap(err, stde) } return nil } diff --git a/v2/pkg/menu/callback.go b/v2/pkg/menu/callback.go index fe616036147..a02664ac099 100644 --- a/v2/pkg/menu/callback.go +++ b/v2/pkg/menu/callback.go @@ -2,7 +2,7 @@ package menu type CallbackData struct { MenuItem *MenuItem - //ContextData string + // ContextData string } type Callback func(*CallbackData) diff --git a/v2/pkg/menu/colours/colours.go b/v2/pkg/menu/colours/colours.go index 28564a09e24..5fb74eabd0b 100644 --- a/v2/pkg/menu/colours/colours.go +++ b/v2/pkg/menu/colours/colours.go @@ -36,7 +36,6 @@ type InputCol struct { var Template string func main() { - var Cols []InputCol resp, err := http.Get("https://jonasjacek.github.io/colors/data.json") @@ -62,5 +61,8 @@ func main() { if err != nil { log.Fatal(err) } - os.WriteFile(filepath.Join("..", "cols.go"), buffer.Bytes(), 0755) + err = os.WriteFile(filepath.Join("..", "cols.go"), buffer.Bytes(), 0o755) + if err != nil { + log.Fatal(err) + } } diff --git a/v2/pkg/menu/keys/keys.go b/v2/pkg/menu/keys/keys.go index fa8027a3347..961edab2d8e 100644 --- a/v2/pkg/menu/keys/keys.go +++ b/v2/pkg/menu/keys/keys.go @@ -16,7 +16,7 @@ const ( // ShiftKey represents the shift key on all systems ShiftKey Modifier = "shift" // SuperKey represents Command on Mac and the Windows key on the other platforms - //SuperKey Modifier = "super" + // SuperKey Modifier = "super" // ControlKey represents the control key on all systems ControlKey Modifier = "ctrl" ) diff --git a/v2/pkg/menu/keys/parser.go b/v2/pkg/menu/keys/parser.go index 91a05783d41..6e8e123766e 100644 --- a/v2/pkg/menu/keys/parser.go +++ b/v2/pkg/menu/keys/parser.go @@ -11,7 +11,6 @@ import ( var namedKeys = slicer.String([]string{"backspace", "tab", "return", "enter", "escape", "left", "right", "up", "down", "space", "delete", "home", "end", "page up", "page down", "f1", "f2", "f3", "f4", "f5", "f6", "f7", "f8", "f9", "f10", "f11", "f12", "f13", "f14", "f15", "f16", "f17", "f18", "f19", "f20", "f21", "f22", "f23", "f24", "f25", "f26", "f27", "f28", "f29", "f30", "f31", "f32", "f33", "f34", "f35", "numlock"}) func parseKey(key string) (string, bool) { - // Lowercase! key = strings.ToLower(key) @@ -38,11 +37,9 @@ func parseKey(key string) (string, bool) { } return "", false - } func Parse(shortcut string) (*Accelerator, error) { - var result Accelerator // Split the shortcut by + diff --git a/v2/pkg/menu/keys/stringify.go b/v2/pkg/menu/keys/stringify.go index ccc8c5e9e70..92498f5d49c 100644 --- a/v2/pkg/menu/keys/stringify.go +++ b/v2/pkg/menu/keys/stringify.go @@ -1,8 +1,9 @@ package keys import ( - "github.com/leaanthony/slicer" "strings" + + "github.com/leaanthony/slicer" ) var modifierStringMap = map[string]map[Modifier]string{ @@ -11,21 +12,21 @@ var modifierStringMap = map[string]map[Modifier]string{ ControlKey: "Ctrl", OptionOrAltKey: "Alt", ShiftKey: "Shift", - //SuperKey: "Win", + // SuperKey: "Win", }, "darwin": { CmdOrCtrlKey: "Cmd", ControlKey: "Ctrl", OptionOrAltKey: "Option", ShiftKey: "Shift", - //SuperKey: "Cmd", + // SuperKey: "Cmd", }, "linux": { CmdOrCtrlKey: "Ctrl", ControlKey: "Ctrl", OptionOrAltKey: "Alt", ShiftKey: "Shift", - //SuperKey: "Super", + // SuperKey: "Super", }, } diff --git a/v2/pkg/menu/menu.go b/v2/pkg/menu/menu.go index 819939bbfaa..86acbd1d098 100644 --- a/v2/pkg/menu/menu.go +++ b/v2/pkg/menu/menu.go @@ -59,8 +59,7 @@ func (m *Menu) Prepend(item *MenuItem) { } func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu { - - var result = NewMenu() + result := NewMenu() result.Append(first) for _, item := range rest { result.Append(item) diff --git a/v2/pkg/menu/menuitem.go b/v2/pkg/menu/menuitem.go index f6ea681d7ad..264b2ebd41a 100644 --- a/v2/pkg/menu/menuitem.go +++ b/v2/pkg/menu/menuitem.go @@ -23,7 +23,7 @@ type MenuItem struct { // Checked indicates if the item is selected (used by Checkbox and Radio types only) Checked bool // Submenu contains a list of menu items that will be shown as a submenu - //SubMenu []*MenuItem `json:"SubMenu,omitempty"` + // SubMenu []*MenuItem `json:"SubMenu,omitempty"` SubMenu *Menu // Callback function when menu clicked @@ -106,7 +106,6 @@ func (m *MenuItem) removeChild(item *MenuItem) { // menu. If there is no parent menu (we are a top level menu) then false is // returned func (m *MenuItem) InsertAfter(item *MenuItem) bool { - // We need to find my parent if m.parent == nil { return false @@ -120,7 +119,6 @@ func (m *MenuItem) InsertAfter(item *MenuItem) bool { // menu. If there is no parent menu (we are a top level menu) then false is // returned func (m *MenuItem) InsertBefore(item *MenuItem) bool { - // We need to find my parent if m.parent == nil { return false @@ -134,8 +132,8 @@ func (m *MenuItem) InsertBefore(item *MenuItem) bool { // in this item's submenu. If we are not a submenu, // then something bad has happened :/ func (m *MenuItem) insertNewItemAfterGivenItem(target *MenuItem, - newItem *MenuItem) bool { - + newItem *MenuItem, +) bool { if !m.isSubMenu() { return false } @@ -154,8 +152,8 @@ func (m *MenuItem) insertNewItemAfterGivenItem(target *MenuItem, // target in this item's submenu. If we are not a submenu, then something bad // has happened :/ func (m *MenuItem) insertNewItemBeforeGivenItem(target *MenuItem, - newItem *MenuItem) bool { - + newItem *MenuItem, +) bool { if !m.isSubMenu() { return false } @@ -176,7 +174,6 @@ func (m *MenuItem) isSubMenu() bool { // getItemIndex returns the index of the given target relative to this menu func (m *MenuItem) getItemIndex(target *MenuItem) int { - // This should only be called on submenus if !m.isSubMenu() { return -1 @@ -196,7 +193,6 @@ func (m *MenuItem) getItemIndex(target *MenuItem) int { // the given index // Credit: https://stackoverflow.com/a/61822301 func (m *MenuItem) insertItemAtIndex(index int, target *MenuItem) bool { - // If index is OOB, return false if index > len(m.SubMenu.Items) { return false diff --git a/v2/pkg/menu/menuroles.go b/v2/pkg/menu/menuroles.go index e6b15b2433b..bcc0657fc81 100644 --- a/v2/pkg/menu/menuroles.go +++ b/v2/pkg/menu/menuroles.go @@ -11,29 +11,29 @@ const ( AppMenuRole Role = 1 EditMenuRole = 2 WindowMenuRole = 3 - //AboutRole Role = "about" - //UndoRole Role = "undo" - //RedoRole Role = "redo" - //CutRole Role = "cut" - //CopyRole Role = "copy" - //PasteRole Role = "paste" - //PasteAndMatchStyleRole Role = "pasteAndMatchStyle" - //SelectAllRole Role = "selectAll" - //DeleteRole Role = "delete" - //MinimizeRole Role = "minimize" - //QuitRole Role = "quit" - //TogglefullscreenRole Role = "togglefullscreen" - //FileMenuRole Role = "fileMenu" - //ViewMenuRole Role = "viewMenu" - //WindowMenuRole Role = "windowMenu" - //HideRole Role = "hide" - //HideOthersRole Role = "hideOthers" - //UnhideRole Role = "unhide" - //FrontRole Role = "front" - //ZoomRole Role = "zoom" - //WindowSubMenuRole Role = "windowSubMenu" - //HelpSubMenuRole Role = "helpSubMenu" - //SeparatorItemRole Role = "separatorItem" + // AboutRole Role = "about" + // UndoRole Role = "undo" + // RedoRole Role = "redo" + // CutRole Role = "cut" + // CopyRole Role = "copy" + // PasteRole Role = "paste" + // PasteAndMatchStyleRole Role = "pasteAndMatchStyle" + // SelectAllRole Role = "selectAll" + // DeleteRole Role = "delete" + // MinimizeRole Role = "minimize" + // QuitRole Role = "quit" + // TogglefullscreenRole Role = "togglefullscreen" + // FileMenuRole Role = "fileMenu" + // ViewMenuRole Role = "viewMenu" + // WindowMenuRole Role = "windowMenu" + // HideRole Role = "hide" + // HideOthersRole Role = "hideOthers" + // UnhideRole Role = "unhide" + // FrontRole Role = "front" + // ZoomRole Role = "zoom" + // WindowSubMenuRole Role = "windowSubMenu" + // HelpSubMenuRole Role = "helpSubMenu" + // SeparatorItemRole Role = "separatorItem" ) /* diff --git a/v2/pkg/menu/styledlabel.go b/v2/pkg/menu/styledlabel.go index 11a0254c9e6..1e996b9717d 100644 --- a/v2/pkg/menu/styledlabel.go +++ b/v2/pkg/menu/styledlabel.go @@ -29,24 +29,31 @@ type StyledText struct { func (s *StyledText) Bold() bool { return s.Style&Bold == Bold } + func (s *StyledText) Faint() bool { return s.Style&Faint == Faint } + func (s *StyledText) Italic() bool { return s.Style&Italic == Italic } + func (s *StyledText) Blinking() bool { return s.Style&Blinking == Blinking } + func (s *StyledText) Inversed() bool { return s.Style&Inversed == Inversed } + func (s *StyledText) Invisible() bool { return s.Style&Invisible == Invisible } + func (s *StyledText) Underlined() bool { return s.Style&Underlined == Underlined } + func (s *StyledText) Strikethrough() bool { return s.Style&Strikethrough == Strikethrough } diff --git a/v2/pkg/menu/tray.go b/v2/pkg/menu/tray.go index 7554795ada4..c8728f1f7f1 100644 --- a/v2/pkg/menu/tray.go +++ b/v2/pkg/menu/tray.go @@ -2,7 +2,6 @@ package menu // TrayMenu are the options type TrayMenu struct { - // Label is the text we wish to display in the tray Label string @@ -27,7 +26,7 @@ type TrayMenu struct { Tooltip string // Callback function when menu clicked - //Click Callback `json:"-"` + // Click Callback `json:"-"` // Disabled makes the item unselectable Disabled bool diff --git a/v2/pkg/options/mac/mac.go b/v2/pkg/options/mac/mac.go index 5758befd011..9d2ccc953c9 100644 --- a/v2/pkg/options/mac/mac.go +++ b/v2/pkg/options/mac/mac.go @@ -21,9 +21,9 @@ type Options struct { WebviewIsTransparent bool WindowIsTranslucent bool Preferences *Preferences - //ActivationPolicy ActivationPolicy + // ActivationPolicy ActivationPolicy About *AboutInfo OnFileOpen func(filePath string) `json:"-"` OnUrlOpen func(filePath string) `json:"-"` - //URLHandlers map[string]func(string) + // URLHandlers map[string]func(string) } diff --git a/v2/pkg/options/mac/preferences.go b/v2/pkg/options/mac/preferences.go index dbc20c805a3..0749ccb18f5 100644 --- a/v2/pkg/options/mac/preferences.go +++ b/v2/pkg/options/mac/preferences.go @@ -2,8 +2,10 @@ package mac import "github.com/leaanthony/u" -var Enabled = u.True -var Disabled = u.False +var ( + Enabled = u.True + Disabled = u.False +) // Preferences allows to set webkit preferences type Preferences struct { diff --git a/v2/pkg/options/mac/titlebar.go b/v2/pkg/options/mac/titlebar.go index c18c4eea8ca..51e0832cab5 100644 --- a/v2/pkg/options/mac/titlebar.go +++ b/v2/pkg/options/mac/titlebar.go @@ -41,7 +41,6 @@ func TitleBarHidden() *TitleBar { // TitleBarHiddenInset results in a hidden title bar with an alternative look where // the traffic light buttons are slightly more inset from the window edge. func TitleBarHiddenInset() *TitleBar { - return &TitleBar{ TitlebarAppearsTransparent: true, HideTitle: true, @@ -50,5 +49,4 @@ func TitleBarHiddenInset() *TitleBar { UseToolbar: true, HideToolbarSeparator: true, } - } diff --git a/v2/pkg/options/options.go b/v2/pkg/options/options.go index 1ecad7fb919..088e7c46ab8 100644 --- a/v2/pkg/options/options.go +++ b/v2/pkg/options/options.go @@ -28,8 +28,7 @@ const ( Fullscreen WindowStartState = 3 ) -type Experimental struct { -} +type Experimental struct{} // App contains options for creating the App type App struct { diff --git a/v2/pkg/options/windows/windows.go b/v2/pkg/options/windows/windows.go index 7adf0bf7237..073450c9f68 100644 --- a/v2/pkg/options/windows/windows.go +++ b/v2/pkg/options/windows/windows.go @@ -36,7 +36,7 @@ const ( ) func RGB(r, g, b uint8) int32 { - var col = int32(b) + col := int32(b) col = col<<8 | int32(g) col = col<<8 | int32(r) return col diff --git a/v2/pkg/runtime/dialog.go b/v2/pkg/runtime/dialog.go index d53a89c1557..16ae659e106 100644 --- a/v2/pkg/runtime/dialog.go +++ b/v2/pkg/runtime/dialog.go @@ -3,6 +3,7 @@ package runtime import ( "context" "fmt" + "github.com/wailsapp/wails/v2/internal/frontend" "github.com/wailsapp/wails/v2/internal/fs" ) diff --git a/v2/pkg/runtime/events.go b/v2/pkg/runtime/events.go index 493d811689a..84aff7d7407 100644 --- a/v2/pkg/runtime/events.go +++ b/v2/pkg/runtime/events.go @@ -10,7 +10,7 @@ func EventsOn(ctx context.Context, eventName string, callback func(optionalData return events.On(eventName, callback) } -// EventsOff unregisters a listener for the given event name, optionally multiple listeneres can be unregistered via `additionalEventNames` +// EventsOff unregisters a listener for the given event name, optionally multiple listeners can be unregistered via `additionalEventNames` func EventsOff(ctx context.Context, eventName string, additionalEventNames ...string) { events := getEvents(ctx) events.Off(eventName) @@ -22,7 +22,7 @@ func EventsOff(ctx context.Context, eventName string, additionalEventNames ...st } } -// EventsOff unregisters a listener for the given event name, optionally multiple listeneres can be unregistered via `additionalEventNames` +// EventsOff unregisters a listener for the given event name, optionally multiple listeners can be unregistered via `additionalEventNames` func EventsOffAll(ctx context.Context) { events := getEvents(ctx) events.OffAll() diff --git a/v2/pkg/runtime/log.go b/v2/pkg/runtime/log.go index 4d3f56d3f20..3c2756f06ca 100644 --- a/v2/pkg/runtime/log.go +++ b/v2/pkg/runtime/log.go @@ -3,6 +3,7 @@ package runtime import ( "context" "fmt" + "github.com/wailsapp/wails/v2/pkg/logger" ) diff --git a/v2/pkg/runtime/menu.go b/v2/pkg/runtime/menu.go index 176c9bb1db9..09bd640c5e9 100644 --- a/v2/pkg/runtime/menu.go +++ b/v2/pkg/runtime/menu.go @@ -2,6 +2,7 @@ package runtime import ( "context" + "github.com/wailsapp/wails/v2/pkg/menu" ) diff --git a/v2/pkg/runtime/runtime.go b/v2/pkg/runtime/runtime.go index 4702b439a8a..6de5ea79892 100644 --- a/v2/pkg/runtime/runtime.go +++ b/v2/pkg/runtime/runtime.go @@ -27,6 +27,7 @@ func getFrontend(ctx context.Context) frontend.Frontend { log.Fatalf("cannot call '%s': %s", funcName, contextError) return nil } + func getLogger(ctx context.Context) *logger.Logger { if ctx == nil { pc, _, _, _ := goruntime.Caller(1) diff --git a/v2/pkg/runtime/screen.go b/v2/pkg/runtime/screen.go index d92ed8308c5..af8fb626ed9 100644 --- a/v2/pkg/runtime/screen.go +++ b/v2/pkg/runtime/screen.go @@ -1,7 +1,10 @@ package runtime -import "context" -import "github.com/wailsapp/wails/v2/internal/frontend" +import ( + "context" + + "github.com/wailsapp/wails/v2/internal/frontend" +) type Screen = frontend.Screen diff --git a/v2/pkg/templates/generate/generate.go b/v2/pkg/templates/generate/generate.go index 3b01e5f2a67..6842dc19678 100644 --- a/v2/pkg/templates/generate/generate.go +++ b/v2/pkg/templates/generate/generate.go @@ -159,7 +159,6 @@ var templates = []*template{ } func main() { - rebuildRuntime() for _, t := range templates { diff --git a/v2/pkg/templates/templates.go b/v2/pkg/templates/templates.go index d982454d04d..9b42ef36559 100644 --- a/v2/pkg/templates/templates.go +++ b/v2/pkg/templates/templates.go @@ -72,7 +72,6 @@ type Options struct { // Template holds data relating to a template // including the metadata stored in template.json type Template struct { - // Template details Name string `json:"name"` ShortName string `json:"shortname"` @@ -100,7 +99,6 @@ func parseTemplate(template gofs.FS) (Template, error) { // List returns the list of available templates func List() ([]Template, error) { - // If the cache isn't loaded, load it if templateCache == nil { err := loadTemplateCache() @@ -114,7 +112,6 @@ func List() ([]Template, error) { // getTemplateByShortname returns the template with the given short name func getTemplateByShortname(shortname string) (Template, error) { - var result Template // If the cache isn't loaded, load it @@ -136,7 +133,6 @@ func getTemplateByShortname(shortname string) (Template, error) { // Loads the template cache func loadTemplateCache() error { - templatesFS, err := debme.FS(templates, "templates") if err != nil { return err @@ -309,11 +305,9 @@ func gitclone(options *Options) (string, error) { _, err = git.PlainClone(dirname, false, cloneOption) return dirname, err - } func generateIDEFiles(options *Options) error { - switch options.IDE { case "vscode": return generateVSCodeFiles(options) @@ -361,7 +355,6 @@ func generateVSCodeFiles(options *Options) error { options: options, } return installIDEFiles(ideoptions) - } func installIDEFiles(o ideOptions) error { diff --git a/v2/tools/release/release.go b/v2/tools/release/release.go index dc6299ed73e..4178fcc950f 100644 --- a/v2/tools/release/release.go +++ b/v2/tools/release/release.go @@ -33,7 +33,7 @@ func updateVersion() string { minorVersion++ vsplit[len(vsplit)-1] = strconv.Itoa(minorVersion) newVersion := strings.Join(vsplit, ".") - err = os.WriteFile(versionFile, []byte(newVersion), 0755) + err = os.WriteFile(versionFile, []byte(newVersion), 0o755) checkError(err) return newVersion } @@ -68,7 +68,7 @@ func main() { newVersion = os.Args[1] currentVersion, err := os.ReadFile(versionFile) checkError(err) - err = os.WriteFile(versionFile, []byte(newVersion), 0755) + err = os.WriteFile(versionFile, []byte(newVersion), 0o755) checkError(err) isPointRelease = IsPointRelease(string(currentVersion), newVersion) } else { @@ -89,7 +89,7 @@ func main() { // Add the new version to the top of the changelog newChangelog := changelogSplit[0] + "## [Unreleased]\n\n## " + newVersion + " - " + today + changelogSplit[1] // Write the changelog back - err = os.WriteFile("src/pages/changelog.mdx", []byte(newChangelog), 0755) + err = os.WriteFile("src/pages/changelog.mdx", []byte(newChangelog), 0o755) checkError(err) if !isPointRelease { @@ -112,7 +112,7 @@ func main() { versions = versions[0 : len(versions)-1] newVersions, err := json.Marshal(&versions) checkError(err) - err = os.WriteFile("versions.json", newVersions, 0755) + err = os.WriteFile("versions.json", newVersions, 0o755) checkError(err) s.ECHO("Removing old version: " + oldestVersion) From 4efc8e358dc0a6dd28d61ad9e4a9bde029c69b5d Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 12 Nov 2023 14:04:48 +1100 Subject: [PATCH 48/87] [chore] fix go.mod --- v2/go.mod | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/v2/go.mod b/v2/go.mod index b166a30b2ef..88bd7c75cda 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -1,8 +1,6 @@ module github.com/wailsapp/wails/v2 -go 1.21 - -toolchain go1.21.0 +go 1.18 require ( github.com/Masterminds/semver v1.5.0 From 801cf3c84ae5c07c1ab31b56e420af4861992d14 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 17 Nov 2023 20:04:22 +1100 Subject: [PATCH 49/87] Bump sharp and sponsorkit in /scripts/sponsors (#3060) Bumps [sharp](https://github.com/lovell/sharp) to 0.32.6 and updates ancestor dependency [sponsorkit](https://github.com/antfu/sponsorkit). These dependencies need to be updated together. Updates `sharp` from 0.31.3 to 0.32.6 - [Release notes](https://github.com/lovell/sharp/releases) - [Changelog](https://github.com/lovell/sharp/blob/main/docs/changelog.md) - [Commits](https://github.com/lovell/sharp/compare/v0.31.3...v0.32.6) Updates `sponsorkit` from 0.8.2 to 0.8.9 - [Release notes](https://github.com/antfu/sponsorkit/releases) - [Commits](https://github.com/antfu/sponsorkit/compare/v0.8.2...v0.8.9) --- updated-dependencies: - dependency-name: sharp dependency-type: indirect - dependency-name: sponsorkit dependency-type: direct:production ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- scripts/sponsors/package-lock.json | 315 ++++++++++++++++++----------- 1 file changed, 194 insertions(+), 121 deletions(-) diff --git a/scripts/sponsors/package-lock.json b/scripts/sponsors/package-lock.json index e3523a1d983..6fa8f99fcc2 100644 --- a/scripts/sponsors/package-lock.json +++ b/scripts/sponsors/package-lock.json @@ -13,19 +13,22 @@ } }, "node_modules/@antfu/utils": { - "version": "0.7.4", - "resolved": "https://registry.npmjs.org/@antfu/utils/-/utils-0.7.4.tgz", - "integrity": "sha512-qe8Nmh9rYI/HIspLSTwtbMFPj6dISG6+dJnOguTlPNXtCvS2uezdxscVBb7/3DrmNbQK49TDqpkSQ1chbRGdpQ==", + "version": "0.7.6", + "resolved": "https://registry.npmjs.org/@antfu/utils/-/utils-0.7.6.tgz", + "integrity": "sha512-pvFiLP2BeOKA/ZOS6jxx4XhKzdVLHDhGlFEaZ2flWWYf2xOqVniqpk38I04DFRyz+L0ASggl7SkItTc+ZLju4w==", "funding": { "url": "https://github.com/sponsors/antfu" } }, - "node_modules/@fastify/busboy": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@fastify/busboy/-/busboy-2.0.0.tgz", - "integrity": "sha512-JUFJad5lv7jxj926GPgymrWQxxjPYuJNiNjNMzqT+HiuP6Vl3dk5xzG+8sTX96np0ZAluvaMzPsjhHZ5rNuNQQ==", + "node_modules/acorn": { + "version": "8.11.2", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.2.tgz", + "integrity": "sha512-nc0Axzp/0FILLEVsm4fNwLCwMttvhEI263QtVPQcbpfZZ3ts0hLsZGOpE6czNlid7CJ9MlyH8reXkpsf3YUY4w==", + "bin": { + "acorn": "bin/acorn" + }, "engines": { - "node": ">=14" + "node": ">=0.4.0" } }, "node_modules/ajv": { @@ -125,6 +128,11 @@ "resolved": "https://registry.npmmirror.com/aws4/-/aws4-1.11.0.tgz", "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==" }, + "node_modules/b4a": { + "version": "1.6.4", + "resolved": "https://registry.npmjs.org/b4a/-/b4a-1.6.4.tgz", + "integrity": "sha512-fpWrvyVHEKyeEvbKZTVOeZF3VSKKWtJxFIxX/jaVPf+cLbGUSitjb49pHLqPV2BUNNZ0LcoeEGfE/YCpyDYHIw==" + }, "node_modules/balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz", @@ -342,9 +350,12 @@ "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" }, "node_modules/consola": { - "version": "2.15.3", - "resolved": "https://registry.npmmirror.com/consola/-/consola-2.15.3.tgz", - "integrity": "sha512-9vAdYbHj6x2fLKC4+oPH0kFzY/orMZyG2Aj+kNylHxKGJ/Ed4dpNyAQYwJOdqO4zdM7XpVHmyejQDcQHrnuXbw==" + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/consola/-/consola-3.2.3.tgz", + "integrity": "sha512-I5qxpzLv+sJhTVEoLYNcTW+bThDCPsit0vLNKShZx6rLtpilNpmmeTPaeqJb9ZE9dV3DGaeby6Vuhrw38WjeyQ==", + "engines": { + "node": "^14.18.0 || >=16.10.0" + } }, "node_modules/core-util-is": { "version": "1.0.2", @@ -419,9 +430,9 @@ } }, "node_modules/defu": { - "version": "6.1.2", - "resolved": "https://registry.npmjs.org/defu/-/defu-6.1.2.tgz", - "integrity": "sha512-+uO4+qr7msjNNWKYPHqN/3+Dx3NFkmIzayk2L1MyZQlvgZb/J1A0fo410dpKrN2SnqFjt8n4JL8fDJE0wIgjFQ==" + "version": "6.1.3", + "resolved": "https://registry.npmjs.org/defu/-/defu-6.1.3.tgz", + "integrity": "sha512-Vy2wmG3NTkmHNg/kzpuvHhkqeIx3ODWqasgCRbKtbXEN0G+HpEEv9BtJLp7ZG1CZloFaC41Ah3ZFbq7aqCqMeQ==" }, "node_modules/delayed-stream": { "version": "1.0.0", @@ -432,14 +443,14 @@ } }, "node_modules/destr": { - "version": "1.2.1", - "resolved": "https://registry.npmmirror.com/destr/-/destr-1.2.1.tgz", - "integrity": "sha512-ud8w0qMLlci6iFG7CNgeRr8OcbUWMsbfjtWft1eJ5Luqrz/M8Ebqk/KCzne8rKUlIQWWfLv0wD6QHrqOf4GshA==" + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/destr/-/destr-2.0.2.tgz", + "integrity": "sha512-65AlobnZMiCET00KaFFjUefxDX0khFA/E4myqZ7a6Sq1yZtR8+FVIvilVX66vF2uobSumxooYZChiRPCKNqhmg==" }, "node_modules/detect-libc": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.1.tgz", - "integrity": "sha512-463v3ZeIrcWtdgIg6vI6XUncguvr2TnGl4SzDXinkt9mSLpBJKXT3mW6xT3VQdDN11+WVs29pgvivTc4Lp8v+w==", + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.2.tgz", + "integrity": "sha512-UX6sGumvvqSaXgdKGUsgZWqcUyIXZ/vZTrlRT/iobiKhGL0zL4d3osHj3uqllWJK+i+sixDS/3COVEOFbupFyw==", "engines": { "node": ">=8" } @@ -483,24 +494,27 @@ } }, "node_modules/domutils": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz", - "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.1.0.tgz", + "integrity": "sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==", "dependencies": { "dom-serializer": "^2.0.0", "domelementtype": "^2.3.0", - "domhandler": "^5.0.1" + "domhandler": "^5.0.3" }, "funding": { "url": "https://github.com/fb55/domutils?sponsor=1" } }, "node_modules/dotenv": { - "version": "16.0.3", - "resolved": "https://registry.npmmirror.com/dotenv/-/dotenv-16.0.3.tgz", - "integrity": "sha512-7GO6HghkA5fYG9TYnNxi14/7K9f5occMlp3zXAuSxn7CKCxt9xbNWG7yF8hTCSUchlfWSe3uLmlPfigevRItzQ==", + "version": "16.3.1", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.3.1.tgz", + "integrity": "sha512-IPzF4w4/Rd94bA9imS68tZBaYyBWSCE47V1RGuMrB94iyTOIEwRmVL2x/4An+6mETpLrKJ5hQkB8W4kFAadeIQ==", "engines": { "node": ">=12" + }, + "funding": { + "url": "https://github.com/motdotla/dotenv?sponsor=1" } }, "node_modules/ecc-jsbn": { @@ -526,9 +540,9 @@ } }, "node_modules/entities": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz", - "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA==", + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", "engines": { "node": ">=0.12" }, @@ -570,6 +584,11 @@ "resolved": "https://registry.npmmirror.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" }, + "node_modules/fast-fifo": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/fast-fifo/-/fast-fifo-1.3.2.tgz", + "integrity": "sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ==" + }, "node_modules/fast-json-stable-stringify": { "version": "2.1.0", "resolved": "https://registry.npmmirror.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", @@ -629,9 +648,9 @@ "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==" }, "node_modules/fs-extra": { - "version": "11.1.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.1.0.tgz", - "integrity": "sha512-0rcTq621PD5jM/e0a3EJoGC/1TC5ZBCERW82LQuwfGnCa1V8w7dpYH1yNu+SLb6E5dkeCBzKEyLGlFrnr+dUyw==", + "version": "11.1.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.1.1.tgz", + "integrity": "sha512-MGIE4HOvQCeUCzmlHs0vXpih4ysz4wg9qiSAu6cd42lVwPbTM1TjV7RusoyQqMmk/95gdQZX72u+YW+c3eEpFQ==", "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", @@ -951,9 +970,9 @@ "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==" }, "node_modules/jiti": { - "version": "1.18.2", - "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.18.2.tgz", - "integrity": "sha512-QAdOptna2NYiSSpv0O/BwoHBSmz4YhpzJHyi+fnMRTXFjp7B8i/YG5Z8IfusxB1ufjcD2Sre1F3R+nX3fvy7gg==", + "version": "1.21.0", + "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.0.tgz", + "integrity": "sha512-gFqAIbuKyyso/3G2qhiO2OM6shY6EPP/R0+mkDbyspxKazh8BXDC5FiFsUjlczgdNz/vfra0da2y+aHrusLG/Q==", "bin": { "jiti": "bin/jiti.js" } @@ -978,6 +997,11 @@ "resolved": "https://registry.npmmirror.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==" }, + "node_modules/jsonc-parser": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.2.0.tgz", + "integrity": "sha512-gfFQZrcTc8CnKXp6Y4/CBT3fTc0OVuDofpre4aEeEpSBPV5X5v4+Vmx+8snU7RLPrNHPKSgLxGo9YuQzz20o+w==" + }, "node_modules/jsonfile": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", @@ -1104,15 +1128,26 @@ "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==" }, + "node_modules/mlly": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/mlly/-/mlly-1.4.2.tgz", + "integrity": "sha512-i/Ykufi2t1EZ6NaPLdfnZk2AX8cs0d+mTzVKuPfqPKPatxLApaBoxJQ9x1/uckXtrS/U5oisPMDkNs0yQTaBRg==", + "dependencies": { + "acorn": "^8.10.0", + "pathe": "^1.1.1", + "pkg-types": "^1.0.3", + "ufo": "^1.3.0" + } + }, "node_modules/napi-build-utils": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-1.0.2.tgz", "integrity": "sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==" }, "node_modules/node-abi": { - "version": "3.33.0", - "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.33.0.tgz", - "integrity": "sha512-7GGVawqyHF4pfd0YFybhv/eM9JwTtPqx0mAanQ146O3FlSh3pA24zf9IRQTOsfTSqXTNzPSP5iagAJ94jjuVog==", + "version": "3.51.0", + "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.51.0.tgz", + "integrity": "sha512-SQkEP4hmNWjlniS5zdnfIXTk1x7Ome85RDzHlTbBtzE97Gfwz/Ipw4v/Ryk20DWIy3yCNVLVlGKApCnmvYoJbA==", "dependencies": { "semver": "^7.3.5" }, @@ -1121,19 +1156,19 @@ } }, "node_modules/node-addon-api": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-5.1.0.tgz", - "integrity": "sha512-eh0GgfEkpnoWDq+VY8OyvYhFEzBk6jIYbRKdIlyTiAXIVJ8PyBaKb0rp7oDtoddbdoHWhq8wwr+XZ81F1rpNdA==" + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-6.1.0.tgz", + "integrity": "sha512-+eawOlIgy680F0kBzPUNFhMZGtJ1YmqM6l4+Crf4IkImjYrO/mqPwRMh352g23uIaQKFItcQ64I7KMaJxHgAVA==" }, "node_modules/node-fetch-native": { - "version": "0.1.8", - "resolved": "https://registry.npmmirror.com/node-fetch-native/-/node-fetch-native-0.1.8.tgz", - "integrity": "sha512-ZNaury9r0NxaT2oL65GvdGDy+5PlSaHTovT6JV5tOW07k1TQmgC0olZETa4C9KZg0+6zBr99ctTYa3Utqj9P/Q==" + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/node-fetch-native/-/node-fetch-native-1.4.1.tgz", + "integrity": "sha512-NsXBU0UgBxo2rQLOeWNZqS3fvflWePMECr8CoSWoSTqCqGbVVsvl9vZu1HfQicYN0g5piV9Gh8RTEvo/uP752w==" }, "node_modules/node-html-parser": { - "version": "6.1.5", - "resolved": "https://registry.npmjs.org/node-html-parser/-/node-html-parser-6.1.5.tgz", - "integrity": "sha512-fAaM511feX++/Chnhe475a0NHD8M7AxDInsqQpz6x63GRF7xYNdS8Vo5dKsIVPgsOvG7eioRRTZQnWBrhDHBSg==", + "version": "6.1.11", + "resolved": "https://registry.npmjs.org/node-html-parser/-/node-html-parser-6.1.11.tgz", + "integrity": "sha512-FAgwwZ6h0DSDWxfD0Iq1tsDcBCxdJB1nXpLPPxX8YyVWzbfCjKWEzaynF4gZZ/8hziUmp7ZSaKylcn0iKhufUQ==", "dependencies": { "css-select": "^5.1.0", "he": "1.2.0" @@ -1212,15 +1247,14 @@ "pipe-functions": "^1.2.0" } }, - "node_modules/ohmyfetch": { - "version": "0.4.21", - "resolved": "https://registry.npmmirror.com/ohmyfetch/-/ohmyfetch-0.4.21.tgz", - "integrity": "sha512-VG7f/JRvqvBOYvL0tHyEIEG7XHWm7OqIfAs6/HqwWwDfjiJ1g0huIpe5sFEmyb+7hpFa1EGNH2aERWR72tlClw==", + "node_modules/ofetch": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/ofetch/-/ofetch-1.3.3.tgz", + "integrity": "sha512-s1ZCMmQWXy4b5K/TW9i/DtiN8Ku+xCiHcjQ6/J/nDdssirrQNOoB165Zu8EqLMA2lln1JUth9a0aW9Ap2ctrUg==", "dependencies": { - "destr": "^1.2.0", - "node-fetch-native": "^0.1.8", - "ufo": "^0.8.6", - "undici": "^5.12.0" + "destr": "^2.0.1", + "node-fetch-native": "^1.4.0", + "ufo": "^1.3.0" } }, "node_modules/once": { @@ -1277,6 +1311,11 @@ "node": ">=0.10.0" } }, + "node_modules/pathe": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/pathe/-/pathe-1.1.1.tgz", + "integrity": "sha512-d+RQGp0MAYTIaDBIMmOfMwz3E+LOZnxx1HZd5R18mmCZY0QBlK0LDZfPc8FW8Ed2DlvsuE6PRjroDY+wg4+j/Q==" + }, "node_modules/performance-now": { "version": "2.1.0", "resolved": "https://registry.npmmirror.com/performance-now/-/performance-now-2.1.0.tgz", @@ -1292,6 +1331,16 @@ "resolved": "https://registry.npmmirror.com/pipe-functions/-/pipe-functions-1.3.0.tgz", "integrity": "sha512-6Rtbp7criZRwedlvWbUYxqlqJoAlMvYHo2UcRWq79xZ54vZcaNHpVBOcWkX3ErT2aUA69tv+uiv4zKJbhD/Wgg==" }, + "node_modules/pkg-types": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/pkg-types/-/pkg-types-1.0.3.tgz", + "integrity": "sha512-nN7pYi0AQqJnoLPC9eHFQ8AcyaixBUOwvqc5TDnIKCMEE6I0y8P7OKA7fPexsXGCGxQDl/cmrLAp26LhcwxZ4A==", + "dependencies": { + "jsonc-parser": "^3.2.0", + "mlly": "^1.2.0", + "pathe": "^1.1.0" + } + }, "node_modules/prebuild-install": { "version": "7.1.1", "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.1.tgz", @@ -1317,6 +1366,32 @@ "node": ">=10" } }, + "node_modules/prebuild-install/node_modules/tar-fs": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz", + "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==", + "dependencies": { + "chownr": "^1.1.1", + "mkdirp-classic": "^0.5.2", + "pump": "^3.0.0", + "tar-stream": "^2.1.4" + } + }, + "node_modules/prebuild-install/node_modules/tar-stream": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", + "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "dependencies": { + "bl": "^4.0.3", + "end-of-stream": "^1.4.1", + "fs-constants": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.1.1" + }, + "engines": { + "node": ">=6" + } + }, "node_modules/psl": { "version": "1.9.0", "resolved": "https://registry.npmmirror.com/psl/-/psl-1.9.0.tgz", @@ -1347,6 +1422,11 @@ "node": ">=0.6" } }, + "node_modules/queue-tick": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/queue-tick/-/queue-tick-1.0.1.tgz", + "integrity": "sha512-kJt5qhMxoszgU/62PLP1CJytzd2NKetjSRnyuj31fDd3Rlcz3fzlFdFLD1SItunPwyqEOkca6GbV612BWfaBag==" + }, "node_modules/rc": { "version": "1.2.8", "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", @@ -1362,9 +1442,9 @@ } }, "node_modules/readable-stream": { - "version": "3.6.1", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.1.tgz", - "integrity": "sha512-+rQmrWMYGA90yenhTYsLWAsLsqVC8osOw6PKE1HDYiO0gdPeKe/xDHNzIAIn4C91YQ6oenEhfYqqc1883qHbjQ==", + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", "dependencies": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", @@ -1443,9 +1523,9 @@ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" }, "node_modules/semver": { - "version": "7.5.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.3.tgz", - "integrity": "sha512-QBlUtyVk/5EeHbi7X0fw6liDZc7BBmEaSYn01fMU1OUYbf6GPsbTtd8WmnqbI20SeycoHSeiybkE/q1Q+qlThQ==", + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", "dependencies": { "lru-cache": "^6.0.0" }, @@ -1457,18 +1537,18 @@ } }, "node_modules/sharp": { - "version": "0.31.3", - "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.31.3.tgz", - "integrity": "sha512-XcR4+FCLBFKw1bdB+GEhnUNXNXvnt0tDo4WsBsraKymuo/IAuPuCBVAL2wIkUw2r/dwFW5Q5+g66Kwl2dgDFVg==", + "version": "0.32.6", + "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.32.6.tgz", + "integrity": "sha512-KyLTWwgcR9Oe4d9HwCwNM2l7+J0dUQwn/yf7S0EnTtb0eVS4RxO0eUSvxPtzT4F3SY+C4K6fqdv/DO27sJ/v/w==", "hasInstallScript": true, "dependencies": { "color": "^4.2.3", - "detect-libc": "^2.0.1", - "node-addon-api": "^5.0.0", + "detect-libc": "^2.0.2", + "node-addon-api": "^6.1.0", "prebuild-install": "^7.1.1", - "semver": "^7.3.8", + "semver": "^7.5.4", "simple-get": "^4.0.1", - "tar-fs": "^2.1.1", + "tar-fs": "^3.0.4", "tunnel-agent": "^0.6.0" }, "engines": { @@ -1535,20 +1615,20 @@ "integrity": "sha512-weHOi8BolsDnGIwhhWHbA+wKSuSpvWwjRrdj8SdbIIis2vSwOE37CQP8x3EleuzxanUr3AK8BdUy4MkiOULPZg==" }, "node_modules/sponsorkit": { - "version": "0.8.2", - "resolved": "https://registry.npmjs.org/sponsorkit/-/sponsorkit-0.8.2.tgz", - "integrity": "sha512-Gxh7hkTUuUVj823+BnwC77Rl5ztFEY00qA2QVULOU0N5qgj1YEP3/0BB/EayfPrHeV7HbAOwSeAnqC8/yoRABA==", + "version": "0.8.9", + "resolved": "https://registry.npmjs.org/sponsorkit/-/sponsorkit-0.8.9.tgz", + "integrity": "sha512-utofJeNDb/Kk0Q2Qmmla5gGuwHi054hGGefxDcLoiUlhcnyiN51zqdN24y9n6073S1+et7nCBeCokrhqGGBMCQ==", "dependencies": { - "consola": "^2.15.3", - "dotenv": "^16.0.3", - "fs-extra": "^11.1.0", + "consola": "^3.2.3", + "dotenv": "^16.3.1", + "fs-extra": "^11.1.1", "image-data-uri": "^2.0.1", - "node-html-parser": "^6.1.5", - "ohmyfetch": "^0.4.21", + "node-html-parser": "^6.1.10", + "ofetch": "^1.3.3", "picocolors": "^1.0.0", - "sharp": "^0.31.3", - "unconfig": "^0.3.7", - "yargs": "^17.7.1" + "sharp": "^0.32.5", + "unconfig": "^0.3.10", + "yargs": "^17.7.2" }, "bin": { "sponsorkit": "bin/sponsorkit.js" @@ -1581,6 +1661,15 @@ "node": ">=0.10.0" } }, + "node_modules/streamx": { + "version": "2.15.5", + "resolved": "https://registry.npmjs.org/streamx/-/streamx-2.15.5.tgz", + "integrity": "sha512-9thPGMkKC2GctCzyCUjME3yR03x2xNo0GPKGkRw2UMYN+gqWa9uqpyNWhmsNCutU5zHmkUum0LsCRQTXUgUCAg==", + "dependencies": { + "fast-fifo": "^1.1.0", + "queue-tick": "^1.0.1" + } + }, "node_modules/string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -1693,29 +1782,23 @@ } }, "node_modules/tar-fs": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz", - "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-3.0.4.tgz", + "integrity": "sha512-5AFQU8b9qLfZCX9zp2duONhPmZv0hGYiBPJsyUdqMjzq/mqVpy/rEUSeHk1+YitmxugaptgBh5oDGU3VsAJq4w==", "dependencies": { - "chownr": "^1.1.1", "mkdirp-classic": "^0.5.2", "pump": "^3.0.0", - "tar-stream": "^2.1.4" + "tar-stream": "^3.1.5" } }, "node_modules/tar-stream": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", - "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-3.1.6.tgz", + "integrity": "sha512-B/UyjYwPpMBv+PaFSWAmtYjwdrlEaZQEhMIBFNC5oEG8lpiW8XjcSdmEaClj28ArfKScKHs2nshz3k2le6crsg==", "dependencies": { - "bl": "^4.0.3", - "end-of-stream": "^1.4.1", - "fs-constants": "^1.0.0", - "inherits": "^2.0.3", - "readable-stream": "^3.1.1" - }, - "engines": { - "node": ">=6" + "b4a": "^1.6.4", + "fast-fifo": "^1.2.0", + "streamx": "^2.15.0" } }, "node_modules/tough-cookie": { @@ -1752,38 +1835,28 @@ "integrity": "sha512-ofhi8kjIje6npGozTip9Fr8iecmYfEbS06i0JnIg+rh51KakryWF4+jX8lLKZVhy6N+ID45WYSFCxPOdTWCzNg==" }, "node_modules/ufo": { - "version": "0.8.6", - "resolved": "https://registry.npmmirror.com/ufo/-/ufo-0.8.6.tgz", - "integrity": "sha512-fk6CmUgwKCfX79EzcDQQpSCMxrHstvbLswFChHS0Vump+kFkw7nJBfTZoC1j0bOGoY9I7R3n2DGek5ajbcYnOw==" + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/ufo/-/ufo-1.3.2.tgz", + "integrity": "sha512-o+ORpgGwaYQXgqGDwd+hkS4PuZ3QnmqMMxRuajK/a38L6fTpcE5GPIfrf+L/KemFzfUpeUQc1rRS1iDBozvnFA==" }, "node_modules/unconfig": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/unconfig/-/unconfig-0.3.9.tgz", - "integrity": "sha512-8yhetFd48M641mxrkWA+C/lZU4N0rCOdlo3dFsyFPnBHBjMJfjT/3eAZBRT2RxCRqeBMAKBVgikejdS6yeBjMw==", + "version": "0.3.11", + "resolved": "https://registry.npmjs.org/unconfig/-/unconfig-0.3.11.tgz", + "integrity": "sha512-bV/nqePAKv71v3HdVUn6UefbsDKQWRX+bJIkiSm0+twIds6WiD2bJLWWT3i214+J/B4edufZpG2w7Y63Vbwxow==", "dependencies": { - "@antfu/utils": "^0.7.2", + "@antfu/utils": "^0.7.6", "defu": "^6.1.2", - "jiti": "^1.18.2" + "jiti": "^1.20.0", + "mlly": "^1.4.2" }, "funding": { "url": "https://github.com/sponsors/antfu" } }, - "node_modules/undici": { - "version": "5.26.3", - "resolved": "https://registry.npmjs.org/undici/-/undici-5.26.3.tgz", - "integrity": "sha512-H7n2zmKEWgOllKkIUkLvFmsJQj062lSm3uA4EYApG8gLuiOM0/go9bIoC3HVaSnfg4xunowDE2i9p8drkXuvDw==", - "dependencies": { - "@fastify/busboy": "^2.0.0" - }, - "engines": { - "node": ">=14.0" - } - }, "node_modules/universalify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", - "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", "engines": { "node": ">= 10.0.0" } @@ -1894,9 +1967,9 @@ "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" }, "node_modules/yargs": { - "version": "17.7.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.1.tgz", - "integrity": "sha512-cwiTb08Xuv5fqF4AovYacTFNxk62th7LKJ6BL9IGUpTJrWoU7/7WdQGTP2SjKf1dUNBGzDd28p/Yfs/GI6JrLw==", + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", From 11fdc6a55884c76a22558f6abf53984d65fa7111 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Thu, 23 Nov 2023 22:14:48 +1100 Subject: [PATCH 50/87] chore: update sponsors.svg (#3072) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index cf3aeef58ba..cbc17a445b2 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -161,7 +161,7 @@ text {
- + From 929dfb41238bb76efad79183e914c9d9a6a384d5 Mon Sep 17 00:00:00 2001 From: "Ian M. Jones" Date: Sat, 25 Nov 2023 18:05:02 +0000 Subject: [PATCH 51/87] docs: add Snippet Expander to Showcase (#3079) * docs: add Snippet Expander to Showcase * Remove package-lock.json --- .../community/showcase/snippetexpander.mdx | 27 ++++++++++++++++++ .../snippetexpandergui-add-snippet.png | Bin 0 -> 37492 bytes .../snippetexpandergui-search-and-paste.png | Bin 0 -> 88231 bytes .../snippetexpandergui-select-snippet.png | Bin 0 -> 143641 bytes 4 files changed, 27 insertions(+) create mode 100644 website/docs/community/showcase/snippetexpander.mdx create mode 100644 website/static/img/showcase/snippetexpandergui-add-snippet.png create mode 100644 website/static/img/showcase/snippetexpandergui-search-and-paste.png create mode 100644 website/static/img/showcase/snippetexpandergui-select-snippet.png diff --git a/website/docs/community/showcase/snippetexpander.mdx b/website/docs/community/showcase/snippetexpander.mdx new file mode 100644 index 00000000000..1f9fb615796 --- /dev/null +++ b/website/docs/community/showcase/snippetexpander.mdx @@ -0,0 +1,27 @@ +# Snippet Expander + +```mdx-code-block +

+ +
+ Screenshot of Snippet Expander's Select Snippet window +

+

+ +
+ Screenshot of Snippet Expander's Add Snippet screen +

+

+ +
+ Screenshot of Snippet Expander's Search & Paste window +

+``` + +[Snippet Expander](https://snippetexpander.org) is "Your little expandable text snippets helper", for Linux. + +Snippet Expander comprises of a GUI application built with Wails for managing snippets and settings, with a Search & Paste window mode for quickly selecting and pasting a snippet. + +The Wails based GUI, go-lang CLI and vala-lang auto expander daemon all communicate with a go-lang daemon via D-Bus. The daemon does the majority of the work, managing the database of snippets and common settings, and providing services for expanding and pasting snippets etc. + +Check out the [source code](https://git.sr.ht/~ianmjones/snippetexpander/tree/trunk/item/cmd/snippetexpandergui/app.go#L38) to see how the Wails app sends messages from the UI to the backend that are then sent to the daemon, and subscribes to a D-Bus event to monitor changes to snippets via another instance of the app or CLI and show them instantly in the UI via a Wails event. diff --git a/website/static/img/showcase/snippetexpandergui-add-snippet.png b/website/static/img/showcase/snippetexpandergui-add-snippet.png new file mode 100644 index 0000000000000000000000000000000000000000..2a88cb75f64846d512ecd7e51da4a0b8531c9598 GIT binary patch literal 37492 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 Si|e}niITj!T!qZrkN*ceN}P@W literal 0 HcmV?d00001 diff --git a/website/static/img/showcase/snippetexpandergui-search-and-paste.png b/website/static/img/showcase/snippetexpandergui-search-and-paste.png new file mode 100644 index 0000000000000000000000000000000000000000..e197805fc376395549a2557b8c2d896e503794c2 GIT binary patch literal 88231 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 Sc!od5UUGttI3Kqs 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

+ +
+

+``` + +L'application [BulletinBoard](https://github.com/raguay/BulletinBoard) est un panneau de messages versitaux pour les messages statiques ou les boîtes de dialogue pour obtenir des informations de l'utilisateur pour un script. Il a une TUI pour créer de nouvelles boîtes de dialogue qui peuvent être utilisées pour obtenir des informations de l'utilisateur. Son design est de rester en fonctionnement sur votre système et de montrer les informations au besoin, puis de se cacher. J'ai un processus pour surveiller un fichier sur mon système et pour envoyer le contenu à BulletinBoard une fois modifié. Cela fonctionne très bien avec mes workflows. Il y a auss un [workflow Alfred](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) pour envoyer les informations au programme. Le workflow fonctionne aussi avec [EmailIt](https://github.com/raguay/EmailIt). diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx new file mode 100644 index 00000000000..ac64e25acce --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/) est un programme Wails 2 qui est un expéditeur de courrier électronique basé sur le markdown uniquement avec neuf blocs-notes, pour manipuler le texte et les modèles. Il a également un terminal pour exécuter des scripts dans EmailIt sur les fichiers de votre système. Les scripts et modèles peuvent être utilisés depuis la ligne de commande elle-même ou avec les extensions Alfred, Keyboard Maestro, Dropzone ou PopClip. Il supporte également les scripts et thèmes téléchargés sous GitHub. La documentation n'est pas complète, mais le programme fonctionne. Il est construit en utilisant Wails2 et Svelte, et le téléchargement est une application macOS universelle. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..7f4bd7a634a --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app) est un outil de chiffrement PGP simple et facile à utiliser, qui gère toutes vos clés et celles de vos contacts. Le chiffrement devrait être simple. Développé avec Wails.** + +Chiffrer les messages à l'aide de PGP est la norme de l'industrie. Tout le monde a une clé privée et publique. Votre clé privée, eh bien, doit être privée afin que vous seul puissiez lire les messages. Votre clé publique est distribuée à toute personne qui veut vous envoyer des messages secrets, chiffrés. Gérer les clés, chiffrer les messages et déchiffrer les messages devrait être une expérience agréable. EncryptEasy a pour but de vous simplifier la tâche. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx new file mode 100644 index 00000000000..7522afa6ecf --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# Utilitaire d'exportation FileHound + +```mdx-code-block +

+ +
+

+``` + +[L'utilitaire d'exportation FileHound](https://www.filehound.co.uk/) est une plate-forme de gestion de documents cloud conçue pour la conservation sécurisée de fichiers, l'automatisation des processus métier et les capacités de SmartCapture. + +L'utilitaire d'exportation FileHound permet aux administrateurs FileHound d'exécuter des tâches sécurisées d'extraction de documents et de données à des fins alternatives de sauvegarde et de récupération. Cette application téléchargera tous les documents et/ou métadonnées enregistrés dans FileHound en fonction des filtres que vous avez choisis. Les métadonnées seront exportées dans les formats JSON et XML. + +Backend construit avec: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +Frontend avec: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..21fd4b11743 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter) est un outil client de test d'API http simple et efficace. Basé sur les Wails, Go et sveltejs. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..5966e75fa63 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Minecraft Updater + +```mdx-code-block +

+ +
+

+``` + +[Minecraft Updater](https://github.com/Gurkengewuerz/MinecraftModUpdater) est un outil utilitaire pour mettre à jour et synchroniser les mods Minecraft pour votre base d'utilisateurs. Il a été conçu en utilisant Wails2 et React avec [antd](https://ant.design/) comme framework frontend. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..fe644bd7bb5 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Modal File Manager + +```mdx-code-block +

+ +
+

+``` + +[Modal File Manager](https://github.com/raguay/ModalFileManager) est un gestionnaire de fichiers à double volet utilisant des technologies web. Mon design original était basé sur NW.js et peut être trouvé [ici](https://github.com/raguay/ModalFileManager-NWjs). Cette version utilise le même code frontend basé sur Svelte (mais il a été grandement modifié depuis le départ de NW.js), mais le backend est une implémentation de [Wails 2](https://wails.io/). En utilisant cette implémentation, je n'utilise plus la ligne de commande `rm`, `cp`, etc. , mais une installation de git doit être présente sur le système pour télécharger des thèmes et des extensions. Il est entièrement codé en utilisant Go et fonctionne beaucoup plus rapidement que les versions précédentes. + +Ce gestionnaire de fichiers est conçu autour du même principe que Vim: l'état est contrôlé par des actions via le clavier. Le nombre d'états n'est pas fixe, mais très programmable. Par conséquent, un nombre infini de configurations de clavier qui peuvent être créées et utilisées. C'est la principale différence par rapport aux autres gestionnaires de fichiers. Il y a des thèmes et des extensions disponibles à télécharger à partir de GitHub. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..70a6cd1f44d --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molly Wallet](https://github.com/grvlle/constellation_wallet/) le portefeuille officiel $DAG du Constellation Network. Cela permettra aux utilisateurs d'interagir avec le réseau Hypergraph de différentes manières, sans se limiter à la production de transactions en $DAG. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx new file mode 100644 index 00000000000..5a9789d8711 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[Octobre](https://october.utf9k.net) est une petite application Wails qui rend vraiment facile d'extraire les surlignements de [Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader) puis de les transférer vers [Readwise](https://readwise.io). + +Il a une taille relativement petite avec toutes les versions de la plate-forme pesant en moins de 10 Mo, et c'est sans activer la [compression UPX](https://upx.github.io/)! + +En revanche, les précédentes tentatives de l'auteur avec Electron ont rapidement gonflé à plusieurs centaines de mégaoctets. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx new file mode 100644 index 00000000000..41744234d37 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# Optimus + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus) est une application d'optimisation d'image de bureau. Il supporte la conversion et la compression entre les formats d’images WebP, JPEG et PNG. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx new file mode 100644 index 00000000000..acdc682becf --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# Portfall + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - Un portail de redirection de port k8 pour un accès facile à toutes les interfaces de votre instance diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..b6597166852 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - Une interface de sauvegarde simple et multiplateforme [restic](https://github.com/restic/restic) pour la navigation et la restauration de dépôts restic. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..e47b2397e63 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +Partage de fichiers facile, sécurisé et gratuit pour tout le monde. Apprenez-en plus sur [Riftshare.app](https://riftshare.app) + +## Fonctionnalités + +- Partage facile et sécurisé de fichiers entre ordinateurs à la fois sur le réseau local et via Internet +- Supporte l'envoi de fichiers ou de répertoires de manière sécurisée par le protocole [magic wormhole](https://magic-wormhole.readthedocs.io/en/latest/) +- Compatible avec toutes les autres applications utilisant magic wormhole (magic-wormhole or wormhole-william CLI, wormhole-gui, etc.) +- Compression automatique de plusieurs fichiers sélectionnés à envoyer en même temps +- Animations complètes, barre de progression et support d'annulation pour l'envoi et la réception +- Sélection de fichier natif au système d'exploitation +- Ouvrir les fichiers en un seul clic une fois reçus +- Mise à jour automatique - ne vous inquiétez pas d'avoir la dernière version! diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..d7215d66126 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) est un programme pour afficher la sortie de scripts ou d'un serveur [Node-Red](https://nodered.org). Il exécute des scripts définis dans le programme EmailIt et affiche la sortie. Des scripts de xBar ou TextBar peuvent être utilisés. Actuellement sur les scripts TextBar fonctionnent bien. Il affiche également la sortie des scripts sur votre système. ScriptBar ne les met pas dans la barre de menus, mais les a tous dans une fenêtre convenable pour une visualisation facile. Vous pouvez avoir plusieurs onglets pour voir plusieurs choses différentes. Vous pouvez également conserver les liens vers vos sites Web les plus visités. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx new file mode 100644 index 00000000000..b5917484303 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/) est une application de partage de fichiers p2p conçue pour utiliser les technologies blockchain afin d'activer les transferts de fichiers 100 % anonymes. Surge est chiffré de bout en bout, décentralisé et open source. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx new file mode 100644 index 00000000000..ba2a7fccc45 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally) est le flasheur officiel du firmware pour les claviers [Ergodox](https://ergodox-ez.com/). C'est un excellent exemple de ce que vous pouvez réaliser avec Wails : la capacité de combiner la puissance de Go et les riches outils graphiques du monde du développement web. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx new file mode 100644 index 00000000000..2e427433069 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# Lanceur Minecraft pour WarMine + +```mdx-code-block +

+ + +
+

+``` + +[Lanceur Minecraft pour WarMine](https://warmine.ru/) est une application Wails qui vous permet facilement de rejoindre le serveur de jeu contenant les mods, ainsi que la gestion de vos comptes de jeu. + +Le Launcher télécharge les fichiers du jeu, vérifie leur intégrité et lance le jeu avec une large gamme d'options de personnalisation. + +Le frontend est écrit en Svelte, le lanceur entier tient dans 9Mo et prend en charge Windows 7-11. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx new file mode 100644 index 00000000000..c431f691952 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# Wombat + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat) est un client gRPC multi-plateforme. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx new file mode 100644 index 00000000000..4f5bd993cb1 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails) est une application pour télécharger des pistes depuis youtube, créer des listes de lecture hors ligne et les partager avec vos amis, vos amis seront en mesure de lire vos playlists ou de les télécharger pour l'écoute hors ligne, a un lecteur intégré. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx new file mode 100644 index 00000000000..6baf0941b8c --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# Modèles + +Cette page sert de liste pour les modèles supportés par la communauté. Veuillez soumettre une PR (cliquez sur `Modifier cette page` en bas) pour inclure vos modèles. Pour construire votre propre modèle, veuillez consulter le guide [Modèles](../guides/templates.mdx). + +Pour utiliser ces modèles, exécutez `wails init -n "Votre nom de projet" -t [le lien ci-dessous[@version]]` + +S'il n'y a pas de suffixe de version, la branche principale du modèle de code sera alors utilisé par défaut. S'il y a un suffixe de version, le modèle de code correspondant au tag de cette version sera utilisé. + +Exemple : `wails init -n "Votre nom de projet" -t https://github.com/misitebao/wails-template-vue` + +:::warning Attention + +**Le projet Wails n'entretient pas, et n'est pas responsable des modèles de tierces parties!** + +Si vous n'êtes pas sûr d'un modèle, inspectez `package.json` et `wails.json` pour savoir quels scripts sont exécutés et quels paquets sont installés. + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - Modèle de Wails basé sur Vue (TypeScript intégré, thème sombre, internationalisation, routage de page unique, TailwindCSS) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Vue 3 TypeScript avec Vite (et instructions pour ajouter des fonctionnalités) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vue 3 TypeScript avec Vite, Vuex, Vue Router, Sass, et ESLint + Prettier +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - Un modèle utilisant JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - Un modèle utilisant TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, Composition API avec <script setup>) +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Modèle Wails basé sur Naive UI (Librairie de composants Vue 3) + +## Angular + +- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - Modèle Angular 15+ prêt à être utilisé en production. +- [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - Angular avec TypeScript, Sass, rechargement à chaud, découpage dynamique de code et i18n + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - Un modèle utilisant reactjs +- [wails-react-template](https://github.com/flin7/wails-react-template) - Un modèle minimal pour React qui supporte le développement en direct +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - Un modèle utilisant Next.js et TypeScript +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - Un modèle pour React + TypeScript + Vite + TailwindCSS +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - Un modèle avec Vite, React, TypeScript, TailwindCSS, et shadcn/ui + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - Un modèle utilisant Svelte +- [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - Un modèle utilisant Svelte et Vite +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - Un modèle utilisant Svelte et Vite avec TailwindCSS v3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - Un modèle mis à jour en utilisant Svelte v4.2.0 et Vite avec TailwindCSS v3.3.3 +- [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - Un modèle utilisant SvelteKit + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - Un modèle utilisant Solid + Ts + Vite +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-js) - Un modèle utilisant Solid + Js + Vite + +## Elm + +- [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Développez votre application GUI avec de la programmation fonctionnelle et une configuration de développement en direct :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine les puissances :muscle: d'Elm + Tailwind CSS + Wails ! Rechargement automatique pris en charge. + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Utilisez une combinaison unique de htmx pour interactivité, et de templ pour créer des composants et des formes + +## Pure JavaScript (Vanilla) + +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - Un modèle avec rien que du JavaScript, du HTML et du CSS de base diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx new file mode 100644 index 00000000000..eb43f4aae2c --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# Compiler votre projet + +À partir du répertoire du projet, exécutez `wails build`. Cela compilera votre projet et sauvegardera le binaire prêt à la production dans le répertoire `build/bin`. + +Si vous exécutez le binaire, vous devriez voir l'application par défaut : + +```mdx-code-block +
+ +
+
+``` + +Pour plus de détails sur les options de compilation, veuillez vous référer à la [documentation du CLI](../reference/cli.mdx#build). diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx new file mode 100644 index 00000000000..307029141d4 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# Développez votre application + +Vous pouvez exécuter votre application en mode développement en exécutant `wails dev` à partir du répertoire de votre projet. Cela fera les choses suivantes : + +- Construire votre application et l'exécuter +- Lier votre code Go au frontend pour qu'il puisse être appelé à partir de JavaScript +- En utilisant la puissance de [Vite](https://vitejs.dev/), surveillera les modifications dans vos fichiers Go et reconstruira / ré-exécutera en cas de changement +- Mettra en place un [serveur web](http://localhost:34115) qui servira votre application via un navigateur. Cela vous permet d'utiliser les extensions de votre navigateur préféré. Vous pouvez même appeler votre code Go depuis la console + +Pour commencer, exécutez `wails dev` dans le répertoire du projet. Plus d'informations à ce sujet peuvent être trouvées [ici](../reference/cli.mdx#dev). + +Prochainement : Tutoriel diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..383925af1e6 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# Créer un projet + +## Génération de projet + +Maintenant que le CLI est installé, vous pouvez générer un nouveau projet en utilisant la commande `wails init`. + +Choisissez votre framework favori : + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Générer un projet
Svelte utilisant JavaScript avec:

+ + wails init -n myproject -t svelte + +Si vous préférez utiliser TypeScript:
+ + wails init -n myproject -t svelte-ts + + + + Générer un projet React utilisant JavaScript avec :

+ + wails init -n myproject -t react + +Si vous préférez utiliser TypeScript:
+ + wails init -n myproject -t react-ts + +
+ + Générer un projet Vue utilisant JavaScript avec:

+ + wails init -n myproject -t vue + +Si vous préférez TypeScript:
+ + wails init -n myproject -t vue-ts + +
+ + Générer un projet Preact utilisant JavaScript avec:

+ + wails init -n myproject -t preact + +Si vous préférez TypeScript:
+ + wails init -n myproject -t preact-ts + +
+ + Générer un projet Lit utilisant JavaScript avec:

+ + wails init -n myproject -t lit + +Si vous préférez TypeScript:
+ + wails init -n myproject -t lit-ts + +
+ + Générer un projet Vanilla utilisant JavaScript avec :

+ + wails init -n myproject -t vanilla + +Si vous préférez TypeScript:
+ + wails init -n myproject -t vanilla-ts + +
+ +``` + +
+ +Il y a aussi [des modèles créés par la communauté](../community/templates.mdx) qui sont disponibles et qui offrent différentes possibilités. + +Pour voir les autres options disponibles, vous pouvez exécuter `wails init -help`. Plus de détails peuvent être trouvés dans la [documentation du CLI](../reference/cli.mdx#init). + +## Structure du projet + +Les projets Wails ont la structure suivante: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### Récapitulatif de la structure du projet + +- `/main.go` - L'application principale +- `/frontend/` - Fichiers de la partie frontend +- `/build/` - Répertoire de construction du projet +- `/build/appicon.png` - L'icône de l'application +- `/build/darwin/` - Fichiers spécifiques pour Mac +- `/build/windows/` - Fichiers spécifiques pour Windows +- `/wails.json` - La configuration du projet +- `/go.mod` - Le fichier du module Go +- `/go.sum` - Le checksum du fichier du module Go + +Le répertoire `frontend` n'a rien de spécifique à Wails et n'importe quel outil de frontend peut être utilisé. + +Le répertoire `build` est utilisé pendant le processus de compilation. Ces fichiers peuvent être mis à jour pour personnaliser vos builds. Si fichiers sont supprimés du répertoire de compilation, les versions par défaut seront régénérées. + +Le nom du module par défaut dans `go.mod` est "changeme". Vous devriez changer cela pour quelque chose de plus approprié. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx new file mode 100644 index 00000000000..86ffde34bde --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx @@ -0,0 +1,90 @@ +--- +sidebar_position: 1 +--- + +# Installation + +## Plates-formes Prises en charge + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## Dépendances + +Wails a un certain nombre de dépendances communes qui sont nécessaires avant l'installation : + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +Télécharger Go à partir de la [Page de téléchargement](https://go.dev/dl/). + +Assurez-vous que vous suivez les instructions officielles de [l'installation de Go](https://go.dev/doc/install). Vous devrez également vous assurer que votre variable d'environnement `PATH` inclut également le chemin vers votre répertoire `~/go/bin`. Redémarrez votre terminal et effectuez les vérifications suivantes : + +- Vérifiez que Go est installé correctement : `go version` +- Vérifiez que "~/go/bin" est dans votre variable PATH : `echo $PATH | grep go/bin` + +### NPM + +Téléchargez le NPM à partir de la [page de téléchargement de Node](https://nodejs.org/en/download/). Il est préférable d'utiliser la dernière version car c'est avec celle-là que nous effectuons nos tests. + +Exécutez `npm --version` pour vérifier. + +## Dépendances spécifiques aux plateformes + +Vous devrez également installer des dépendances spécifiques liés à la plateforme que vous utilisez : + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wails a besoin que les outils de command line xocde soient installés. Cela peut être fait + en exécutant xcode-select --install. + + + Wails a besoin que WebView2 runtime soit installé. Certaines installations de Windows auront déjà installé cette fonctionnalité. Vous pouvez vérifier en utilisant la commande wails doctor. + + + Linux a besoin de gcc comme outil de compilation en plus de libgtk3 et libwebkit. Plutôt que de lister une tonne de commandes pour différentes distributions, Wails peut essayer de déterminer ce que sont les commandes d'installation pour votre distribution. Exécutez wails doctor après l'installation pour voir de quelles dépendances vous avez besoin. Si votre gestionnaire de distribution/paquet n'est pas pris en charge, veuillez consulter le guide Ajouter une distribution Linux. + + +``` + +## Dépendances optionnelles + +- [UPX](https://upx.github.io/) pour compresser vos applications. +- [NSIS](https://wails.io/docs/guides/windows-installer/) pour générer des installateurs Windows. + +## Installer Wails + +Exécutez `go go install github.com/wailsapp/wails/v2/cmd/wails@latest` pour installer le CLI. + +Note: Si vous obtenez une erreur similaire à ceci: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +vérifiez que vous avez installé Go 1.18+ : +```shell +go version +``` + +## Vérification du système + +Exécuter `wails doctor` qui vérifiera si vous avez les bonnes dépendances installées. Si ce n'est pas le cas, il vous conseillera sur ce qui manque et vous aidera à corriger tout problème. + +## La commande `wails` semble manquer ? + +Si votre système signale que la commande `wails` est manquante, assurez-vous que vous avez suivi le guide d'installation correctement. Normalement, cela signifie que le répertoire `go/bin` du répertoire racine de votre utilisateur n'est pas dans la variable d'environnement `PATH` . Vous devrez également normalement fermer et réouvrir toutes les commandes ouvertes afin que les modifications apportées à l'environnement par l'installateur soient reflétées dans l'invite de commande. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx new file mode 100644 index 00000000000..1fe2f199ffe --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +Bien que Wails n'ait pas de modèle Angular, il est possible d'utiliser Angular avec Wails. + +## Dev Mode + +Pour que le mode développeur fonctionne avec Angular, vous devez ajouter ce qui suit à votre fichier `wails.json`: + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx new file mode 100644 index 00000000000..411853fd6f1 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx @@ -0,0 +1,215 @@ +# Développement d'applications + +Il n'y a pas de règles gravées dans le marbre pour le développement d'applications avec Wails, mais il y a quelques lignes directrices de base. + +## Configuration de l'application + +Le modèle utilisé par défaut défini que `main.go` est utilisé pour configurer et démarrer l'application, tandis que `app.go` est utilisé pour définir la logique de l'application. + +Le fichier `app.go` va définir une structure qui a 2 méthodes qui agissent comme crochets dans l'application principale: + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- La méthode startup est appelée d-s que Wails a donné les ressources nécessaires et qu'il est dans un bon état pour créer les ressources, mettre en place les event listeners et tout ce dont l'application peut avoir besoin pour démarrer. Il est donné un `context.Context` qui est généralement sauvegardé dans un champ struct. Ce contexte est nécessaire pour appeler le [runtime](../reference/runtime/intro.mdx). Si cette méthode renvoie une erreur, l'application se fermera. En mode développement, l'erreur sera affichée dans la console. + +- La méthode d'arrêt sera appelée par Wails à la fin du processus d'arrêt. C'est un bon endroit pour vider la mémoire et effectuer toutes les tâches d'arrêt. + +Le fichier `main.go` consiste généralement en un seul appel à `wails.Run()`, qui accepte la configuration de l'application. Le modèle utilisé par les templates fait qu'avant l'appel à `wails.Run()`, une instance du struct que l'on a définie dans `app.go` est créée et instanciée dans une variable appelée `app`. Cette configuration est l'endroit où nous ajoutons nos callbacks : + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +Plus d'informations sur les crochets du cycle de vie des applications peuvent être trouvées [ici](../howdoesitwork.mdx#application-lifecycle-callbacks). + +## Méthodes de liaison + +Il est probable que vous vouliez appeler les méthodes Go depuis le frontend. Cela se fait normalement en ajoutant des méthodes publiques à le struct déjà défini dans `app.go`: + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +Dans la configuration principale de l'application, le paramètre `Bind` est l'endroit où nous pouvons dire à Wails ce que nous voulons lier : + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +Cela liera toutes les méthodes publiques de notre structure `App` (cela ne liera jamais les méthodes de démarrage et d'arrêt du système). + +### Traiter avec le contexte lors de la liaison de plusieurs structures + +Si vous voulez lier des méthodes pour des structures multiples, mais que vous voulez que chaque struct conserve une référence au contexte pour que vous puissiez utiliser les fonctions d'exécution... Un bon choix est de passer le contexte de la méthode `OnStartup` à vos instances struct : + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +Plus d'informations à sur Binding peuvent être trouvées [ici](../howdoesitwork.mdx#method-binding). + +## Menu de l’application + +Wails prend en charge l'ajout d'un menu à votre application. Ceci est fait en passant un [Menu](../reference/menus.mdx#menu) structuré à la configuration de l'application. Il est courant d'utiliser une méthode qui renvoie un Menu, et encore plus courant pour que cela soit une méthode sur la struct de l'`app` qui soit utilisée pour les hooks du cycle de vie. + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## Ressources + +La grande chose à propos de la façon dont Wails v2 gère les ressources pour le frontend, est que ce n'est pas le cas! La seule chose que vous devez donner à Wails est un `embed.FS`. C'est à vous de décider comment vous y arrivez. Vous pouvez utiliser les fichiers html/css/js vanilla comme dans le modèle vanilla. Vous pourriez avoir un système de compilation compliqué, peu importe. + +Quand la commande `wails dev` est exécutée, elle vérifiera le fichier de projet `wails.json` à la racine du projet. Il y a 2 clés dans le fichier du projet qui sont lues : + +- "frontend:install" +- "frontend:build" + +Le premier, si fourni, sera exécuté dans le répertoire `frontend` pour installer les modules. Le second, si fourni, sera exécuté dans le répertoire `frontend` pour construire le projet frontend. + +Si ces 2 clés ne sont pas fournies, alors Wails ne fait absolument rien avec le frontend. Il n'attend que `embed.FS`. + +### AssetsHandler + +Une application Wails v2 peut éventuellement définir un `http.Handler` dans `options.app`, qui permet de se connecter à l'AssetServer pour créer des fichiers à la volée ou traiter les requêtes POST/PUT. Les requêtes GET sont toujours traitées d'abord par le `assets` FS. Si le FS ne trouve pas le fichier demandé, la requête sera transmise au `http.Handler`. Toute requête autre que GET sera traitée directement par le `AssetsHandler` si spécifié. Il est également possible d'utiliser le `AssetsHandler` uniquement en spécifiant `nil` dans l'option `Assets`. + +## Serveur de développement embarqué + +Exécuter `wails dev` démarrera le serveur de développement intégré qui démarrera un observateur de fichiers dans votre répertoire de projet. Par par défaut, si un fichier change, wails vérifie s'il s'agit d'un fichier d'application (par défaut: `.go`, configurable avec l'option `-e`). Si c'est le cas, il reconstruira votre application et la relancera. Si le fichier modifié se trouvait dans les actifs, il lancera un rechargement après un court laps de temps. + +Le serveur de développement utilise une technique appelée "debouncing", ce qui signifie qu'il ne se recharge pas tout de suite, comme il peut y avoir plusieurs fichiers modifiés en un court laps de temps. Lorsqu'un déclencheur se produit, il attend un temps défini avant d'émettre un rechargement. Si un autre déclencheur se produit, le temps d'attente se réinitialise avant un prochain rechargement. Par défaut, cette période est définie à `100ms`. Si cette valeur ne fonctionne pas pour votre projet, elle peut être configurée en utilisant l'option `-debounce`. Si elle est utilisée, cette valeur sera enregistrée dans la configuration de votre projet et deviendra la valeur par défaut. + +## Serveur de développement externe + +Certains frameworks sont fournis avec leur propre serveur de rechargement en direct, cependant ils ne seront pas en mesure de tirer parti des liaisons Wails Go. Dans ce scénario, il est préférable d'exécuter un script qui va surveiller le projet dans dossier build, dossier que Wails surveille aussi. Pour un exemple, voir le modèle svelte par défaut qui utilise [rollup](https://rollupjs.org/guide/en/). + +### Créer une application React + +Le processus pour créer un projet Reactest un peu plus compliqué. Afin de prendre en charge le rechargement du frontend en direct, la configuration suivante doit être ajoutée à votre `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +La commande `frontend:dev:watcher` démarrera le serveur de développement React (hébergé sur le port `3000` typiquement). La commande `frontend:dev:serverUrl` demande ensuite à Wails d'exposer les ressources depuis le serveur de développement lors du chargement du frontend, plutôt que depuis le dossier de construction. En plus de ce qui précède, le fichier `index.html` doit être mis à jour avec les éléments suivants : + +```html + + + + + +``` + +Ceci est nécessaire, car la commande watcher qui reconstruit le frontend empêche Wails de les injecter. Ça contourne le problème en assurant les scripts sont toujours injectés. Avec cette configuration, `wails dev` peut être exécuté, ce qui construira le frontend et le backend de manière appropriée avec le rechargement à chaud activé. De plus, lorsque vous accédez à l'application à partir d'un navigateur, les outils de développement de React peuvent maintenant être utilisés sur une version non minifiée de l'application pour le débogage. Enfin, pour des compilations plus rapides, `wails dev -s` peut être exécuté pour passer la construction par défaut du frontend par Wails car c'est une étape inutile. + +## Module Go + +Les modèles Wails par défaut génèrent un fichier `go.mod` qui contient le nom de module "changeme". Vous devriez changer ceci pour quelque chose de plus approprié après la génération du projet. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..97f81432dc2 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# Ressources dynamiques + +Si vous voulez charger ou générer des ressources pour votre frontend de manière dynamique, vous pouvez y parvenir en utilisant l'option [AssetsHandler](../reference/options#assetshandler). Le AssetsHandler est un générique`http.Handler` qui sera appelé pour toute requête non GET sur le serveur d'assets et pour les requêtes GET qui ne peuvent pas être servies car l'asset n'est pas trouvé. + +En installant un AssetsHandler personnalisé, vous pouvez servir vos propres ressources en utilisant un serveur de ressources personnalisé. + +## Exemple + +Dans notre exemple de projet, nous allons créer un gestionnaire de ressources simple qui chargera les fichiers à partir du disque: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +Lorsque nous exécutons l'application en mode dev en utilisant `wails dev`, nous verrons la sortie suivante : + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +Comme vous pouvez le voir, le gestionnaire d'actifs est appelé lorsque le serveur d'assets par défaut est incapable de servir le fichier `favicon.ico`. + +Si vous faites un clic droit sur l'application principale et sélectionnez "inspecter" pour afficher les devtools, vous pouvez tester cette fonctionnalité en tapant ce qui suit dans la console : + +``` +let response = await fetch('does-not-exist.txt'); +``` + +Cela générera une erreur dans les devtools. Nous pouvons voir que l'erreur est ce que nous attendons est retourné par notre gestionnaire de ressources personnalisées : + +```mdx-code-block +

+ +

+``` + +Cependant, si nous demandons `go.mod`, nous verrons la sortie suivante : + +```mdx-code-block +

+ +

+``` + +Cette technique peut être utilisée pour charger des images directement dans la page. Si nous avons mis à jour notre modèle vanilla par défaut et a remplacé l'image du logo : + +```html + +``` + +avec : + +```html + +``` + +Nous verrions ensuite ce qui suit: + +```mdx-code-block +

+ +

+``` + +:::warning + +Exposer votre système de fichiers de cette manière est un risque de sécurité. Il est recommandé de gérer correctement l'accès à votre système de fichiers. + +::: diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..167955b12d7 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,244 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx new file mode 100644 index 00000000000..bbeb338f1b6 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx @@ -0,0 +1,87 @@ +# Applications sans cadre + +Wails prend en charge la création d'applications qui n'ont pas de cadres. Ceci peut être réalisé en utilisant le champ [frameless](../reference/options.mdx#frameless) dans [Application Options](../reference/options.mdx#application-options). + +Wails offre une solution simple pour faire glisser la fenêtre: N'importe quel élément HTML qui a le style CSS `--wails-draggable:drag` agira comme une "poignée de glisser". Cette propriété s'applique à tous les éléments enfants. Si vous devez indiquer qu'un élément imbriqué ne doit pas glisser, alors utilisez l'attribut '--wails-draggable:no-drag' sur cet élément. + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +Pour certains projets, l'utilisation d'une variable CSS peut ne pas être possible en raison du style dynamique. Dans ce cas, vous pouvez utiliser les options `CSSDragProperty` et `CSSDragValue` pour définir une propriété et une valeur qui seront utilisées pour indiquer régions glissables : + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info Plein écran + +Si vous autorisez votre application à être en plein écran, cette fonctionnalité de glissement sera désactivée. + +::: diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx new file mode 100644 index 00000000000..ea101019ad8 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx @@ -0,0 +1,72 @@ +# Frontend + +## Injection de script + +Quand Wails sert votre fichier `index.html`, par défaut, il injectera 2 entrées de script dans la balise `` pour charger `/wails/ipc.js` et `/wails/runtime.js`. Ces fichiers installent respectivement les bindings et les runtime. + +Le code ci-dessous montre où ils sont injectés par défaut : + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### Remplacer l'injection de script par défaut + +Pour fournir plus de flexibilité aux développeurs, il y a une balise meta qui peut être utilisée pour personnaliser ce comportement: + +```html + +``` + +Les options sont les suivantes : + +| Valeur | Description | +| ------------------- | -------------------------------------------------------------- | +| noautoinjectruntime | Pour désactiver l'injection automatique de `/wails/runtime.js` | +| noautoinjectipc | Pour désactiver l'injection automatique de `/wails/ipc.js` | +| noautoinject | Pour désactiver l'injection automatique de tous les scripts | + +Plusieurs options peuvent être utilisées à condition qu'elles soient séparées par des virgules. + +Ce code est parfaitement valide et fonctionne de la même manière que la version avec l'auto-injection : + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx new file mode 100644 index 00000000000..f35b2f57f77 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx @@ -0,0 +1,127 @@ +# IDEs + +Wails vise à fournir une grande expérience de développement. À cet effet, nous supportons maintenant la génération d'une configuration spécifique IDE pour fournir une configuration plus souple du projet. + +Actuellement, nous prenons en charge [Visual Studio Code](https://code.visualstudio.com/) mais nous visons à prendre en charge d'autres IDE comme Goland. + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +Lors de la génération d'un projet en utilisant l'option `-ide vscode` , les fichiers IDE seront créés à côté des autres fichiers du projet. Ces fichiers sont placés dans le répertoire `.vscode` et fournissent la configuration correcte pour déboguer votre application. + +Les 2 fichiers générés sont `tasks.json` et `launch.json`. Ci-dessous se trouvent les fichiers générés par défaut : + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### Configuration des étapes d'installation et de construction + +Le fichier `tasks.json` est simple pour le projet par défaut car il n'y a pas d'étapes `npm install` ou `npm build` nécessaire. Pour les projets qui ont une étape de construction en frontend comme avec Svelte, nous devrions modifier `tasks.json` pour ajouter les étapes d'installation et de construction suivantes : + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +:::info Améliorations futures + +Dans le futur, nous espérons générer un `tasks.json` qui inclut les étapes d'installation et de construction automatiquement. + +::: diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..6bd2002fce7 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Prise en charge des distributions Linux + +## Vue d'ensemble + +Wails offre le support de Linux, mais fournir des instructions d'installation pour toutes les distributions disponibles est une tâche impossible. À la place, Wails essaie de déterminer si les paquets dont vous avez besoin pour développer des applications sont disponibles via le gestionnaire de paquets de votre système. Actuellement, nous supportons les gestionnaires de paquets suivants : + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## Ajout des noms de paquets + +Il peut y avoir des cas où votre distribution de linux utilise un des gestionnaires de paquets pris en charge mais le nom du paquet est différent. Par exemple, vous pouvez utiliser un dérivé Ubuntu, mais le nom du paquet pour gtk peut être différent. Wails tente de trouver le paquet correct en itérant une liste de noms de paquets. La liste des paquets est stockée dans un fichier spécifique dans le dossier `v2/internal/system/packagemanager` . Dans notre exemple, ce serait `v2/internal/system/packagemanager/apt.go`. + +Dans ce fichier, la liste des paquets est définie par la méthode `Packages()`: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +Supposons que dans notre distribution linux, `libgtk-3` est empaqueté sous le nom `lib-gtk3-dev`. Nous pourrions ajouter le support de ce paquet en ajoutant la ligne suivante : + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## Ajout de nouveaux gestionnaires de paquets + +Pour ajouter un nouveau gestionnaire de paquets, effectuez les étapes suivantes : + +- Créez un nouveau fichier dans `v2/internal/system/packagemanager` appelé `.go`, où `` est le nom du gestionnaire de paquets. +- Définit une structure conforme à l'interface du gestionnaire de paquets définie dans `pm.go`: + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()` doit retourner le nom du gestionnaire de paquets +- `Packages()` doit retourner une `packagemap`, qui fournit des noms de fichiers candidats pour les dépendances +- `PackageInstalled()` devrait retourner `true` si le paquet donné est installé +- `PackageAvailable()` devrait retourner `true` si le paquet donné n'est pas installé mais disponible pour l'installation +- `InstallCommand()` doit retourner la commande exacte pour installer le nom du paquet donné + +Jetez un coup d'œil au code des autres gestionnaires de paquets pour avoir une idée de comment cela fonctionne. + +:::info Rappel + +Si vous ajoutez le support d'un nouveau gestionnaire de paquets, n'oubliez pas de mettre également à jour cette page ! + +::: diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx new file mode 100644 index 00000000000..19e45313a44 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +Cette page a divers guides liés au développement d'applications Wails pour Linux. + +## Le tag vidéo ne déclenche pas l'événement "terminé" + +Lorsque vous utilisez un tag vidéo, l'événement "terminé" n'est pas déclenché lorsque la vidéo est finie. Ceci est un bogue dans WebkitGTK, cependant vous pouvez utiliser le contournement suivant pour le corriger : + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +Source : [Lyimmi](https://github.com/Lyimmi) sur le [forum de discussion](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx new file mode 100644 index 00000000000..0c3878d7888 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx @@ -0,0 +1,55 @@ +# Développement local + +## Vue d'ensemble + +Wails est en développement constant et les nouvelles versions sont régulièrement "tagguées". Cela se produit généralement lorsque tout le nouveau code sur `master` a été testé et confirmé fonctionnel. Si vous avez besoin d'un correctif ou d'une fonctionnalité qui ne l'a pas encore fait pour une version, il est possible d'utiliser la dernière version "non validée" en utilisant les étapes suivantes : + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +REMARQUE : Le répertoire dans lequel vous avez cloné le projet sera maintenant appelé "clonedir". + +Le CLI de Wails sera maintenant à la dernière version. + +### Mise à jour du projet + +Pour mettre à jour vos projets pour utiliser la dernière version de la bibliothèque Wails, mettez à jour le fichier `go.mod` et assurez-vous que la ligne suivante est en bas du fichier : + +`replace github.com/wailsapp/wails/v2 => ` + +Exemple: + +Sur Windows: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +Sur 'nix: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +Pour revenir à une version stable, exécutez : + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## Tester une branche + +Si vous voulez tester une branche, suivez les instructions ci-dessus, mais assurez-vous de bien vous mettre sur la branche que vous voulez tester avant d'installer : + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +Assurez-vous de [mettre à jour votre projet](#updating-your-project) comme décrit ci-dessus. + +## Tester une PR + +Si vous voulez tester une PR, suivez les instructions ci-dessus, mais assurez-vous de récupérer la PR et d'être sur la branche de la PR avant de faire l'installation. Veuillez remplacer `[IDofThePR]` par l'ID de la PR affiché sur github.com: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +Assurez-vous de [mettre à jour votre projet](#updating-your-project) comme décrit ci-dessus. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx new file mode 100644 index 00000000000..fe040591634 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Guide pour Mac App Store + +Cette page donne un bref aperçu de la façon de soumettre votre application Wails au Mac App Store. + +## Prérequis + +- Vous devrez avoir un compte développeur Apple. Veuillez trouver plus d'informations sur le site [Apple Developer Program](https://developer.apple.com/support/compare-memberships/) +- Vous aurez besoin que vos certificats, identifiants et applications soient créés sur le portail développeur. Plus d'infos sur ce sujet ci-dessous +- L'utilitaire Xcode devront être installés sur votre machine locale pour être utilisé en ligne de commandes + +#### Créer des certificats et des identifiants + +1. Allez sur votre [Compte Développeur Apple](https://developer.apple.com/account/) +2. Sous `Certificats, Identificateurs & Profils`, cliquez sur `Identifiants` et Enregistrez un nouvel identifiant d'application. Utiliser le format (com.example.app) +3. Sous la même page, cliquez sur `Certificats` et générez de nouveaux certificats pour la distribution de l'App Store Mac. Téléchargez-les et importez les certificats dans votre trousseau sur votre machine locale. + +#### Créer une soumission d'application + +1. Allez sur le [site de connexion de l'App Store](https://appstoreconnect.apple.com/apps) +2. Enregistrez une nouvelle application et liez l'ID du lot que vous avez créé à l'étape précédente +3. Remplissez votre application avec les bonnes captures d'écran, descriptions, etc. selon les besoins d'Apple +4. Créer une nouvelle version de votre application + +#### Create Provisioning Profile +1. Go to the [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) page +2. Add a new provisioning profile for Mac App Store Distribution +3. Set the Profile Type as Mac and select the App ID for the application created above +4. Select the Mac App Distribution certificate +5. Name the Provisioning Profile embedded and download the created profile. + +## Processus Mac App Store + +#### Activation du Sandbox Apple + +Les applications soumises au Mac App Store doivent tourner dans la [Sandbox](https://developer.apple.com/app-sandboxing/) Apple. Vous devez créer un fichier `entitlements.plist` pour que cela fonctionne. La recommandation est de créer ce fichier sous ce chemin `{PROJECT_DIR}/build/darwin/entitlements.plist`. + +**Example de fichier Entitlements** + +Ceci est un exemple du fichier entitlements de l'application [RiftShare](https://github.com/achhabra2/riftshare). Pour référence, veuillez mettre dans les droits requis par votre application. Reportez-vous à [ce site](https://developer.apple.com/documentation/bundleresources/entitlements) pour plus d'informations. You will need to replace the Team ID and Application Name with the ones you registered above. + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +**Add the Embedded Provisioning Profile** The Provisioning Profile created above needs to be added to the root of the applicaton. It needs to be named embedded.provisionprofile. + +#### Construire et signer le package de l'application + +Ce qui suit est un exemple de script pour construire et signer votre application pour la soumission de l'App Store Mac. Il suppose que vous exécutez le script depuis la racine de votre projet. + +Notez que les certificats pour signer l'application et l'installateur sont différents. Veuillez vous assurer que les deux sont importés dans votre trousseau. Trouvez les chaînes de caractères dans Trousseau et insérez-les ci-dessous. Remplissez le nom de votre certificat et le nom de l'application ci-dessous. Exécuter le script suivant générera un fichier `app.pkg` signé à la racine de votre application. + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### Télécharger l'application + +Vous devrez télécharger le fichier de package généré et l'associer à votre application avant de pouvoir le soumettre pour vérification. + +1. Téléchargez l' [App Transporter](https://apps.apple.com/us/app/transporter/id1450874784) depuis le Mac App Store +2. Ouvrez-le et connectez-vous avec votre identifiant Apple +3. Cliquez sur le signe + et sélectionnez le fichier `APP_NAME.pkg` que vous avez généré à l'étape précédente. Télécharger le +4. Retournez sur le site [App Store Connect](https://appstoreconnect.apple.com/apps) et retournez dans la soumission de votre application. Sélectionnez la version que vous êtes prêt à mettre à disposition sur l'App Store. Sous `Build` sélectionnez le package que vous avez téléchargé via Transporter. + +C'est terminé ! Vous pouvez maintenant utiliser le site pour soumettre votre application pour vérification. Après quelques jours ouvrables si tout se passe bien, vous devriez voir votre application en direct sur le Mac App Store. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx new file mode 100644 index 00000000000..98457d19cc9 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# Compilations manuelles + +Le CLI Wails fait beaucoup de travail pour le projet, mais il est parfois souhaitable de construire manuellement votre projet. Ce document discutera des différentes opérations que fait le CLI et des différentes façons d'y parvenir. + +## Processus de construction + +Lorsque `wails build` ou `wails dev` sont utilisés, le CLI Wails effectue un processus de construction commun: + + - Installation des dépendances frontend + - Construire le projet frontend + - Générer des ressources de construction + - Compiler l'application + - [optionnel] Compresser l'application + +### Installation des dépendances frontend + +#### Étapes CLI + +- Si l'option `-s` est donné, cette étape est ignorée +- Vérifie `wails.json` pour voir s'il y a une commande install dans `frontend:install` +- S'il n'y en a pas, il saute cette étape +- Si le fichier existe, vérifie si `package.json` existe dans le répertoire du frontend. S'il n'existe pas, il saute cette étape +- Un hash MD5 est générée à partir du contenu du fichier `package.json` +- Il vérifie l'existence de `package.json.md5` et, s'il existe, compare son contenu (une somme MD5) avec celui généré pour voir si le contenu a changé. S'ils sont les mêmes, cette étape est ignorée +- Si `package.json.md5` n'existe pas, il le crée en utilisant la somme MD5 générée +- Si une compilation est maintenant requise, ou si `node_modules` n'existe pas, ou si l'option `-f` est donnée, la commande install est exécutée dans le répertoire frontend + +#### Étapes manuelles + +Cette étape peut être réalisée à partir de la ligne de commande ou d'un script avec `npm install`. + +### Construire le projet frontend + +#### CLI Wails + +- Si l'option `-s` est donné, cette étape est ignorée +- Vérifie `wails.json` pour voir s'il y a une commande de construction dans la clé `frontend:build` +- S'il n'y en a pas, il saute cette étape +- S'il existe, il est exécuté dans le répertoire du frontend + +#### Étapes manuelles + +Cette étape peut être réalisée à partir de la ligne de commande ou d'un script avec `npm run build` ou quel que soit le script de construction du frontend. + +### Générer les ressources + +#### CLI Wails + +- Si l'option `-nopackage` est activée, cette étape est ignorée +- Si le fichier `build/appicon.png` n'existe pas, un fichier par défaut est créé +- Pour Windows, voir [ Empaquetage pour Windows](#windows) +- Si `build/windows/icon.ico` n'existe pas, il la créera à partir de l'image `build/appicon.png`. + +##### Windows + +- If `build/windows/icon.ico` does not exist, it will create it from `build/appicon.png` using icon sizes of 256, 128, 64, 48, 32 and 16. This is done using [winicon](https://github.com/leaanthony/winicon). +- If the `build/windows/.manifest` file does not exist, it creates it from a default version. +- Compiles the application as a production build (above) +- Uses [winres](https://github.com/tc-hib/winres) to bundle the icon and manifest into a `.syso` file ready for linking. + +#### Étapes manuelles + +- Create `icon.ico` using the [winicon](https://github.com/leaanthony/winicon) CLI tool (or any other tool). +- Create / Update a `.manifest` file for your application +- Use the [winres CLI](https://github.com/tc-hib/go-winres) to generate a `.syso` file. + +### Compiler l'application + +#### CLI Wails + +- If the `-clean` flag is provided, the `build` directory is deleted and recreated +- For `wails dev`, the following default Go flags are used: `-tags dev -gcflags "all=-N -l"` +- For `wails build`, the following default Go flags are used: `-tags desktop,production -ldflags "-w -s"` + - On Windows, `-ldflags "-w -h -H windowsgui"` +- Additional tags passed to the CLI using `-tags` are added to the defaults +- Additional ldflags passed to the CLI using `-ldflags` are added to the defaults +- The `-o` flag is passed through +- The Go compiler specified by `-compiler` will be used for compilation + +#### Manual steps + +- For dev build, the minimum command would be: `go build -tags dev -gcflags "all=-N -l"` +- For production build, the minimum command would be: `go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- Ensure that you compile in the same directory as the `.syso` file + +### Compress application + +#### CLI Wails + +- If the `-upx` flag has been given, the `upx` program will be run to compress the application with the default settings +- If `-upxflags` is also passed, these flags are used instead of the default ones + +#### Manual steps + +- Run `upx [flags]` manually to compress the application. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx new file mode 100644 index 00000000000..7123cbe6b60 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx @@ -0,0 +1,191 @@ +# Migrating from v1 + +## Overview + +Wails v2 is a significant change from v1. This document aims to highlight the changes and the steps in migrating an existing project. + +### Creating the Application + +In v1, the main application is created using `wails.CreateApp`, bindings are added with `app.Bind`, then the application is run using `app.Run()`. + +Example: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +In v2, there is just a single method, `wails.Run()`, that accepts [application options](../reference/options.mdx#application-options). + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### Binding + +In v1, it was possible to bind both arbitrary functions and structs. In v2, this has been simplified to only binding structs. The struct instances that were previously passed to the `Bind()` method in v1, are now specified in the `Bind` field of the [application options](../reference/options.mdx#application-options): + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +In v1, bound methods were available to the frontend at `window.backend`. This has changed to `window.go`.`` + +### Application Lifecycle + +In v1, there were 2 special methods in a bound struct: `WailsInit()` and `WailsShutdown()`. These have been replaced with 3 lifecycle hooks as part of the [application options](../reference/options.mdx#application-options): + +- [OnStartup](../reference/options.mdx#onstartup) +- [OnShutdown](../reference/options.mdx#onshutdown) +- [OnDomReady](../reference/options.mdx#ondomready) + +Note: [OnDomReady](../reference/options.mdx#ondomready) replaces the `wails:ready` system event in v1. + +These methods can be standard functions, but a common practice is to have them part of a struct: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### Runtime + +The runtime in v2 is much richer than v1 with support for menus, window manipulation and better dialogs. The signature of the methods has changed slightly - please refer the the [Runtime Reference](../reference/runtime/intro.mdx). + +In v1, the [runtime](../reference/runtime/intro.mdx) was available via a struct passed to `WailsInit()`. In v2, the runtime has been moved out to its own package. Each method in the runtime takes the `context.Context` that is passed to the [OnStartup](../reference/options.mdx#onstartup) method. + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startup is called at application startup +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} + +``` + +### Assets + +The _biggest_ change in v2 is how assets are handled. + +In v1, assets were passed via 2 application options: + +- `JS` - The application's JavaScript +- `CSS` - The application's CSS + +This meant that the responsibility of generating a single JS and CSS file was on the developer. This essentially required the use of complicated packers such as webpack. + +In v2, Wails makes no assumptions about your frontend assets, just like a webserver. All of your application assets are passed to the application options as an `embed.FS`. + +**This means there is no requirement to bundle your assets, encode images as Base64 or attempt the dark art of bundler configuration to use custom fonts**. + +At startup, Wails will scan the given `embed.FS` for `index.html` and use its location as the root path for all the other application assets - just like a webserver would. + +Example: An application has the following project layout. All final assets are placed in the `frontend/dist` directory: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +Those assets may be used by the application by simply creating an `embed.FS`: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +Of course, bundlers can be used if you wish to. The only requirement is to pass the final application assets directory to Wails using an `embed.FS` in the `Assets` key of the [application options](../reference/options.mdx#application-options). + +### Project Configuration + +In v1, the project configuration was stored in the `project.json` file in the project root. In v2, the project configuration is stored in the `wails.json` file in the project root. + +The format of the file is slightly different. Here is a comparison: + +

+ +| v1 | v2 | Notes | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | Removed | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | Changed | +| frontend / dir | | Removed | +| frontend / install | frontend:install | Changed | +| frontend / build | frontend:build | Changed | +| frontend / bridge | | Removed | +| frontend / serve | | Removed | +| tags | | Removed | +| | wailsjsdir | The directory to generate wailsjs modules | +| | assetdir | The directory of the compiled frontend assets for `dev` mode. This is normally inferred and could be left empty. | +| | reloaddirs | Comma separated list of additional directories to watch for changes and to trigger reloads in `dev` mode. This is only needed for some more advanced asset configurations. | + +

diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..4a3de2a61b5 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# Mouse Buttons + +The Wails runtime intercepts mouse clicks to determine whether a frameless window needs resizing or a window needs to be moved. It has been asked how to detect when a mouse click has occurred, because `window.onclick` doesn't report the mouse buttons correctly. The following code shows how to detect mouse clicks: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +Reference: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx new file mode 100644 index 00000000000..21f7875e389 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# Obfuscated Builds + +Wails includes support for obfuscating your application using [garble](https://github.com/burrowers/garble). + +To produce an obfuscated build, you can use the `-obfuscate` flag with the `wails build` command: + +```bash +wails build -obfuscated +``` + +To customise the obfuscation settings, you can use the `-garbleargs` flag: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +These settings may be persisted in your [project config](../reference/project-config). + +## How it works + +In a standard build, all bound methods are available in the frontend under the `window.go` variable. When these methods are called, the corresponding backend method is called using the fully qualified function name. When using an obfuscated build, methods are bound using an ID instead of a name. The bindings generated in the `wailsjs` directory use these IDs to call the backend functions. + +:::note + +To ensure that your application will work in obfuscated mode, you must use the generated bindings under the `wailsjs` directory in your application. + +::: + +## Example + +Importing the "Greet" method from the bindings like this: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +will ensure that the method will work correctly in obfuscated mode, as the bindings will be regenerated with IDs and the call mechanism updated. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx new file mode 100644 index 00000000000..9d1d772d0fb --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# Overscroll + +[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) is the "bounce effect" you sometimes get when you scroll beyond a page's content boundaries. This is common in mobile apps. This can be disabled using CSS: + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/versioned_docs/version-v2.5.0/guides/routing.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/routing.mdx rename to website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx new file mode 100644 index 00000000000..4c7cf45ba99 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx @@ -0,0 +1,387 @@ +# Code Signing + +This is a guide on how you can sign your binaries generated with Wails on MacOS and Windows. The guide will target CI environments, more specifically GitHub Actions. + +## Windows + +First off you need a code signing certificate. If you do not already have one, Microsoft's info page lists some providers [here](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate). Please note that an EV certificate is not required unless you need to write kernel-level software such as device drivers. For signing your Wails app, a standard code signing certificate will do just fine. + +It may be a good idea to check with your certificate provider how to sign your binaries on your local machine before targeting automated build systems, just so you know if there are any special requirements. For instance, [here](https://www.ssl.com/how-to/using-your-code-signing-certificate/) is SSL.com's code signing guide for Windows. If you know how to sign locally, it will be easier to troubleshoot any potential issues in a CI environment. For instance, SSL.com code signing certificates require the `/tr` flag for [SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool) while other providers may only need the `/t` flag for providing the timestamping server. Popular GitHub Actions for signing Windows binaries like [this one](https://github.com/Dana-Prajea/code-sign-action) does not support the `/tr` flag on SignTool.exe. Therefore this guide will focus on signing our app manually with PowerShell commands, but you can use actions like the [code-sign-action](https://github.com/Dana-Prajea/code-sign-action) Action if you prefer. + +First off, let's make sure we are able to build our Wails app in our GitHub CI. Here is a small workflow template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend manually here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +Next we need to give the GitHub workflow access to our signing certificate. This is done by encoding your .pfx or .p12 certificate into a base64 string. To do this in PowerShell, you can use the following command assuming your certificate is called 'my-cert.p12': + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +You should now have your .txt file with the base64 encoded certificate. It should start with _-----BEGIN CERTIFICATE-----_ and end with _-----END CERTIFICATE-----_. Now you need to make two action secrets on GitHub. Navigate to _Settings -> Secrets -> Actions_ and create the two following secrets: + +- **WIN_SIGNING_CERT** with the contents of your base64 encoded certificate text. +- **WIN_SIGNING_CERT_PASSWORD** with the contents of your certificate password. + +Now we're ready to implement the signing in our workflow using one of the two methods: + +### Method 1: signing with commands + +This method uses PowerShell commands to sign our app, and leaves you control over the entire signing process. + +After the `"Build Wails app"` step, we can add the following step to our workflow: + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +This script creates a new directory for your certificate file, creates the certificate file from our base64 secret, converts it to a .pfx file, and finally signs the binary. The following variables needs to be replaced in the last line: + +- **signing algorithm**: usually sha256. +- **timestamping server**: URL to the timestamping server to use with your certificate. +- **path to binary**: path to the binary you want to sign. + +Given that our Wails config has `outputfilename` set to "app.exe" and that we have a certificate from SSL.com, this would be our workflow: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### Method 2: automatically signing with Action + +It is possible to use a Windows code signing Action like [this](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) one, but note it requires a SHA1 hash for the certificate and a certificate name. View an example of how to configure it on the Action's [marketplace](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate). + +--- + +## MacOS + +First off you need your code signing certificate from Apple. If you do not have one, a simple Google search will help you acquire one. Once you have your certificate, you need to export it and encode it to base64. [This tutorial](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions) shows you how to do that in an easy manner. Once you have exported your .p12 certificate file, you can encode it to base64 as seen in the tutorial with the following command: + +```bash +base64 Certificates.p12 | pbcopy +``` + +Now you're ready to create some GitHub project secrets, just as with Windows: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64** with the contents of your newly copied base64 certificate. +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD** with the contents of your certificate password. +- **APPLE_PASSWORD** with the contents of an App-Specific password to your Apple-ID account which you can generate [here](https://appleid.apple.com/account/manage). + +Let's make sure we are able to build our Wails app in our GitHub Action workflow. Here is a small template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +For code signing on macOS, [gon](https://github.com/mitchellh/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and will be used in this guide. + +After the `Build Wails app` step, add the following to the workflow: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +Now we need to configure some gon config files in our `build/darwin` directory: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +Where `source` is your Wails binary, `bundle_id` is your bundle ID, `apple_id` contains your Apple ID username and App-Specific password which you created earlier, and `sign.application_identity` is your identity which you can find by running the following command: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +In this file you configure the entitlements you need for you app, e.g. camera permissions if your app uses the camera. Read more about entitlements [here](https://developer.apple.com/documentation/bundleresources/entitlements). + +Make sure you have updated your `Info.plist` file with the same bundle ID as you entered in `gon-sign.json`. Here's an example `Info.plist` file: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +Now we're ready to add the signing step in our workflow after building the Wails app: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +Please note that signing binaries with Apple could take anywhere from minutes to hours. + +## Combined workflow file: + +Here is our GitHub workflow file with Windows + macOS combined: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# End notes + +This guide inspired by the RiftShare project and its workflow, which is highly recommended to check out [here](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml). diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..4651c422ed1 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +This guide will go into: + +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. + +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. + +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. + +- `npm create svelte@latest frontend` + +##### Modify wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. + +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Install/remove dependencies using your favorite package manager. + +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js + +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. + +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. + +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. + +- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! + +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html + +``` + +... + + + +... + +``` + +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. + +- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling + +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions + +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx new file mode 100644 index 00000000000..790e3107f04 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx @@ -0,0 +1,97 @@ +# Templates + +Wails generates projects from pre-created templates. In v1, this was a difficult to maintain set of projects that were subject to going out of date. In v2, to empower the community, a couple of new features have been added for templates: + +- Ability to generate projects from [Remote Templates](../reference/cli.mdx#remote-templates) +- Tooling to help create your own templates + +## Creating Templates + +To create a template, you can use the `wails generate template` command. To generate a default template, run: + +`wails generate template -name mytemplate` + +This creates the directory "mytemplate" with default files: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### Template Overview + +The default template consists of the following files and directories: + +| Filename / Dir | Description | +| --------------- | -------------------------------------------- | +| NEXTSTEPS.md | Instructions on how to complete the template | +| README.md | The README published with the template | +| app.tmpl.go | `app.go` template file | +| frontend/ | The directory containing frontend assets | +| go.mod.tmpl | `go.mod` template file | +| main.tmpl.go | `main.go` template file | +| template.json | The template metadata | +| wails.tmpl.json | `wails.json` template file | + +At this point it is advisable to follow the steps in `NEXTSTEPS.md`. + +## Creating a Template from an Existing Project + +It's possible to create a template from an existing frontend project by passing the path to the project when generating the template. We will now walk through how to create a Vue 3 template: + +- Install the vue cli: `npm install -g @vue/cli` +- Create the default project: `vue create vue3-base` + - Select `Default (Vue 3) ([Vue 3] babel, eslint)` +- After the project has been generated, run: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- The template may now be customised as specified in the `NEXTSTEPS.md` file +- Once the files are ready, it can be tested by running: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- To test the new project, run: `cd my-vue3-project` then `wails build` +- Once the project has compiled, run it: `.\build\bin\my-vue3-project.exe` +- You should have a fully functioning Vue3 application: + +```mdx-code-block +
+ +
+``` + +## Publishing Templates + +Publishing a template is simply pushing the files to GitHub. The following best practice is encouraged: + +- Remove any unwanted files and directories (such as `.git`) from your frontend directory +- Ensure that `template.json` is complete, especially `helpurl` +- Push the files to GitHub +- Create a PR on the [Community Templates](../community/templates.mdx) page +- Announce the template on the [Template Announcement](https://github.com/wailsapp/wails/discussions/825) discussion board diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..1c3b6a9cc52 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# Troubleshooting + +An assortment of troubleshooting tips. + +## The `wails` command appears to be missing? + +If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment variable. You will also normally need to close and reopen any open command prompts so that changes to the environment made by the installer are reflected at the command prompt. + +## My application is displaying a white/blank screen + +Check that your application includes the assets from the correct directory. In your `main.go` file, you will have something similar to the following code: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +Check that `frontend/dist` contains your application assets. + +### Mac + +If this happens on Mac, try adding the following to your `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Mac application not valid + +If your built application looks like this in finder: + +```mdx-code-block +

+ +

+``` + +it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to the `build/darwin` directory. + +## My application is not displaying the correct icon in Windows Explorer + +If your application is not displaying the correct icon, try deleting the hidden `IconCache.db` file located in the `C:\Users\<your username>\AppData\Local` directory. This will force Windows to rebuild the icon cache. + +Source: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## Cannot call backend method from frontend with variadic arguments + +If you have a backend method defined with variadic parameters, eg: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +calling this method from the frontend like this will fail: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Workaround: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Credit: https://github.com/wailsapp/wails/issues/1186 + +## I'm having getting proxy errors when trying to install Wails + +If you are getting errors like this: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +it's probably because the official Go Proxy is being blocked (Users in China have reported this). The solution is to set up the proxy manually, eg: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +Source: https://github.com/wailsapp/wails/issues/1233 + +## The generated TypeScript doesn't have the correct types + +Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, it is possible to specify what types should be generated using the `ts_type` struct tag. For more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). + +## When I navigate away from `index.html`, I am unable to call methods on the frontend + +If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding the following imports to the `` section of any new page you navigate to: + +```html + + + + +``` + +Source: https://github.com/wailsapp/wails/discussions/1512 + +## I get `too many open files` errors on my Mac when I run `wails dev` + +By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. This limit can be increased by running: `ulimit -n 1024` in the terminal. + +FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). + +## My Mac app gives me weird compilation errors + +A few users have reported seeing compilation errors such as the following: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. + +If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: + +`xcode-select -p` + +If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" + +It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine that may have different versions of Node installed, you may not be able to run your application. + +If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. + +## Build process stuck on "Generating bindings" + +Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx new file mode 100644 index 00000000000..ed258656d63 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +This page is for miscellaneous tips and tricks when using Visual Studio Code with Wails. + +## Vetur Configuration + +Many thanks to [@Lyimmi](https://github.com/Lyimmi) for this tip. Originally posted [here](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349). + +Vetur is a popular plugin for Visual Studio Code that provides syntax highlighting and code completion for Vue projects. When loading a Wails project in VSCode, Vetur will throw an error as it is expecting to find the frontend project in the root directory. To fix this, you can do the following: + +Create a file named `vetur.config.js` in the project's root. + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +Next, configure `frontend/tsconfig.json`: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +This should enable you to now use Vetur as expected. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx new file mode 100644 index 00000000000..a819d070d3e --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# Installateur NSIS + +```mdx-code-block +

+ +
+

+``` + +Wails prend en charge la génération d'installateurs Windows en utilisant l'installateur [NSIS](https://nsis.sourceforge.io/). + +## Installation de NSIS + +### Windows + +L'installateur est disponible sur la page [de téléchargement NSIS](https://nsis.sourceforge.io/Download). + +Si vous utilisez le gestionnaire de paquets chocolatey, exécutez le script suivant : + +``` +choco install nsis +``` + +Si vous installez NSIS manuellement, vous devez ajouter le dossier _Bin_ , qui contient `makensis.exe`, dans la variable d'environnement PATH. [Cette page](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/) est un bon tutoriel sur comment ajouter un dossier dans votre variable d'environnement PATH sur Windows. + +### Linux + +Le paquet `nsis` devrait être disponible via le gestionnaire de paquets de votre distribution. + +### MacOS + +NSIS est disponible via homebrew en utilisant : `brew install nsis`. + +## Génération de l'installateur + +Lorsqu'un nouveau projet est créé, Wails génère les fichiers de configuration NSIS dans `build/windows/installer`. La configuration est lue dans `installer/info.json`, et est configuré pour utiliser la section info du fichier `wails.json` : + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +Pour générer l'installateur de votre application, utilisez l'option `-nsis` avec la commande `wails build`: + +``` +wails build -nsis +``` + +L'installateur sera ensuite disponible dans le dossier `build/bin`. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx new file mode 100644 index 00000000000..6b24979d741 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +This page has miscellaneous guides related to developing Wails applications for Windows. + +## Handling the WebView2 Runtime Dependency + +Wails applications built for Windows have a runtime requirement on the Microsoft [WebView2 Runtime](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). Windows 11 will have this installed by default, but some machines won't. Wails offers an easy approach to dealing with this dependency. + +By using the `-webview2` flag when building, you can decide what your application will do when a suitable runtime is not detected (including if the installed runtime is too old). The four options are: + +1. Download +2. Embed +3. Navigateur +4. Error + +### Download + +This option will prompt the user that no suitable runtime has been found and then offer to download and run the official bootstrapper from Microsoft's WebView2 site. If the user proceeds, the official bootstrapper will be downloaded and run. + +### Embed + +This option embeds the official bootstrapper within the application. If no suitable runtime has been found, the application will offer to run the bootstrapper. This adds ~150k to the binary size. + +### Navigateur + +This option will prompt the user that no suitable runtime has been found and then offer to open a browser to the official WebView2 page where the bootstrapper can be downloaded and installed. The application will then exit, leaving the installation up to the user. + +### Error + +If no suitable runtime is found, an error is given to the user and no further action taken. + +## Fixed version runtime + +Another way of dealing with webview2 dependency is shipping it yourself. You can download [fixed version runtime](https://developer.microsoft.com/microsoft-edge/webview2/#download-section) and bundle or download it with your application. + +Also, you should specify path to fixed version of webview2 runtime in the `windows.Options` structure when launching wails. + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +Note: When `WebviewBrowserPath` is specified, `error` strategy will be forced in case of minimal required version mismatch or invalid path to a runtime. + +## Spawning other programs + +When spawning other programs, such as scripts, you will see the window appear on the screen. To hide the window, you can use the following code: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +Solution provided by [sithembiso](https://github.com/sithembiso) on the [discussions board](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172). diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx new file mode 100644 index 00000000000..0a6544735ee --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# Comment ça marche ? + +Une application Wails est une application Go standard, avec une interface graphique webkit. La partie Go de l'application se compose du code de l'application et d'une bibliothèque d'exécution qui fournit un certain nombre d'opérations utiles, comme le contrôle de la fenêtre de l'application. Le frontend est une fenêtre webkit qui affichera les ressources graphiques. Une version de la bibliothèque runtime de Javascript est aussi disponible depuis le frontend. Enfin, il est possible de lier les méthodes Go au frontend, et ceux-ci apparaîtront comme des méthodes Javascript qui peuvent être appelées, comme s'il s'agissait de méthodes locales Javascript. + +```mdx-code-block +
+ +
+``` + +## L'Application Principale + +### Vue d’ensemble + +L'application principale consiste en un seul appel à `wails.Run()`. Il accepte la configuration de l'application qui décrit la taille de la fenêtre d'application, le titre de la fenêtre, qu'elles sont les ressources à utiliser, etc. Une application de base pourrait ressembler à ceci : + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### Description des options + +Cet exemple a les options suivantes : + +- `Title` - Le texte qui devrait apparaître dans la barre de titre de la fenêtre +- `Width` & `Height` - Les dimensions de la fenêtre +- `Assets` - Les ressources du frontend de l'application +- `OnStartup` - Nom de la fonction à appeler quand la fenêtre est créée et est sur le point de commencer à charger les ressources du frontend +- `OnShutdown` - Nom de la fonction à appeler quand la fenêtre est sur le point d'être fermée +- `Bind` - La liste des structures Go à exposer au frontend + +Une liste complète des options d'application peut être trouvée dans la [Référence d'options](reference/options). + +#### Ressources + +L'option `Assets` est obligatoire car vous ne pouvez pas avoir d'application Wails sans ressources en frontend. Ces ressources peuvent être n'importe quel fichier que vous attendriez à trouver dans une application web - html, js, css, svg, png, etc. **Il n'y a aucune obligation d'utiliser un générateur de code ou framework** - des fichiers bruts suffisent. Lorsque l'application démarre, elle tentera de charger le fichier `index.html` à partir de vos ressources et le frontend fonctionnera essentiellement comme un navigateur à partir de ce point. Il est intéressant de noter que il n'y a pas de condition sur l'emplacement de `embed.FS`. Il est probable que le chemin d'intégration utilise un répertoire imbriqué par rapport au code de votre application principale, comme `frontend/dist`: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +Au démarrage, Wails va itérer les fichiers embarqués à la recherche du répertoire contenant `index.html`. Tous les autres actifs seront chargés par rapport à à ce répertoire. + +Comme les binaires de production utilisent les fichiers contenus dans `embed.FS`, il n'y a aucun fichier externe requis pour être expédié avec l'application. + +Lorsque vous exécutez en mode développement en utilisant la commande `wails dev` , les assets sont chargés à partir du disque, et tous les changements résultent en un "rechargement en direct". L'emplacement des actifs sera déduit de la `embed.FS`. + +Plus de détails peuvent être trouvés dans le [Guide de développement d'applications](guides/application-development.mdx). + +#### Callbacks du cycle de vie de l'application + +Juste avant que le frontend ne soit sur le point de charger `index.html`, un callback est fait à la fonction fournie dans [OnStartup](reference/options.mdx#onstartup). Un contexte standard Go est passé à cette méthode. Ce contexte est requis lors de l'appel à l'exécution, donc une bonne pratique est de sauvegarder une référence dans cette méthode. Juste avant que l'application ne s'arrête, la fonction de rappel [OnShutdown](reference/options.mdx#onshutdown) est appelée de la même manière, à nouveau avec le contexte. Il y a aussi un callback [OnDomReady](reference/options.mdx#ondomready) pour quand le frontend a terminé le chargement de tous les assets de `index.html` et est équivalent à l'événement [`body onload`](https://www.w3schools.com/jsref/event_onload.asp) en JavaScript. Il est également possible de s'accrocher à l'événement de fermeture de la fenêtre (ou de quitter l'application) en définissant l'option [OnBeforeClose](reference/options.mdx#onbeforeclose). + +#### Binding de méthodes + +L'option `Bind` est l'une des options les plus importantes dans une application Wails. Il spécifie quelles méthodes de structs sont à exposer au frontend. Pensez à des "contrôleurs" dans une application web traditionnelle. Quand l'application démarre, elle examine les instances structurées listées dans l'option `Bind`, détermine quelles méthodes sont publiques (commence par une lettre majuscule) et générera des versions JavaScript de ces méthodes qui peuvent être appelées par le code en frontend. + +:::info Note + +Wails exige que vous passiez dans une _instance_ du struct pour qu'il le lie correctement + +::: + +Dans cet exemple, nous créons une nouvelle instance `App` puis ajoutons cette instance à l'option `Bind` dans `wails.Run`: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +Vous pouvez lier autant de structures que vous le souhaitez. Assurez-vous juste de créer une instance de celle-ci et de la passer dans `Bind`: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +Lorsque vous exécutez `wails dev` (ou `wails generate module`), un module frontend sera généré contenant les éléments suivants : + +- JavaScript bindings pour toutes les méthodes liées +- Déclarations TypeScript pour toutes les méthodes liées +- Définitions TypeScript pour toutes les structures Go utilisées comme entrées ou sorties par les méthodes liées + +Cela rend incroyablement simple d'appeler le code Go depuis le frontend, en utilisant les mêmes structures de données. + +## Le frontend + +### Vue d’ensemble + +Le frontend est une collection de fichiers rendus par webkit. C'est comme un navigateur et un serveur web en un. Il y a virtuellement[^1] aucune limite vis à vis des frameworks ou des bibliothèques que vous pouvez utiliser. Les principaux points d'interaction entre le frontend et votre code Go sont: + +- L'appel des méthodes Go liées +- L'appel des méthodes d'exécution + +### L'appel des méthodes Go liées + +Lorsque vous exécutez votre application avec `wails dev`, il générera automatiquement des liaisons JavaScript pour vos structures dans un répertoire appelé `wailsjs/go` (Vous pouvez aussi le faire en exécutant `wails generate module`). Les fichiers générés reflètent les noms de paquets dans votre application. Dans l'exemple ci-dessus, nous associons `app`, qui a une méthode publique `Greet`. Cela conduira à la génération des fichiers suivants : + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +Ici nous pouvons voir qu'il y a un dossier `main` qui contient les liaisons JavaScript pour la structure `App` liée, ainsi que que le fichier de déclaration TypeScript pour ces méthodes. Pour appeler `Greet` depuis notre frontend, nous importons simplement la méthode et l'appelons comme une fonction JavaScript régulière: + +```javascript +// ... +import { Greet } from "../wailsjs/go/main/App"; + +function doGreeting(name) { + Greet(name).then((result) => { + // Do something with result + }); +} +``` + +La déclaration en TypeScript vous donne les bons types pour les méthodes paramètres et la valeur retournée : + +```ts +export function Greet(arg1: string): Promise; +``` + +Les méthodes générées retournent une Promise. Un appel réussi entraînera la première valeur de retour de l'appel Go à passer au `resolve` handler. Un appel infructueux est quand une méthode Go qui a un type d'erreur comme valeur de deuxième retour, passe une erreur à l'appelant. Ceci est passé en arrière via le handler `reject`. Dans l'exemple ci-dessus, `Greet` ne retourne qu'un `string` donc l'appel JavaScript ne sera jamais rejeté - à moins que des données non valides ne lui soient passées. + +Tous les types de données sont correctement traduits entre Go et JavaScript. Même les structs. Si vous renvoyez un struct d'un appel Go, il sera retourné à votre frontend en tant que classe JavaScript. + +:::info Note + +Les champs Struct _doivent avoir_ le champ `json` de défini afin de pouvoir l'inclure dans le TypeScript généré. + +Les structures imbriquées anonymes ne sont pas supportées pour le moment. + +::: + +Il est possible d'envoyer des structures à Go. N'importe quelle map/classe JavaScript passée comme argument, sera convertie en son équivalent. Pour faciliter ce processus, en mode `dev` un module TypeScript est généré, définissant tous les types de structures utilisés dans les méthodes liées. En utilisant ce module, il est possible de construire et envoyer des objets JavaScript natifs au code Go. + +Il y a aussi le support des méthodes Go qui utilisent les structures dans leur signature. Toutes les structures Go spécifiées par une méthode liée (que ce soit en tant que paramètres ou types de retour) auront les versions TypeScript automatiques générées dans le module de gestion de code Go. En utilisant ceux-ci, il est possible de partager le même modèle de données entre Go et JavaScript. + +Exemple: Nous mettons à jour notre méthode `Greet` pour accepter une `Person` au lieu d'une chaîne de caractères : + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +Le fichier `wailsjs/go/main/App.js` aura toujours le code suivant : + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +Mais le fichier `wailsjs/go/main/App.d.ts` sera mis à jour avec le code suivant : + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +Comme nous pouvons le voir, le namespace "main" est importé à partir du nouveau fichier "models.ts". Ce fichier contient toutes les définitions de struct utilisées par nos méthodes liées. Dans cet exemple, c'est une struct `Person`. Si nous regardons `models.ts`, nous pouvons voir comment les modèles sont définis : + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +Tant que vous avez TypeScript dans votre configuration de compilation en frontend, vous pouvez utiliser ces modèles de la manière suivante: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +La combinaison des liaisons générées et des modèles TypeScript crée un environnement de développement puissant. + +Plus d'informations sur la liaison peuvent être trouvées dans la section [Méthodes de liaison](guides/application-development.mdx#binding-methods) de la [Guide de développement d'applications](guides/application-development.mdx). + +### Appeler les méthodes runtime + +Le runtime JavaScript se trouve dans `window.runtime` et contient de nombreuses méthodes pour faire diverses tâches telles qu'émettre un événement ou effectuer des opérations de journalisation : + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +Plus de détails sur l'exécutable JS peuvent être trouvés dans la [Référence d'exécution](reference/runtime/intro). + +[^1]: Il y a un très petit sous-ensemble de bibliothèques qui utilisent des fonctionnalités non prises en charge dans WebViews. Il y a souvent des alternatives et des solutions de contournement pour de tels cas. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx new file mode 100644 index 00000000000..680c0bba22a --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# Introduction + +Wails est un projet qui vous permet d'écrire des applications de bureau en utilisant les technologies Go et web. + +Considérez cela comme une alternative légère et rapide d'Electron pour Go. Vous pouvez facilement construire des applications avec la flexibilité et la puissance de Go, combinée à un frontend riche et moderne. + +### Fonctionnalités + +- Menus natifs, Boîtes de dialogues, Thèmes et Translucidité +- Prise en charge de Windows, macOS et Linux +- Modèles intégrés pour Svelte, React, Preact, Vue, Lit et Vanilla JS +- Appeler facilement les méthodes Go depuis JavaScript +- Génération automatique du modèle TypeScript à partir des struct Go +- Aucun CGO ou DLL externe requis sous Windows +- Mode développement en direct en utilisant la puissance de [Vite](https://vitejs.dev/) +- CLI puissant pour créer, construire et empaqueter facilement des applications +- Une riche bibliothèque [runtime](/docs/reference/runtime/intro) +- Les applications construites avec Wails sont conformes aux Stores Apple & Microsoft + +Ceci est [varly](https://varly.app) - une application de bureau pour MacOS & Windows écrite à l'aide de Wails. Non seulement elle est belle, elle utilise les menus natifs et la translucidité - tout ce que vous pouvez attendre d'une application native moderne. + +```mdx-code-block +

+ + + +

+``` + +### Modèles de créations rapides + +Wails est livré avec un certain nombre de modèles préconfigurés qui vous permettent de faire fonctionner votre application rapidement. Il y a des modèles pour les frameworks suivants : Svelte, React, Vue, Preact, Lit et Vanilla. Il existe à la fois des versions JavaScript et TypeScript pour chaque modèle. + +### Éléments natifs + +Wails utilise une bibliothèque conçue pour gérer les éléments natifs tels que les fenêtres, menus, boîtes de dialogues, etc, pour que vous puissiez construire des applications de bureau riches en fonctionnalités. + +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. Sous Windows, c'est la nouvelle bibliothèque Microsoft Webview2, construite sur Chromium. + +### Interopérabilité Go & Javascript + +Wails met automatiquement vos méthodes Go à la disposition de Javascript, afin que vous puissiez les appeler par nom depuis votre frontend ! Il génère même des modèles Typescript pour les structures utilisées par vos méthodes Go, pour que vous puissiez passer les mêmes structures de données entre Go et Javascript. + +### Librairie d'exécution + +Wails fournit une bibliothèque runtime, pour Go et Javascript, qui gère beaucoup de choses dont les applications modernes ont besoin, comme Logging, Boîtes de dialogue, etc. + +### Expérience de développement en direct + +#### Reconstructions automatiques + +Lorsque vous exécutez votre application en mode "dev", Wails construira votre application en tant qu'application de bureau native, mais lira vos ressources depuis le disque. Il détectera toutes les modifications apportées à votre code Go, puis reconstruira et relancera automatiquement votre application . + +#### Rechargement automatique + +Lorsque des changements sont détectés dans les ressources de votre application, votre application en cours d'exécution sera "rechargée", reflétant presque immédiatement vos modifications . + +#### Développez votre application dans un navigateur + +Si vous préférez déboguer et vous développer dans un navigateur, Wails vous couvre. L'application en cours d'exécution a également un serveur web qui exécutera votre application depuis n'importe quel navigateur qui s'y connecte. Il sera aussi actualisé lorsque vos fichiers seront modifiés. + +### Binaires natifs prêts à la production + +Lorsque vous êtes prêt à faire une version finale de votre application, le CLI le compilera en un seul exécutable, avec tous les actifs qui y sont intégrés. Sous Windows et MacOS, il est possible de créer un paquet natif pour la distribution. Les ressources utilisées dans la compilation de l'application (icône, info. list, fichier manifest, etc) font partie de votre projet et peuvent être personnalisés, ce qui vous donne le contrôle total sur la façon dont vos applications sont construites. + +### Outils + +Le CLI Wails fournit un moyen sans tracas de générer, de construire et de regrouper vos applications. Il s'occupera de la lourde tâche de créer des icônes, de compiler votre application avec des paramètres optimaux et de fournir un binaire distribuable et prêt à la production. Choisissez parmi un certain nombre de modèles de démarrage pour démarrer rapidement ! diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx new file mode 100644 index 00000000000..30721c11204 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx @@ -0,0 +1,241 @@ +--- +sidebar_position: 2 +--- + +# CLI + +Le CLI Wails a un certain nombre de commandes qui sont utilisées pour gérer vos projets. Toutes les commandes sont exécutées de la manière suivante: + +`wails ` + +## init + +`wails init` est utilisé pour générer des projets. + +| Option | Description | Par défaut | +|:------------------------- |:---------------------------------------------------------------------------------------------------------------------- |:-------------:| +| -n "nom du projet" | Nom du projet. **Obligatoire**. | | +| -d "Répertoire du projet" | Dossier de projet à créer | Nom du projet | +| -g | Initialisation du dépôt git | | +| -l | Liste des modèles de projet disponibles | | +| -q | Supprimer les logs dans la console | | +| -t "nom du modèle" | Modèle de projet à utiliser. Cela peut être le nom d'un modèle par défaut ou d'une URL d'un projet hébergé sur github. | vanilla | +| -ide | Générer les fichiers du projet IDE | | +| -f | Forcer la compilation de l'application | false | + +Exemple: `wails init -n test -d mytestproject -g -ide vscode -q` + +Cela va générer un projet appelé "test" dans le répertoire "mytestproject", initialiser git, générer des fichiers de projet vscode et le faire silencieusement. + +Plus d'informations sur l'utilisation des IDEs avec Wails peuvent être trouvées [ici](../guides/ides.mdx). + +### Modèles à distance + +Les modèles distants (hébergés sur GitHub) sont pris en charge et peuvent être installés en utilisant l'URL du projet du modèle. + +Exemple : `wails init -n test -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +Une liste de modèles gérés par la communauté peut être trouvée [ici](../community/templates.mdx) + +:::warning Attention + +**Le projet Wails n'entretient pas, n'est pas responsable ni responsable des modèles tiers !** + +Si vous n'êtes pas sûr d'un modèle, inspectez les fichiers `package.json` et `wails.json` pour savoir quels scripts sont exécutés et quels paquets sont installés. + +::: + +## build + +`wails build` est utilisé pour compiler votre projet vers un binaire prêt à la production. + +| Option | Description | Par défaut | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Nettoie le répertoire `build/bin` | | +| -compiler "compiler" | Utiliser un autre compilateur pour compiler, par exemple go1.15beta1 | go | +| -debug | Conserve les informations de débogage dans l'application et affiche la console de débogage. Permet l'utilisation des outils de développement dans la fenêtre de l'application | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Affiche la commande build sans l'exécuter | | +| -f | Forcer la compilation de l'application | | +| -garbleargs | Arguments à passer à garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Options supplémentaires à passer au compilateur | | +| -m | Permet d'ignorer mod tidy avant la compilation | | +| -nopackage | Ne pas empaqueter l'application | | +| -nocolour | Désactive la couleur des logs dans le terminal | | +| -nosyncgomod | Ne pas synchroniser go.mod avec la version Wails | | +| -nsis | Génère l'installateur NSIS pour Windows | | +| -o filename | Nom du fichier de sortie | | +| -obfuscated | Cacher le code de l'application en utilisant [garble](https://github.com/burrowers/garble) | | +| -platform | Construit pour les [plates-formes](../reference/cli.mdx#platforms) données (séparées par des virgules) par exemple. `windows/arm64`. Notez que si vous ne donnez pas l'architecture, `runtime.GOARCH` est utilisé. | platform = le contenu de la variable d'environnement `GOOS` si elle existe, autrement `runtime.GOOS`.
arch = le contenu de la variable d'environnement `GOARCH` si elle existe, autrement `runtime.GOARCH`. | +| -race | Construire avec le détecteur Go race | | +| -s | Ignorer la construction du frontend | | +| -skipbindings | Ignorer la génération des liaisons | | +| -tags "extra tags" | Options de compilation à passer au compilateur Go. Doivent être entre guillemets. Séparés par un espace ou une virgule (pas les deux) | | +| -trimpath | Supprimer tous les chemins vers les fichiers système de l'exécutable final. | | +| -u | Met à jour le `go.mod de votre projet` pour utiliser la même version de Wails que le CLI | | +| -upx | Compresser le binaire final en utilisant "upx" | | +| -upxflags | Options à passer à upx | | +| -v int | Niveau de verbosité (0 - silencieux, 1 - par défaut, 2 - verbeux) | 1 | +| -webview2 | Stratégie d'installation WebView2 : download,embed,browser,error | download | +| -windowsconsole | Garder la fenêtre de la console lors de la construction d'une version pour Windows | | + +Pour une description détaillée des options `webview2` , veuillez vous référer au Guide de [Windows](../guides/windows.mdx). + +Si vous préférez construire en utilisant l'outil Go standard, veuillez consulter le guide [Constructions manuelles](../guides/manual-builds.mdx) . + +Exemple: + +`wails build -clean -o myproject.exe` + +:::info + +Info +Sur Mac, l'application sera livrée avec `Info.plist`, pas `Info.dev.plist`. + +::: + +:::info UPX sur Apple Silicon + +Il y a [problèmes](https://github.com/upx/upx/issues/446) avec l'utilisation de UPX avec Apple Silicon. + +::: + +:::info UPX sur Windows + +Certains antivirus marquent de manière erronée les binaires compressés d'`upx` comme virus, voir [la description du problème](https://github.com/upx/upx/issues/437). + +::: + +### Platformes + +Plateformes supportées: + +| Plateforme | Description | +|:---------------- |:-------------------------------------------------- | +| darwin | MacOS + Architecture de la machine de construction | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | Application universelle MacOS AMD64+ARM64 | +| windows | Windows 10/11 + architecture de la machine | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + architecture de la machine | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## doctor + +`wails doctor` effectuera des diagnostics pour vous assurer que votre système est prêt pour développer avec wails. + +Exemple: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## dev + +`wails dev` est utilisé pour exécuter votre application en mode « développement en direct ». Ceci signifie que : + +- Le fichier `go.mod` de l'application sera mis à jour pour utiliser la même version de Wails que le CLI +- L'application est compilée et exécutée automatiquement +- Un observateur est démarré et déclenchera une reconstruction de votre application de développement s'il détecte des changements dans vos fichiers go +- Un serveur web est lancé sur `http://localhost:34115` qui sert votre application (et pas seulement le frontend) sur http. Cela vous permet d'utiliser les extensions de développement de votre navigateur favori +- Tous les assets de l'application sont chargés à partir du disque. Si elles sont modifiées, l'application se rechargera automatiquement (pas de recompilation). Tous les navigateurs connectés rechargeront également +- Un module JS est généré fournissant les éléments suivants : +- Les méthodes Javascript permettant d'appeler vos méthodes Go avec JSDoc autogénérée, vous fournissant des indications sur les méthodes +- Les versions TypeScript de vos structures Go, qui peuvent être construites et transmises à vos méthodes +- Un second module JS est généré qui fournit une déclaration des méthodes et structures pour l'exécutable +- Sur macOS, il regroupera l'application dans un fichier `.app` et l'exécutera. Il utilisera un `build/darwin/Info.dev.plist` pour le développement. + +| Option | Description | Par défaut | +|:------------------------------------ |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:------------------------ | +| -appargs "args" | Arguments passés à l'application en style shell | | +| -assetdir "./chemin/vers/les/assets" | Sert les assets depuis le répertoire donné au lieu d'utiliser le fichier FS fourni | Valeur dans `wails.json` | +| -browser | Ouvre un navigateur à `http://localhost:34115` au démarrage | | +| -compiler "compiler" | Utiliser un autre compilateur pour compiler, par exemple go1.15beta1 | go | +| -debounce | Le temps d'attente pour le rechargement après qu'une modification d'actif est détectée | 100 (millisecondes) | +| -devserver "host:port" | L'adresse à laquelle lier le serveur de développement wails | "localhost:34115" | +| -extensions | Extensions pour déclencher les rebuilds (séparés par des virgules) | go | +| -forcebuild | Force la compilation de l'application | | +| -frontenddevserverurl "url" | Utiliser l'url du serveur de développement tiers pour servir les actifs, EG Vite | "" | +| -ldflags "flags" | Options supplémentaires à passer au compilateur | | +| -loglevel "loglevel" | Niveau de log à utiliser - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | Désactiver la couleur dans le terminal | false | +| -noreload | Désactiver le rechargement automatique lorsque les actifs changent | | +| -nosyncgomod | Ne pas synchroniser go.mod avec la version Wails | false | +| -race | Construire avec le détecteur Go race | false | +| -reloaddirs | Répertoires supplémentaires pour déclencher les recharges (séparés par des virgules) | Valeur dans `wails.json` | +| -s | Ignorer la construction du frontend | false | +| -save | Sauvegarde les options `assetdir`, `reloaddirs`, `wailsjsdir`, `debounce`, `devserver` and `frontenddevserverurl` dans `wails.json` pour quelles deviennent les informations par défaut pour les prochaines utilisations. | | +| -skipbindings | Ignorer la génération des liaisons | | +| -tags "extra tags" | Options de construction à passer au compilateur (séparées par des guillemets et des espaces) | | +| -v | Niveau de verbosité (0 - silencieux, 1 - par défaut, 2 - verbeux) | 1 | +| -wailsjsdir | Le répertoire où stocker les modules JS Wails générés | Valeur dans `wails.json` | + +Exemple: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +Cette commande fera ce qui suit : + +- Construisez l'application et exécutez-la (plus de détails [ici](../guides/manual-builds.mdx) +- Générer les modules JS Wails dans `./frontend/src` +- Surveillez les mises à jour des fichiers dans `./frontend/dist` et rechargez en cas de changement +- Ouvre un navigateur et se connecte à l'application + +Il y a plus d'informations sur l'utilisation de cette fonctionnalité avec les scripts de framework existants [ici](../guides/application-development.mdx#live-reloading). + +## generate + +### template + +Wails utilise des modèles pour la génération de projets. La commande `wails génère le template` aide à échafauder un modèle afin que il puisse être utilisé pour générer des projets. + +| Option | Description | +|:---------------- |:-------------------------------------------------------- | +| -name | Le nom du modèle (Obligatoire) | +| -frontend "path" | Chemin vers le projet frontend à utiliser dans le modèle | + +Pour plus de détails sur la création de modèles, consultez le [Guide sur les modèles](../guides/templates.mdx). + +### module + +La commande `wails génère le module` vous permet de générer manuellement le répertoire `wailsjs` pour votre application. + +## update + +`wails update` va mettre à jour la version du CLI Wails. + +| Option | Description | +|:------------------ |:----------------------------------------------------- | +| -pre | Mettre à jour la version avec la dernière pre-release | +| -version "version" | Installer une version spécifique du CLI | + +## version + +`wails version` va simplement afficher la version actuelle du CLI. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx new file mode 100644 index 00000000000..80dbaa0c5cf --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# Menus + +Il est possible d'ajouter un menu applicatif aux projets Wails. Ceci est réalisé en définissant une structure [Menu](#menu) et en la définissant dans la configuration de l'application [`Menu`](../reference/options.mdx#menu) , ou en appelant la méthode d'exécution [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). + +Un exemple de définition d'un menu : + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +Il est également possible de mettre à jour dynamiquement le menu, en mettant à jour le menu struct et en appelant [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). + +L'exemple ci-dessus utilise des méthodes d'aide, cependant il est possible de construire le menu manuellement. + +## Menu + +Un Menu est une collection de MenuItems: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +Pour le menu de l'application, chaque MenuItem représente un seul menu tel que "Edit". + +Une méthode simple d'aide est fournie pour les menus de construction : + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +Cela rend la mise en page du code plus semblable à celle d'un menu sans avoir à ajouter les éléments de menu manuellement après leur création. Vous pouvez également créer les liens de menu et les ajouter au menu manuellement. + +## MenuItem + +Un MenuItem représente un élément dans un Menu. + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| Champ | Type | Notes | +| ----------- | ------------------------------------ | --------------------------------------------------------------------------------------- | +| Label | string | Le texte du menu | +| Accelerator | [\*keys.Accelerator](#accelerator) | Raccourci pour ce lien de menu | +| Type | [Type](#type) | Type de MenuItem | +| Disabled | bool | Désactive l'élément de menu | +| Hidden | bool | Masque cet élément de menu | +| Checked | bool | Ajoute une coche à l'élément (case à cocher & Types de radio) | +| SubMenu | [\*Menu](#menu) | Définit un sous-menu | +| Click | [Callback](#callback) | Fonction à appeler quand un click est fait sur cet élément du menu. | +| Role | string | Définit un rôle [](#role) pour cet élément de menu. Pour Mac seulement, pour le moment. | + +### Accelerator + +Les accélérateurs (parfois appelés raccourcis clavier) définissent une liaison entre une clé et un élément du menu. Wails définit un accélérateur comme une combinaison ou une clé + [modificateur](#modifier). Ils sont disponibles dans le paquet `"github.com/wailsapp/wails/v2/pkg/menu/keys"`. + +Exemple: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +Les clés sont n'importe quel caractère sur un clavier à l'exception de `+`, qui est défini comme `plus`. Certaines clés ne peuvent pas être représentées comme des caractères, il y a donc un ensemble de caractères nommés qui peuvent être utilisés : + +| | | | | +|:----------------:|:-----:|:-----:|:---------:| +| `retour arrière` | `f1` | `f16` | `f31` | +| `tabulation` | `f2` | `f17` | `f32` | +| `retour` | `f3` | `f18` | `f33` | +| `entrée` | `f4` | `f19` | `f34` | +| `echap` | `f5` | `f20` | `f35` | +| `gauche` | `f6` | `f21` | `numlock` | +| `droite` | `f7` | `f22` | | +| `haut` | `f8` | `f23` | | +| `bas` | `f9` | `f24` | | +| `espace` | `f10` | `f25` | | +| `suppr` | `f11` | `f36` | | +| `début` | `f12` | `f37` | | +| `fin` | `f13` | `f38` | | +| `page haut` | `f14` | `f39` | | +| `page bas` | `f15` | `f30` | | + +Wails prend également en charge l'analyse des accélérateurs en utilisant la même syntaxe qu'Electron. Ceci est utile pour stocker les accélérateurs dans les fichiers de configuration . + +Exemple: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### Modifier + +Les modificateurs suivants sont des touches qui peuvent être utilisées en combinaison avec la touche accélérateur: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +Un certain nombre de méthodes d'aide sont disponibles pour créer des accélérateurs en utilisant des modificateurs: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +Les modificateurs peuvent être combinés en utilisant `keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)`: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### Type + +Chaque lien de menu doit avoir un type et il y a 5 types disponibles: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +Pour plus de commodité, des méthodes d'aide sont fournies pour créer rapidement un lien de menu : + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +Vous pouvez également créer des liens directement dans un menu en utilisant les méthodes "Add" : + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +Une note sur les groupes radio : Un groupe radio est défini comme un certain nombre d'éléments du menu radio qui sont à côté l'un de l'autre dans le menu. Cela signifie que vous n'avez pas besoin de regrouper les éléments car il est automatique. Cependant, cela signifie également que vous ne pouvez pas avoir 2 groupes radio les uns à côté des autres - il doit y avoir un élément non-radio entre eux. + +### Callback + +Chaque lien de menu peut avoir une fonction qui est exécutée lorsque l'élément est cliqué : + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +La fonction reçoit une structure `CallbackData` qui indique quel élément de menu a été cliqué. Ceci est utile lorsque utilise des groupes radio qui peuvent partager une fonction. + +### Role + +:::info Roles + +Les rôles ne sont actuellement pris en charge que sur Mac. + +::: + +Un lien de menu peut avoir un rôle, qui est essentiellement un lien de menu prédéfini. Nous supportons actuellement les rôles suivants : + +| Role | Description | +| ------------ | ----------------------------------------------------------------------------------- | +| AppMenuRole | Le menu standard de l'application Mac. Peut être créé en utilisant `menu.AppMenu()` | +| EditMenuRole | Le menu d'édition standard pour Mac. Peut être créé en utilisant `menu.EditMenu()` | diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx new file mode 100644 index 00000000000..a80e3526732 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx @@ -0,0 +1,853 @@ +--- +sidebar_position: 3 +--- + +# Options + +## Options de l'application + +La structure `Options.App` contient la configuration de l'application. Il est passé à la méthode `wails.Run()`: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // User messages that can be customised + Messages *windows.Messages + // OnSuspend is called when Windows enters low power mode + OnSuspend func() + // OnResume is called when Windows resumes from low power mode + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### Title + +Le texte affiché dans la barre de titre de la fenêtre. + +Nom : Title
Type : `string` + +### Width + +La largeur initiale de la fenêtre. + +Nom: Width
Type: `int`
Défaut: 1024. + +### Height + +La hauteur initiale de la fenêtre. + +Nom: Height
Type: `int`
Défaut: 768 + +### DisableResize + +Par défaut, la fenêtre principale est redimensionnable. Mettre ceci à `true` le conservera une taille fixe. + +Nom: DisableResize
Type: `bool` + +### Fullscreen + +Obsolète: Veuillez utiliser [WindowStartState](#windowstartstate). + +### WindowStartState + +Définit comment la fenêtre devrait se présenter au démarrage. + +| Valeur | Win | Mac | Lin | +| ---------- | --- | --- | --- | +| Fullscreen | ✅ | ✅ | ✅ | +| Maximised | ✅ | ✅ | ✅ | +| Minimised | ✅ | ❌ | ✅ | + +Nom: WindowStartState
Type: `options.WindowStartState` + +### Frameless + +Quand réglé sur `true`, la fenêtre n'aura pas de bordure ou de barre de titre. Voir aussi les [fenêtres sans cadre sous Windows](../guides/frameless.mdx). + +Nom: Frameless
Type: `bool` + +### MinWidth + +Définit la largeur minimale de la fenêtre. Si la valeur donnée dans `Width` est inférieure à cette valeur, la fenêtre sera définie à `MinWidth` par défaut. + +Nom: MinWidth
Type: `int` + +### MinHeight + +Définit la hauteur minimale de la fenêtre. Si la valeur donnée dans `Height` est inférieure à cette valeur, la fenêtre sera définie à `MinHeight` par défaut. + +Nom: MinHeight
Type: `int` + +### MaxWidth + +Définit la largeur maximale de la fenêtre. Si la valeur donnée dans `Width` est supérieure à cette valeur, la fenêtre sera définie à `MaxWidth` par défaut. + +Nom: MaxWidth
Type: `int` + +### MaxHeight + +Définit la hauteur maximale de la fenêtre. Si la valeur donnée en `Height` est supérieure à cette valeur, la fenêtre sera définie à `MaxHeight` par défaut. + +Nom: MaxHeight
Type: `int` + +### StartHidden + +Lorsque réglé sur `true`, l'application sera masquée jusqu'à ce que [WindowShow](../reference/runtime/window.mdx#windowshow) soit appelé. + +Nom: StartHidden
Type: `bool` + +### HideWindowOnClose + +Par défaut, la fermeture de la fenêtre fermera l'application. Définir ceci à `true` signifie que + +la fenêtre sera cachée à la place. + +Nom: HideWindowOnClose
Type: `bool` + +### BackgroundColour + +Cette valeur est la couleur de fond par défaut de la fenêtre. Exemple: options.NewRGBA(255,0,0,128) - Rouge à 50% de transparence + +Nom: BackgroundColour
Type: `*options.RGBA`
Défaut: white + +### AlwaysOnTop + +Indique que la fenêtre doit rester au-dessus des autres fenêtres lors de la perte de focus. + +Nom: AlwaysOnTop
Type: `bool` + +### Assets + +Obsolète: Veuillez utiliser des actifs sur les options [AssetServer spécifiques](#assetserver). + +### AssetsHandler + +Obsolète : Veuillez utiliser AssetsHandler sur [Options spécifiques à AssetServer](#assetserver). + +### AssetServer + +Ceci définit les options spécifiques à AssetServer. Il permet de personnaliser l'AssetServer avec des actifs statiques, servant les assets dynamiquement avec un `http.Handler` ou brancher dans la chaîne de requêtes avec un `assetserver.Middleware`. + +Toutes les fonctionnalités d'une `http.Request` ne sont pas actuellement prises en charge, veuillez consulter la matrice de fonctionnalité suivante : + +| Fonctionalité | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +Nom: AssetServer
Type: `*assetserver.Options` + +#### Assets + +Les ressources statiques du frontend à être utilisées par l'application. + +Une requête GET est d'abord tentée d'être servie à partir de ce `fs.FS`. Si le `fs.FS` retourne `os. rrNotExist` pour ce fichier, le traitement des requêtes va revenir au [Handler](#handler) et essaie de répondre à la requête GET. + +Si la valeur est nulle, toutes les requêtes GET seront envoyées à [Handler](#handler). + +Nom: Assets
Type: `fs.FS` + +#### Handler + +Le gestionnaire d'assets est un `http.Handler` générique pour la gestion de secours des assets qui ne peuvent pas être trouvés. + +Le gestionnaire sera appelé pour chaque requête GET qui ne peut pas être servie à partir de [Assets](#assets), en raison de `os.ErrNotExist`. De plus, toutes les requêtes non GET seront toujours servies par ce gestionnaire. Si non défini, le résultat est le suivant dans les cas où le Gestionnaire aurait été appelé : + +- Requête GET : `http.StatusNotFound` +- Autre requête : `http.StatusMethodNotAllowed` + +REMARQUE : Lorsqu'il est utilisé en combinaison avec un serveur de développement Frontend, il peut y avoir des limitations, par exemple. Vite affiche l'index.html sur chaque chemin qui ne contient pas d'extension de fichier. + +Nom: AssetsHandler
Type: `http.Handler` + +#### Middleware + +Middleware est un Middleware HTTP qui permet de se connecter à la chaîne de requêtes AssetServer. Il permet de sauter dynamiquement le gestionnaire de requête par défaut, par exemple implémenter un routage spécialisé, etc. Le Middleware est appelé pour construire un nouveau `http.Handler` utilisé par l'AssetSever et reçoit également le gestionnaire par défaut utilisé par le serveur AssetServer comme argument. + +Si elle n'est pas définie, la chaîne de requête AssetServer par défaut est exécutée. + +Nom: Middleware
Type: `assetserver.Middleware` + +### Menu + +Le menu à utiliser par l'application. Plus de détails sur les menus dans la [Référence des Menu](../reference/runtime/menu.mdx). + +:::note + +Sur Mac, si aucun menu n'est spécifié, un menu par défaut sera créé. + +::: + +Nom: Menu
Type: `*menu.Menu` + +### Logger + +Le logger à utiliser par l'application. Plus de détails sur la connexion dans la [Référence du logger](../reference/runtime/log.mdx). + +Nom: Logger
Type: `logger.Logger`
Défaut: Logs envoyé à Stdout + +### LogLevel + +Le niveau de log par défaut. Plus de détails sur la connexion dans la [Référence du logger](../reference/runtime/log.mdx). + +Nom: LogLevel
Type: `logger.LogLevel`
Défaut: `Info` en mode dev, `Error` en mode production + +### LogLevelProduction + +Le niveau de log par défaut pour les compilations de production. Plus de détails sur la connexion dans la [Référence du logger](../reference/runtime/log.mdx). + +Nom: LogLevelProduction
Type: `logger.LogLevel`
Défaut: `Error` + +### OnStartup + +Ce callback est appelé après la création du frontend, mais avant que `index.html` n'ait été chargé. Il lui donne le contexte de l'application. + +Nom: OnStartup
Type: `func(ctx context.Context)` + +### OnDomReady + +Ce callback est appelé après que le frontend ait chargé `index.html` et ses ressources. Il lui donne le contexte de l'application. + +Nom: OnDomReady
Type: `func(ctx context.Context)` + +### OnShutdown + +Ce calllback est appelé après que le frontend ait été détruit, juste avant la fin de l'application. Il lui donne le contexte de l'application. + +Nom: OnShutdown
Type: `func(ctx context.Context)` + +### OnBeforeClose + +Si ce callback est défini, il sera appelé lorsque l'application est sur le point de quitter, soit en cliquant sur la fenêtre fermez le bouton ou en appelant `runtime.Quit`. Retourner "true" dans cette méthode entraînera la poursuite de l'application, "false" continuera à éteindre comme d'habitude. C'est un bon exemple pour confirmer avec l'utilisateur si il souhaite quitter le programme. + +Exemple: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +Nom: OnBeforeClose
Type: `func(ctx context.Context) bool` + +### CSSDragProperty + +Indique la propriété CSS à utiliser pour identifier quels éléments peuvent être utilisés pour faire glisser la fenêtre. Par défaut : `--wails-draggable`. + +Nom: CSSDragProperty
Type: `string` + +### CSSDragValue + +Indique quelle valeur le style `CSSDragProperty` doit avoir pour faire glisser la fenêtre. Par défaut: `drag`. + +Nom: CSSDragValue
Type: `string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenu active le menu contextuel par défaut du navigateur en production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +Lorsque cette option est activée, par défaut, le menu contextuel ne sera affiché que pour du texte (où Couper/Copier/Coller est nécessaire), pour remplacer ce comportement, vous pouvez utiliser la propriété CSS `--default-contextmenu` sur n'importe quel élément HTML (y compris le corps ``) avec les valeurs suivantes : + +| Style CSS | Comportement | +| ------------------------------ | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `--default-contextmenu: auto;` | (**défaut**) n'affichera le menu contextuel par défaut que si :
contentEditable est vrai OU le texte a été sélectionné OU l'élément est entrée ou la zone de texte | +| `--default-contextmenu: show;` | affichera toujours le menu de contexte par défaut | +| `--default-contextmenu: hide;` | masquera toujours le menu contextuel par défaut | + +Cette règle est héritée comme n'importe quelle règle CSS normale, donc l'imbrication fonctionne comme prévu. + +:::note +Cette fonctionnalité de filtrage n'est activée qu'en production, donc en développement et en construction de débogage, le menu contextuel complet est toujours disponible partout. +::: + +:::warning +Cette fonctionnalité de filtrage n'est PAS une mesure de sécurité, le développeur devrait s'attendre à ce que le menu contextuel complet puisse être divulgué à tout moment qui pourrait contenir des commandes comme (Télécharger l'image, Recharger, Enregistrer la page web), si c'est une préoccupation, le développeur DEVRAIT NE PAS activer le menu contextuel par défaut. +::: + + +Nom: EnableDefaultContextMenu
Type: `bool` + +### EnableFraudulentWebsiteDetection + +EnableFraudulentWebWebDetection permet de rechercher des contenus frauduleux, tels que des programmes malveillants ou des tentatives d'hameçonnage. Ces services peuvent envoyer des informations à partir de votre application, telles que les URL vers lesquelles vous avez navigué et éventuellement d'autres contenus vers le cloud, des services d'Apple et de Microsoft. + +Nom: EnableFraudulentWebsiteDetection
Type: `bool` + +### ZoomFactor + +Nom: ZoomFactor
Type: `float64` + +Ceci définit le facteur de zoom pour WebView2. Il s'agit de l'option correspondant au zoom avant ou arrière défini par l'utilisateur. + +### IsZoomControlEnabled + +Nom : IsZoomControlEnabled
Type : `bool` + +Cela permet de modifier le facteur de zoom par l'utilisateur. Veuillez noter que le facteur de zoom peut être défini dans les options tandis que ne permet pas à l'utilisateur de le modifier à l'exécution (f.e. pour une application vitrine ou similaire). + +### Bind + +La liste des structs Go définissant des méthodes qui doivent être liées au frontend. + +Nom: Bind
Type: `[]interface{}` + +### ErrorFormatter + +Une fonction qui détermine comment les erreurs sont formatées lorsqu'elles sont retournées par un appel de méthode JS-to-Go. La valeur retournée sera sous format JSON. + +Nom: ErrorFormatter
Type: `func (error) any` + +### Windows + +Ceci définit les options [spécifiques à Windows](#windows). + +Nom: Windows
Type: `*windows.Options` + +#### WebviewIsTransparent + +Mettre ceci à `true` rendra l'arrière-plan du webview transparent quand une valeur alpha de `0` est utilisée. Cela signifie que si vous utilisez `rgba(0,0,0,0)` pour `la couleur d'arrière-plan` dans votre CSS, la fenêtre d'hôte sera affichée. Souvent combiné avec [WindowIsTranslucent](#WindowIsTranslucent) pour faire des applications d'apparence de givre. + +Nom : WebviewIsTransparent
Type : `bool` + +#### WindowIsTranslucent + +Définir ceci à `true` rendra l'arrière-plan de la fenêtre translucide. Souvent combiné avec [WebviewIsTransparent](#WebviewIsTransparent). + +Pour les versions de Windows 11 avant la version 22621, cela utilisera la méthode [BlurBehind](https://learn.microsoft.com/en-us/windows/win32/dwm/blur-ovw) pour la translucidité, qui peut être lente. Pour les versions de Windows 11 après la version 22621, cela activera les nouveaux types de transparence qui sont beaucoup plus rapides. Par défaut, le type de transparence utilisé sera déterminé par Windows. Pour configurer ceci, utilisez l'option [BackdropType](#BackdropType). + +Nom: WindowIsTranslucent
Type: `bool` + +#### BackdropType + +:::note + +Nécessite Windows 11 version 22621 ou supérieure. + +::: + +Définit le type de transparence de la fenêtre. Ceci n'est applicable que si [WindowIsTranslucent](#WindowIsTranslucent) est défini à `true`. + +Nom: BackdropType
Type `windows.BackdropType` + +La valeur peut être l'une des valeurs suivantes : + +| Valeur | Description | +| ------- | ------------------------------------------------------------------------------------------------- | +| Auto | Laisser Windows décider quel arrière-plan utiliser | +| None | Ne pas utiliser de transparence | +| Acrylic | Utilisez l'effet [Acrylique](https://learn.microsoft.com/en-us/windows/apps/design/style/acrylic) | +| Mica | Utiliser l'effet [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) | +| Tabbed | Utiliser Tabbed. C'est un arrière-plan qui est similaire à Mica. | + +#### DisableWindowIcon + +Définir ceci à `true` supprimera l'icône dans le coin supérieur gauche de la barre de titre. + +Nom: DisableWindowIcon
Type: `bool` + +#### DisableFramelessWindowDecorations + +Définir ceci à `true` supprimera les décorations de fenêtre en mode [sans cadre](#Frameless). Cela signifie qu'il n'y aura pas de « Aero Shadow» et aucun « Coins arrondis» ne sera affiché pour la fenêtre. Veuillez noter que les "coins arrondis" ne sont pris en charge que sur Windows 11. + +Nom: DisableFramelessWindowDecorations
Type: `bool` + +#### WebviewUserDataPath + +Ceci définit le chemin où WebView2 stocke les données de l'utilisateur. Si vide, `%APPDATA%\[BinaryName.exe]` sera utilisé. + +Nom: WebviewUserDataPath
Type: `string` + +#### WebviewBrowserPath + +Ceci définit le chemin vers un répertoire avec les fichiers exécutables et bibliothèques WebView2. Si l'option est vide, l'instance de webview2 installé dans le système sera utilisé. + +Informations importantes sur la version corrigée : + +- [Comment récupérer et extraire l'exécutable](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [Problèmes connus pour la version corrigée](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [Le chemin de la version corrigée du runtime WebView2 ne doit pas contenir \Edge\Application\.](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +Nom: WebviewBrowserPath
Type: `string` + +#### Theme + +Version minimale de Windows : Windows 10 2004/20H1 + +Ceci définit le thème que l'application doit utiliser : + +| Valeur | Description | +| ------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| SystemDefault | _Default_. Le thème sera basé sur la valeur par défaut du système. Si l'utilisateur change de thème, l'application se mettra à jour pour utiliser le nouveau paramètre | +| Dark | L'application utilisera uniquement un thème sombre | +| Light | L'application utilisera uniquement un thème clair | + +Nom: Theme
Type: `windows.Theme` + +#### CustomTheme + +:::note + +Version minimale de Windows : Windows 10/11 2009/21H2 Build 22000 + +::: + +Vous permet de spécifier des couleurs personnalisées pour la barre de titre, le texte de titre et la bordure pour le mode clair et foncé. ainsi que lorsque la fenêtre est active ou inactive. + +Nom: CustomTheme
Type: `windows.CustomTheme` + +##### Type CustomTheme + +Le struct CustomTheme utilise `int32` pour spécifier les valeurs de couleurs. Celles-ci sont au format standard(!) Windows soit : `0x00BBGGAA`. Une fonction d'aide est fournie pour effectuer les conversions de RGB dans ce format : `windows.RGB(r,g,b uint8)`. + +NOTE : Toute valeur non fournie sera par défaut noire. + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +Exemple: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### Messages + +Un struct de chaînes utilisées par l'installateur webview2 si un runtime webview2 valide n'est pas trouvé. + +Nom: Messages
Type: `*windows.Messages` + +Personnalisez ceci pour n'importe quelle langue que vous choisissez de supporter. + +#### ResizeDebounceMS + +ResizeDebounceMS est le temps entre deux réajustements du contenu de la fenêtre lors du redimensionnement de la fenêtre. La valeur par défaut (0) effectuera des réajustements aussi vite qu'il le peut. + +Nom: ResizeDebounceMS
Type: `uint16` + +#### OnSuspend + +Si défini, cette fonction sera appelée lorsque Windows passera en mode économie d'énergie + +Nom: OnSuspend
Type: `func()` + +#### OnResume + +Si défini, cette fonction sera appelée lorsque Windows sortira du mode économie d'énergie + +Nom: OnResume
Type: `func()` + +#### WebviewGpuIsDisabled + +Définir ceci à `true` désactivera l'accélération matérielle GPU pour la webview. + +Nom: WebviewGpuIsDisabled
Type: `bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +Ceci définit [les options spécifiques à Mac](#mac). + +Nom: Mac
Type: `*mac.Options` + +#### TitleBar + +La structure TitleBar permet de configurer l'apparence de la barre de titre. + +Nom: TitleBar
Type: [`*mac.TitleBar`](#titlebar-struct) + +##### Struct de la Titlebar + +La barre de titre de l'application peut être personnalisée en utilisant les options suivantes de TitleBar : + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| Nom | Description | +| -------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TitlebarAppearsTransparent | Rend la barre de titre transparente. Cela a pour effet de masquer la barre de titre et le contenu remplit la fenêtre. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | Masque le titre de la fenêtre. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | Supprime [NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/) du style | +| FullSizeContent | Fait que la webview remplisse toute la fenêtre. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | Ajoute une barre d'outils par défaut à la fenêtre. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | Supprime la ligne située sous la barre d'outils. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +Des paramètres préconfigurés sont disponibles : + +| Configuration | Exemple | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +Exemple: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +Cliquez sur [ici](https://github.com/lukakerr/NSWindowStyles) si vous voulez de l'inspiration sur la personnalisation de la barre de titre. + +#### Appearance + +L'apparence est utilisée pour définir le style de votre application en accord avec les noms [NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc) d'Apple. + +Nom: Appearance
Type: [`mac.AppearanceType`](#appearance-type) + +##### Type d'Appearance + +Vous pouvez spécifier l'apparence [de l'application](https://developer.apple.com/documentation/appkit/nsappearance?language=objc). + +| Valeur | Description | +| ----------------------------------------------------- | ------------------------------------------------------------------- | +| DefaultAppearance | DefaultAppararance utilise la valeur système par défaut | +| NSAppearanceNameAqua | Utilise l'apparence thème clair standard | +| NSAppearanceNameDarkAqua | Utilise l'apparence thème sombre standard | +| NSAppearanceNameVibrantLight | Utilise une apparence avec une lumière vibrante | +| NSAppearanceNameAccessibilityHighContrastAqua | Utilise l'apparence thème clair standard avec un constrate élevé | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | Utilise l'apparence thème sombre standard avec un contraste élevé | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | Utilise l'apparence lumière vibrante avec un constrate élevé | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | Utilise l'apparence du thème sombre vibrant avec un constrate élevé | + +Exemple: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### WebviewIsTransparent + +Mettre ceci à `true` rendra l'arrière-plan du webview transparent quand une valeur alpha de `0` est utilisée. Cela signifie que si vous utilisez `rgba(0,0,0,0)` pour `la couleur d'arrière-plan` dans votre CSS, la fenêtre d'hôte sera affichée. Souvent combiné avec [WindowIsTranslucent](#WindowIsTranslucent) pour faire des applications d'apparence de givre. + +Nom : WebviewIsTransparent
Type : `bool` + +#### WindowIsTranslucent + +Définir ceci à `true` rendra l'arrière-plan de la fenêtre translucide. Souvent combiné avec [WebviewIsTransparent](#WebviewIsTransparent) pour donner un aspect givré à la fenêtre. + +Nom: WindowIsTranslucent
Type: `bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Nom | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Exemple: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### About + +Cette configuration vous permet de définir le titre, le message et l'icône pour l'élément de menu "À propos" dans le menu de l'application créé par le rôle "AppMenu". + +Nom: About
Type: [`*mac.AboutInfo`](#about-struct) + +##### Struct de About + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +Si ces paramètres sont fournis, un lien de menu "À propos" apparaîtra dans le menu de l'application (lors de l'utilisation du rôle `AppMenu`). Exemple: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +L'élément de menu "À propos" apparaîtra dans le menu de l'application: + +```mdx-code-block +
+ +
+
+``` + +Lorsqu'il est cliqué, cela ouvrira la boîte de message "à propos" : + +```mdx-code-block +
+ +
+
+``` + +### Linux + +Ceci définit [les options spécifiques à Linux](#linux). + +Nom: Linux
Type: `*linux.Options` + +#### Icon + +Définit l'icône représentant la fenêtre. Cette icône est utilisée lorsque la fenêtre est réduite (aussi appelée iconified). + +Nom: Icon
Type: `[]byte` + +Certains gestionnaires de fenêtres ou environnements de bureau peuvent également le placer dans le cadre de la fenêtre, ou l'afficher dans d'autres contextes. Sur d'autres, l'icône n'est pas du tout utilisée, donc son utilisation peut varier. + +NOTE : Gnome sur Wayland n'affiche pas cette icône. Pour y avoir une icône d'application, un fichier `.desktop` doit être utilisé. Sous KDE, cela devrait fonctionner. + +L’icône doit être fournie dans la taille qu’elle a été dessinée naturellement, c’est-à-dire ne pas redimensionner l’image avant de la passer. La mise à l'échelle est reportée à la dernière minute, lorsque la taille finale désirée est connue, pour permettre une meilleure qualité. + +#### WindowIsTranslucent + +Définir ceci à `true` rendra l'arrière-plan de la fenêtre translucide. Certains gestionnaires de fenêtres peuvent l'ignorer, ou résulter en une fenêtre noire. + +Nom: WindowIsTranslucent
Type: `bool` + +#### WebviewGpuPolicy + +Cette option est utilisée pour déterminer la politique d'accélération matérielle effectuée par webview. + +Nom: WebviewGpuPolicy
Type: [`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
Défaut: `WebviewGpuPolicyAlways` + +##### Type de WebviewGpuPolicy + +| Valeur | Description | +| ------------------------ | ---------------------------------------------------------------------------- | +| WebviewGpuPolicyAlways | L'accélération matérielle est toujours activée | +| WebviewGpuPolicyOnDemand | L'accélération matérielle est activée/désactivée à la demande du contenu web | +| WebviewGpuPolicyNever | L'accélération matérielle est toujours désactivée | + +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` property differs form the executable's filename. + +Name: ProgramName
Type: string
+ +### Debug + +Ceci définit [les options spécifiques au débogage](#Debug) qui s'appliquent aux compilations de débogage. + +Nom: Debug
Type: `options.Debug` + +#### OpenInspectorOnStartup + +Définir cette option à `true` ouvrira l'inspecteur Web au démarrage de l'application. + +Nom: OpenInspectorOnStartup
Type: `bool` + +[^1]: Cela nécessite la prise en charge de WebKit2GTK 2.36+ et votre application doit être construite avec la balise de compilation `webkit2_36` pour activer le support de cette fonctionnalité. Cela augmente aussi la version minnimale de WebKit2GTK à 2.36 pour votre application. +[^2]: Cela nécessite la prise en charge de WebKit2GTK 2.40+ et votre application doit être construite avec la balise de compilation `webkit2_40` pour activer le support de cette fonctionnalité. Cela augmente aussi la version minnimale de WebKit2GTK à 2.40 pour votre application. [ [ ↩](#fnref2:2){.footnote-backref} ↩](#fnref:2){.footnote-backref} diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx new file mode 100644 index 00000000000..a5f067dd870 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx @@ -0,0 +1,92 @@ +--- +sidebar_position: 5 +--- + +# Configuration du projet + +La configuration du projet se trouve dans le fichier `wails.json` du répertoire du projet. La structure de la configuration est : + +```json5 +{ + // Project config version + "version": "", + // The project name + "name": "", + // Relative path to the directory containing the compiled assets, this is normally inferred and could be left empty + "assetdir": "", + // Additional directories to trigger reloads (comma separated), this is only used for some advanced asset configurations + "reloaddirs": "", + // The directory where the build files reside. Defaults to 'build' + "build:dir": "", + // Relative path to the frontend directory. Defaults to 'frontend' + "frontend:dir": "", + // The command to install node dependencies, run in the frontend directory - often `npm install` + "frontend:install": "", + // The command to build the assets, run in the frontend directory - often `npm run build` + "frontend:build": "", + // This command has been replaced by frontend:dev:build. If frontend:dev:build is not specified will falls back to this command. \nIf this command is also not specified will falls back to frontend:build + "frontend:dev": "", + // This command is the dev equivalent of frontend:build. If not specified falls back to frontend:dev + "frontend:dev:build": "", + // This command is the dev equivalent of frontend:install. If not specified falls back to frontend:install + "frontend:dev:install": "", + // This command is run in a separate process on `wails dev`. Useful for 3rd party watchers or starting 3d party dev servers + "frontend:dev:watcher": "", + // URL to a 3rd party dev server to be used to serve assets, EG Vite. \nIf this is set to 'auto' then the devServerUrl will be inferred from the Vite output + "frontend:dev:serverUrl": "", + // Relative path to the directory that the auto-generated JS modules will be created + "wailsjsdir": "", + // The name of the binary + "outputfilename": "", + // The default time the dev server waits to reload when it detects a change in assets + "debounceMS": 100, + // Address to bind the wails dev sever to. Default: localhost:34115 + "devServer": "", + // Arguments passed to the application in shell style when in dev mode + "appargs": "", + // Defines if build hooks should be run though they are defined for an OS other than the host OS. + "runNonNativeBuildHooks": false, + "preBuildHooks": { + // The command that will be executed before a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed before a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed before every build: ${platform} is replaced with the "GOOS/GOARCH". + "*/*": "" + }, + "postBuildHooks": { + // The command that will be executed after a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed after a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed after every build: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. + "*/*": "" + }, + // Data used to populate manifests and version info. + "info": { + // The company name. Default: [The project name] + "companyName": "", + // The product name. Default: [The project name] + "productName": "", + // The version of the product. Default: '1.0.0' + "productVersion": "", + // The copyright of the product. Default: 'Copyright.........' + "copyright": "", + // A short comment of the app. Default: 'Built using Wails (https://wails.app)' + "comments": "" + }, + // 'multiple': One installer per architecture. 'single': Single universal installer for all architectures being built. Default: 'multiple' + "nsisType": "", + // Whether the app should be obfuscated. Default: false + "obfuscated": "", + // The arguments to pass to the garble command when using the obfuscated flag + "garbleargs": "" +} +``` + +This file is read by the Wails CLI when running `wails build` or `wails dev`. + +The `assetdir`, `reloaddirs`, `wailsjsdir`, `debounceMS`, `devserver` and `frontenddevserverurl` flags in `wails build/dev` will update the project config +and thus become defaults for subsequent runs. + +The JSON Schema for this file is located [here](https://wails.io/schemas/config.v2.json). diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx new file mode 100644 index 00000000000..ab2da957dd4 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# Navigateur + +Ces méthodes sont liées au navigateur du système. + +### BrowserOpenURL + +Ouvre l'URL donnée dans le navigateur système. + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..c37399d0bf0 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# Clipboard + +Cette partie du runtime fournit un accès au presse-papiers du système d'exploitation.
L'implémentation actuelle ne gère que le texte. + +### ClipboardGetText + +Cette méthode lit le texte actuellement stocké dans le presse-papiers. + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
Retourne: une chaîne de caractères (si le presse papier est vide, il retournera une chaîne vide) ou une erreur. + +JS: `ClipboardGetText(): Promise`
Retourne : Un promise d'une chaine de caractères (si le presse papier est vide, il retournera une chaîne vide). + +### ClipboardSetText + +Cette méthode écrit du texte dans le presse-papiers. + +Go: `ClipboardSetText(ctx context.Context, text string) error`
Retourne: Une erreur si il y en a une. + +JS: `ClipboardSetText(text: string): Promise`
Retourne: Un promise avec true si le texte a été écrit avec succès dans le presse papier, autrement il contiendra false. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..1bbc6522e0f --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# Boîte de dialogue + +Cette partie du runtime fournit un accès aux boîtes de dialogue natives, telles que les sélecteurs de fichiers et les boîtes de messages. + +:::info JavaScript + +La boîte de dialogue n'est actuellement pas prise en charge dans l'exécuteur JS. + +::: + +### OpenDirectoryDialog + +Ouvre une boîte de dialogue qui invite l'utilisateur à sélectionner un répertoire. Peut être personnalisé en utilisant [OpenDialogOptions](#opendialogoptions). + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Renvoie : le répertoire sélectionné (vide si l'utilisateur a annulé) ou une erreur + +### OpenFileDialog + +Ouvre une boîte de dialogue qui invite l'utilisateur à sélectionner un fichier. Peut être personnalisé en utilisant [OpenDialogOptions](#opendialogoptions). + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Renvoie : le fichier sélectionné (vide si l'utilisateur a annulé) ou une erreur + +### OpenFileDialog + +Ouvre une boîte de dialogue qui invite l'utilisateur à sélectionner plusieurs fichiers. Peut être personnalisé en utilisant [OpenDialogOptions](#opendialogoptions). + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +Renvoie : les fichiers sélectionnés (nil si l'utilisateur a annulé) ou une erreur + +### SaveFileDialog + +Ouvre une boîte de dialogue qui invite l'utilisateur à saisir un nom pour le fichier à enregistrer. Peut être personnalisé en utilisant [SaveDialogOptions](#savedialogoptions). + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +Renvoie : le fichier sélectionné (vide si l'utilisateur a annulé) ou une erreur + +### MessageDialog + +Affiche un message en utilisant une boîte de dialogue. Peut être personnalisé en utilisant [MessageDialogOptions](#messagedialogoptions). + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +Renvoie : Le texte du bouton sélectionné ou une erreur + +## Options + +### OpenDialogOptions + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| Champ | Description | Win | Mac | Lin | +| -------------------------- | -------------------------------------------------------------- | --- | --- | --- | +| DefaultDirectory | Le répertoire que la boîte de dialogue affichera à l'ouverture | ✅ | ✅ | ✅ | +| DefaultFilename | Le nom du fichier par défaut | ✅ | ✅ | ✅ | +| Title | Titre pour la boite de dialogue | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | Une liste de filtres de fichiers | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Afficher les fichiers cachés par le système | | ✅ | ✅ | +| CanCreateDirectories | Autoriser l'utilisateur de créer des dossiers | | ✅ | | +| ResolvesAliases | Si vrai, retourne le fichier et non l'alias | | ✅ | | +| TreatPackagesAsDirectories | Autoriser la navigation dans les dossiers | | ✅ | | + +### SaveDialogOptions + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| Champ | Description | Win | Mac | Lin | +| -------------------------- | -------------------------------------------------------------- | --- | --- | --- | +| DefaultDirectory | Le répertoire que la boîte de dialogue affichera à l'ouverture | ✅ | ✅ | ✅ | +| DefaultFilename | Le nom du fichier par défaut | ✅ | ✅ | ✅ | +| Title | Titre pour la boite de dialogue | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | Une liste de filtres de fichiers | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Afficher les fichiers cachés par le système | | ✅ | ✅ | +| CanCreateDirectories | Autoriser l'utilisateur de créer des dossiers | | ✅ | | +| TreatPackagesAsDirectories | Autoriser la navigation dans les dossiers | | ✅ | | + +### MessageDialogOptions + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| Champ | Description | Win | Mac | Lin | +| ------------- | ---------------------------------------------------------------------------------------------- | -------------- | --- | --- | +| Type | Le type de boîte de dialogue de message: question, info... | ✅ | ✅ | ✅ | +| Title | Titre pour la boite de dialogue | ✅ | ✅ | ✅ | +| Message | Le message à afficher à l'utilisateur | ✅ | ✅ | ✅ | +| Buttons | La liste des noms des boutons | | ✅ | | +| DefaultButton | Le bouton avec ce texte doit être traité comme le bouton par défaut. Lié à la touche `entrée`. | ✅[*](#windows) | ✅ | | +| CancelButton | Le bouton avec ce texte doit être traité permettant d'annuler. Lié à la touche `echap` | | ✅ | | + +#### Windows + +Windows a des boites de dialogue standard dans lesquels les boutons ne sont pas personnalisables. La valeur retournée sera une des valeurs suivantes : "Ok", "Cancel", "Abort", "Retry", "Ignore", "Yes", "No", "Try Again" ou "Continue". + +Pour les boites de dialogues pour les questions, le bouton par défaut est "Oui" et le bouton d'annulation est "Non". Cela peut être modifié en définissant la valeur `DefaultButton` à `"No"`. + +Exemple: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linux a des boites de dialogue standard dans lesquels les boutons ne sont pas personnalisables. La valeur retournée sera de :"Ok", "Cancel", "Yes", "No" + +#### Mac + +Une boîte de dialogue sur Mac peut avoir jusqu'à 4 boutons. Si aucun `DefaultButton` ou `CancelButton` n'est donné, le premier bouton est considéré comme par défaut et est lié à la touche `entrée`. + +Pour le code suivant : + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "C'est ton tour", + Message: "Choisi un nombre", + Buttons: []string{"un", "deux", "trois", "quatre"}, +}) +``` + +le premier bouton est affiché comme étant celui par défaut : + +```mdx-code-block +
+ +
+
+``` + +Et si nous spécifions que le `DefaultButton` est "deux": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +le deuxième bouton est affiché comme étant la réponse par défaut. Lorsque la touche `entrée` est appuyée, la valeur "deux" est retournée. + +```mdx-code-block +
+ +
+
+``` + +Si nous spécifions maintenant `CancelButton` à "trois": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +le bouton avec "trois" est affiché en bas de la boîte de dialogue. Si la touche `echap` est appuyée, la valeur "trois" est retournée: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### DialogType + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### FileFilter + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +Windows vous permet d'utiliser plusieurs filtres de fichiers dans les boîtes de dialogue. Chaque FileFilter s'affichera comme une entrée séparée dans la boîte de dialogue : + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linux vous permet d'utiliser plusieurs filtres de fichiers dans les boîtes de dialogue. Chaque FileFilter s'affichera comme une entrée séparée dans la boîte de dialogue : + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +Les dialogues Mac n'ont qu'un ensemble unique de motifs pour filtrer les fichiers. Si plusieurs filtres de fichiers sont fournis, Wails utilisera tous les modèles définis. + +Exemple: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +Cela se traduira par l'ouverture de la boîte de dialogue "Ouvrir un fichier" en utilisant `*.png,*.jpg,*.mov,*.mp4` comme filtres. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx new file mode 100644 index 00000000000..43d72885ea7 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# Événements + +Le runtime Wails fournit un système d'événements unifiés, où les événements peuvent être émis ou reçus par Go ou JavaScript. En option, des données peuvent être transmises avec les événements. Les écouteurs vont recevoir les données dans les types de donnée du langage correspondant. + +### EventsOn + +Cette méthode définit un listener pour le nom d'événement donné. Lorsqu'un événement de type `eventName` est [émis](#EventsEmit), la méthode définie en callback est déclenchée. Toutes les données supplémentaires envoyées avec l'événement émis seront passées en paramètre à la méthode définie en callback. Il renvoie une fonction pour annuler l'écouteur. + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff + +Cette méthode désactive l'évènement correspondant au nom donné, éventuellement plusieurs évènements peuvent être désinscrits via `additionalEventNames`. + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce + +Cette méthode définit un évènement pour le nom donné, mais ne se déclenchera qu'une seule fois. Il renvoie une fonction pour annuler l'évènement. + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple + +Cette méthode définit un évènement pour le nom donné, mais ne se déclenchera au maximum qu'un certain nombre de fois, définit avec le paramètre `counter`. Il renvoie une fonction pour annuler l'écouteur. + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit + +Cette méthode émet l'événement donné. En option, des données peuvent être transmises avec l'événement. Cela déclenchera tous les événements. + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx new file mode 100644 index 00000000000..915ebc4ec9c --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# Introduction + +Le runtime est une bibliothèque qui fournit des méthodes utilitaires pour votre application. Il y a à la fois un runtime Go et JavaScript et le but est d'essayer de les maintenir à parité dans la mesure du possible. + +Il a des méthodes utilitaires pour : + +- [Fenêtre](window.mdx) +- [Menu](menu.mdx) +- [Boîte de dialogue](dialog.mdx) +- [Événements](events.mdx) +- [Navigateur](browser.mdx) +- [Log](log.mdx) +- [Clipboard](clipboard.mdx) + +Le Go Runtime est disponible en important `github.com/wailsapp/wails/v2/pkg/runtime`. Toutes les méthodes de ce paquet prennent un contexte comme premier paramètre. Ce contexte doit être obtenu à partir des méthodes [OnStartup](../options.mdx#onstartup) ou [OnDomReady](../options.mdx#ondomready). + +:::info Note + +Alors que le contexte sera fourni à la méthode [OnStartup](../options.mdx#onstartup) , il n'y a aucune garantie que l'exécution fonctionnera dans cette méthode car la fenêtre s'initialise dans un autre thread. Si vous souhaitez appeler des méthodes d'exécution au démarrage, utilisez [OnDomReady](../options.mdx#ondomready). + +::: + +La bibliothèque JavaScript est disponible sur le frontend via la carte `window.runtime`. Il y a un package d'exécution généré lors de l'utilisation du mode `dev` qui fournit des déclarations TypeScript pour l'exécution. Ceci devrait être situé dans le répertoire `wailsjs` dans votre répertoire frontend. + +### Cacher + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +Cache l'application. + +:::info Note + +Sur Mac, cela masquera l'application de la même manière que le bouton `Masquer` du menu des applications Mac standard. C'est une manière différente de cacher l'application, mais elle sera toujours au premier plan. Pour Windows et Linux, c'est actuellement la même chose que `WindowHide`. + +::: + +### Afficher + +Affiche l'application. + +:::info Note + +Sur Mac, cela va ramener l'application au premier plan. Pour Windows et Linux, c'est actuellement la même chose que `WindowShow`. + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### Quitter + +Quitte l'application. + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### Environnement + +Renvoie les détails de l'environnement actuel. + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### EnvironmentInfo + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx new file mode 100644 index 00000000000..45b7a07ed27 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# Log + +Le runtime Wails fournit un mécanisme de journalisation qui peut être appelé depuis Go ou JavaScript. Comme la plupart des loggers, il y a un certain nombre de niveaux de log : + +- Trace +- Debug +- Info +- Warning +- Error +- Fatal + +Le logger affichera tous les messages de log au niveau actuel ou supérieur. Exemple : Le niveau `Debug` affichera tous les messages sauf ceux du niveau `Trace`. + +### LogPrint + +Ajoute le message donné dans les logs en tant que message brut. + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf + +Ajoute le message donné dans les logs en tant que message brut. + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace + +Ajoute le message donné dans les logs avec le niveau de log `Trace`. + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef + +Ajoute le message donné dans les logs avec le niveau de log `Trace`. + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug + +Ajoute le message donné dans les logs avec le niveau de log `Debug`. + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf + +Ajoute le message donné dans les logs avec le niveau de log `Debug`. + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo + +Ajoute le message donné dans les logs avec le niveau de log `Info`. + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof + +Ajoute le message donné dans les logs avec le niveau de log `Info`. + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning + +Ajoute le message donné dans les logs avec le niveau de log `Warning`. + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf + +Ajoute le message donné dans les logs avec le niveau de log `Warning`. + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError + +Ajoute le message donné dans les logs avec le niveau de log `Error`. + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf + +Ajoute le message donné dans les logs avec le niveau de log `Error`. + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal + +Ajoute le message donné dans les logs avec le niveau de log `Fatal`. + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf + +Ajoute le message donné dans les logs avec le niveau de log `Fatal`. + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel + +Définit le niveau des logs. En JavaScript, le nombre se rapporte aux niveaux de log suivants : + +| Valeur | Niveau de log | +| ------ | ------------- | +| 1 | Trace | +| 2 | Debug | +| 3 | Info | +| 4 | Warning | +| 5 | Error | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## Utiliser un Logger Personnalisé + +Un logger personnalisé peut être utilisé en le définissant dans l'option de l'application [Logger](../options.mdx#logger) . La seule condition est que le logger implémente l'interface `logger.Logger` définie dans `github.com/wailsapp/wails/v2/pkg/logger`: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx new file mode 100644 index 00000000000..cd78d31aa31 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# Menu + +Ces méthodes sont liées au menu de l'application. + +:::info JavaScript + +Le menu n'est actuellement pas pris en charge lors de l'exécution de JS. + +::: + +### MenuSetApplicationMenu + +Définit le menu de l'application dans le [menu](../menus.mdx) donné. + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu + +Met à jour le menu de l'application, avec toutes les autres modifications déjà appliquées via `MenuSetApplicationMenu`. + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx new file mode 100644 index 00000000000..457c92ebff9 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# Screen + +These methods provide information about the currently connected screens. + +### ScreenGetAll + +Returns a list of currently connected screens. + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### Screen + +Go struct: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript interface: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx new file mode 100644 index 00000000000..e9d915a4f9b --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# Fenêtre + +Ces méthodes donnent le contrôle de la fenêtre de l'application. + +### WindowSetTitle + +Définit le texte dans la barre de titre de la fenêtre. + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen + +Mets la fenêtre en plein écran. + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen + +Restaure les dimensions et la position de la fenêtre avant le plein écran. + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen + +Renvoie vrai si la fenêtre est en plein écran. + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter + +Centre la fenêtre sur le moniteur sur laquelle la fenêtre est actuellement ouverte. + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS + +Exécute du code JS dans la fenêtre. + +Cette méthode exécute le code dans le navigateur de manière asynchrone et retourne immédiatement le résultat. Si le script cause des erreurs, elles ne seront disponibles que dans la console du navigateur. + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload + +Effectue un "rechargement" (Recharge la page courante). + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp + +Recharge le frontend de l'application. + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme + +Windows seulement. + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +Définit le thème de fenêtre à la valeur par défaut du système (sombre/clair). + +### WindowSetLightTheme + +Windows seulement. + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +Définit le thème clair à la fenêtre. + +### WindowSetDarkTheme + +Windows seulement. + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +Définit le thème sombre à la fenêtre. + +### WindowShow + +Affiche la fenêtre, si elle est actuellement masquée. + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowHide + +Masque la fenêtre, si elle est actuellement visible. + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal + +Renvoie vrai si la fenêtre n'est pas minimisée, maximisée ou plein écran. + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize + +Définit la largeur et la hauteur de la fenêtre. + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowGetSize + +Retourne la largeur et la hauteur de la fenêtre. + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize + +Définit la taille minimale de la fenêtre. Redimensionnera la fenêtre si la fenêtre est actuellement plus petite que les dimensions données. + +Définir une taille de `0,0` désactivera cette contrainte. + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize + +Définit la taille maximale de la fenêtre. Redimensionnera la fenêtre si la fenêtre est actuellement plus grande que les dimensions données. + +Définir une taille de `0,0` désactivera cette contrainte. + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop + +Paramètre pour faire en sorte que la fenêtre soit toujours au dessus des autres ou non. + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition + +Définit la position de la fenêtre par rapport au moniteur sur lequel la fenêtre est activée. + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition + +Récupère la position de la fenêtre relative au moniteur sur lequel la fenêtre est activée. + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximiseWindowMaximise + +Maximise la fenêtre pour remplir l'écran. + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise + +Restaure la fenêtre aux dimensions et à la position avant qu'elle soit maximisée. + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised + +Renvoie vrai si la fenêtre est maximisée. + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise + +Option permettant de basculer entre la maximisation de la fenêtre et sa non maximisation. + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise + +Minimise la fenêtre. + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise + +Restaure la fenêtre aux dimensions et à la position avant qu'elle soit minimisée. + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised + +Renvoie vrai si la fenêtre est minimisée. + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour + +Définit la couleur de fond de la fenêtre avec la couleur RGBA donnée. Cette couleur sera visible au travers de tous les pixels transparents. + +Les valeurs valides pour R, G, B et A sont entre 0 et 255 inclus. + +:::info Windows + +Sous Windows, seules les valeurs 0 et 255 sont prises en charge pour A. Toute valeur qui n'est pas 0 sera considérée comme 255. + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## Définitions d'objets TypeScript + +### Position + +```ts +interface Position { + x: number; + y: number; +} +``` + +### Size (taille) + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..cb2b780b665 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# API Chiens + +```mdx-code-block +
+ +
+
+``` + +:::note + +Ce tutoriel a été gracieusement fourni par [@tatadan](https://twitter.com/tatadan) et fait partie de leur [dépôt d'exemples Wails](https://github.com/tataDan/wails-v2-examples). + +::: + +Dans ce tutoriel, nous allons développer une application qui récupère des photos de chiens du web et les affiche. + +### Créer le projet + +Créons l'application. Depuis un terminal saisissez : `wails init -n dogs-api -t svelte` + +Note: Nous pouvons ajouter l'une des options suivantes `-ide vscode` ou `-ide goland` à la fin de la commande si vous voulez ajouter le support d'un IDE. + +Maintenant, exécutons `cd dogs-api` et commençons à éditer les fichiers du projet. + +### Retirer le code inutilisé + +Nous allons commencer par supprimer certains éléments que nous savons que nous n'utiliserons pas : + +- Ouvrez `app.go` et supprimez les lignes suivantes : + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- Ouvrez `frontend/src/App.svelte` et supprimez toutes les lignes. +- Supprimer le fichier `frontend/src/assets/images/logo-universal.png` + +### Créer notre application + +Ajoutons maintenant notre nouveau code Go. + +Ajoutez les déclarations struct suivantes dans `app.go` avant la déclaration des fonctions: + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +Ajouter les fonctions suivantes dans `app.go` (après la déclaration de la fonction déjà existante): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +Modifiez la section `import` de `app.go` pour ressembler à ceci : + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +Ajouter les lignes suivantes dans `frontend/src/App.svelte`: + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### Tester l'application + +Pour générer les liaisons et tester l'application, exécutez `wails dev`. + +### Compiler l'application + +Pour compiler l'application en un seul binaire, exécutez `wails build`. diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx new file mode 100644 index 00000000000..2beff220ec5 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# Hello World + +Le but de ce tutoriel est de vous faire créer l'application la plus basique en utilisant Wails. Vous serez capables de : + +- Créer une nouvelle application Wails +- Compiler l'application +- Démarrer l'application + +:::note + +Ce tutoriel utilise Windows comme plate-forme cible. La sortie variera légèrement selon votre système d'exploitation. + +::: + +## Créer une nouvelle application Wails + +Pour créer une nouvelle application Wails utilisant le template Vanilla JS par défaut, vous devez exécuter la commande suivante : + +```bash +wails init -n helloworld +``` + +Vous devriez voir quelque chose de similaire à ce qui suit : + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +Cela va créer un nouveau dossier nommé `helloworld` dans le dossier actuel. Dans ce dossier, vous trouverez un certain nombre de fichiers : + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## Compiler l'application + +Pour compiler l'application, déplacez-vous dans le dossier du nouveau projet `helloworld` et exécutez la commande suivante : + +```bash +wails build +``` + +Vous devriez voir quelque chose comme ça : + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +Cela a compilé l'application et l'a enregistré dans le dossier `build/bin`. + +## Démarrer l'application + +Si on voit le dossier `build/bin` dans l'explorateur Windows, nous devrions voir le binaire de notre projet : + +```mdx-code-block +
+ +
+
+``` + +On peut l'exécuter en simplement double-cliquant sur le fichier `helloworld.exe`. + +Sur Max, Wails va générer un fichier `helloworld.app` qui peut être exécuté en simplement double-cliquant dessus. + +Sur Linux, vous pouvez exécuter l'application en utilisant `./helloworld` depuis le répertoire `build/bin`. + +Vous devriez voir l'application fonctionner comme prévu : + +```mdx-code-block +
+ +
+
+``` diff --git a/website/versioned_docs/version-v2.5.0/appendix/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/appendix/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/appendix/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/appendix/_category_.json diff --git a/website/versioned_docs/version-v2.5.0/community/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx new file mode 100644 index 00000000000..b7b80b83939 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# リンク + +このページは、コミュニティ関連のリンクの一覧です。 リンクを追加したい場合は、プルリクエストを送信(ページ下部の`このページを編集`をクリック) してください。 + +## Awesome Wails + +Wailsに関する[最高のリンク一覧](https://github.com/wailsapp/awesome-wails)です。 + +## サポートチャネル + +- [Wails Discordサーバ](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [v2ベータディスカッションボード](https://github.com/wailsapp/wails/discussions/828) + +## ソーシャルメディア + +- [Twitter](https://twitter.com/wailsapp) +- [Wails中国語コミュニティQQグループ](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - グループナンバー: 1067173054 + +## その他のチュートリアルや記事 + +- [掲示板を作ってみる](https://blog.customct.com/building-bulletin-board) diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..7cf3d52b5a6 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +

+ +
+

+``` + +[BulletinBoard](https://github.com/raguay/BulletinBoard)は、静的なメッセージや、スクリプト用にユーザから情報を得るためのダイアログを表示する、汎用的なメッセージボードアプリケーションです。 後者はユーザから情報を取得するために使用でき、新しいダイアログを作成するためのTUIを備えています。 システム上で動作し続け、必要に応じて情報を表示・非表示できるように設計されています。 システム上のファイルを監視し、内容が変更されたときにBulletinBoardに送信するという処理もできます。 これは、私のワークフローと相性が良いのです。 なお、プログラムに情報を送信するための[Alfredワークフロー](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow)があります。 また、[EmailIt](https://github.com/raguay/EmailIt)と連携するためのワークフローもあります。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx new file mode 100644 index 00000000000..970e632a2d7 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/)はWails 2で作成されており、9個のノートパット、テキスト操作のためのスクリプト、およびテンプレートを備えた、マークダウンベースのEメール送信ソフトです。 また、システム内のファイルに対してEmailItのスクリプトを実行するためのスクリプトターミナルを備えています。 スクリプトとテンプレートは、コマンドライン自体から、またはAlfred、Keyboard Maestro、Dropzone、PopClipの拡張機能から使用することができます。 さらに、GitHubからダウンロードしたスクリプトやテーマにも対応しています。 ドキュメントは未完成ですが、ブログラム自体は動作します。 Wails 2とSvelteを使用して構築されており、ユニバーサルmacOSアプリケーションとしてダウンロードできます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..b6395c7abd0 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app)は、シンプルで使いやすいPGP暗号化ツールで、すべての連絡先の鍵を管理することができます。 暗号化は簡単にできるべきです。 Wailsで開発されました。** + +PGPを使用したメッセージの暗号化は、業界の標準です。 誰もが、秘密鍵と公開鍵を所持しています。 あなたの秘密鍵は、あなただけがメッセージを読めるように、常に秘密にしておく必要があります。 あなたの公開鍵は、暗号化されたメッセージを送信したい人に配布します。 鍵の管理、メッセージの暗号化、およびメッセージの復号化は、スムーズにできなければなりません。 EncryptEasyはそれらを簡単に実現します。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx new file mode 100644 index 00000000000..a9d9ad651ef --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# FileHound Export Utility + +```mdx-code-block +

+ +
+

+``` + +[FileHound Export Utility](https://www.filehound.co.uk/) FileHoundは、安全なファイル保管、ビジネスプロセスの自動化、およびSmartCapture機能のための、クラウドドキュメント管理プラットフォームです。 + +FileHound Export Utilityを使用すると、FileHoundの管理者はバックアップやリカバリのために、安全なドキュメントとデータ抽出タスクを実行できます。 このアプリケーションは、選択したフィルタにもとづいて、FileHoundに保存されているすべての文書やメタデータをダウンロードします。 メタデータはJSON形式とXML形式の両方でエクスポートされます。 + +バックエンド: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +フロントエンド: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..611c60d5fac --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter)は、シンプルで効率的に使用できる、http APIテストクライアントツールです。 Wails、Go、sveltejsで構築されました。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..668173ebe9e --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Minecraft Updater + +```mdx-code-block +

+ +
+

+``` + +[Minecraft Updater](https://github.com/Gurkengewuerz/MinecraftModUpdater)は、ユーザベースMinecraft Modを更新および同期するためのユーティリティツールです。 Wails2とReactで構築されており、フロントエンドフレームワークには[antd](https://ant.design/)を使用しています。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..a8020f6be3a --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Modal File Manager + +```mdx-code-block +

+ +
+

+``` + +[Modal File Manager](https://github.com/raguay/ModalFileManager)は、Web技術を使用した、デュアルペインファイルマネージャです。 [こちら](https://github.com/raguay/ModalFileManager-NWjs)でもともと公開していたNW.jsで構築されたものをベースとしています。 本バージョンでは、フロントエンドのコードには前と同様にSvelteを使用(NW.jsを使用していた時から大幅な修正はありました)しましたが、バックエンドの実装にはWails 2を使用しました。 この実装によって、コマンドラインの`rm`、`cp`などのコマンドを使用しなくなりましたが、テーマや拡張機能のダウンロードのために、システム上でgitのインストールは必要となります。 Goで完全にコード化されており、以前のバージョンよりもはるかに高速に実行されます。 + +このファイルマネージャは、Vimと同じ、状態制御キーボード操作の原理に基づいて設計されています。 状態の数に制限はなく、とてもプログラマブルです。 つまり、無数にキーボード構成を作成・使用することができます。 この点は、他のファイルマネージャとの主な違いと言えるでしょう。 また、GitHubからテーマや拡張機能をダウンロードできます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..00159268f08 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molly Wallet](https://github.com/grvlle/constellation_wallet/)は、Constellation Networkの公式$DAGウォレットです。 ユーザは、$DAGトランザクションの生成に限らず、様々な方法でHypergraph Networkとやり取りすることができます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx new file mode 100644 index 00000000000..7b5f59d15fb --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[October](https://october.utf9k.net)は、[Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader)からハイライトを抽出し、簡単に[Readwise](https://readwise.io)へ転送できる、シンプルなWailsアプリケーションです。 + +[UPX圧縮](https://upx.github.io/)なしで、すべてのプラットフォームで10MB以下という、比較的小さなサイズとなっています! + +これとは対照的に、開発者が以前にElectronで作成したものは、簡単に数百MBにまで膨らんでしまいました。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx new file mode 100644 index 00000000000..b02394d2779 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# Optimus + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus)は、画像最適化のためのデスクトップアプリケーションです。 WebP、JPEG、PNG形式画像の相互変換や圧縮をサポートしています。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx new file mode 100644 index 00000000000..3066a9799e9 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# Portfall + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - すべてのクラスタUIに簡単にアクセスできる、デスクトップk8s port-forwardingポータルです。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..348ffdd5290 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - resticリポジトリを参照およびリストアするための、シンプルでクロスプラットフォームな[restic](https://github.com/restic/restic)バックアップGUIツールです。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..b1e66ae7b8c --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +誰にとっても簡単、安全、そして無料のファイル共有。 詳しくは[Riftshare.app](https://riftshare.app)をご覧ください。 + +## 機能 + +- ローカルネットワーク内またはインターネット経由で、簡単にセキュアなコンピュータ間ファイル共有が可能 +- [magic wormhole プロトコル](https://magic-wormhole.readthedocs.io/en/latest/)を介したファイル・ディレクトリのセキュアな送信をサポート +- magic wormholeを使用する他のすべてのアプリ(magic-wormhole、wormhole-william CLI、wormhole-guiなど) との互換性 +- 選択された複数ファイルを自動圧縮して一括送信 +- フルアニメーション、プログレスバー、および送受信のキャンセルをサポート +- OSネイティブのファイル選択UIを利用可能 +- 受信したファイルをワンクリックで開ける +- 自動アップデート - 最新のリリースを気にする必要はありません! diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..9ab6d88d51e --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp)は、スクリプトまたは[Node-Red](https://nodered.org)サーバの出力を表示するためのプログラムです。 EmailItプログラムで定義されたプログラムを実行し、出力を表示します。 xBarやTextBarのスクリプトも使用できますが、現時点ではTextBarのスクリプトが正常に動作します。 また、システム上にスクリプトの出力を表示できます。 ScriptBarはそれらの情報をメニューバーには表示させず、簡単に参照できる便利なウィンドウ上に表示されます。 複数のタブを使用して、様々な情報を表示できます。 よくアクセスするWebサイトへのリンクを配置することもできます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx new file mode 100644 index 00000000000..bcca578edcc --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/)は、ブロックチェーン技術を使用して、100%匿名でのファイル転送ができるように設計された、p2pファイル共有アプリです。 エンドツーエンド暗号化に対応、分散型、そしてオープンソースである点が、Surgeの特徴です。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx new file mode 100644 index 00000000000..d44b2280183 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally)は、[Ergodox](https://ergodox-ez.com/)キーボード公式の、ファームウェアフラッシャーです。 とても見栄えが良く、GoのパワーとWeb開発技術の豊富なグラフィカルツールを組み合わせるという、Wailsで達成できる機能を示す良い例となるアプリです。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx new file mode 100644 index 00000000000..6674a7d1652 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# Minecraft launcher for WarMine + +```mdx-code-block +

+ + +
+

+``` + +[Minecraft launcher for WarMine](https://warmine.ru/)は、Modを導入したゲームサーバに簡単にアクセスしたり、ゲームアカウントを管理できるWailsアプリケーションです。 + +ランチャーでは、ゲームファイルをダウンロードし、その整合性のチェックして、様々なカスタマイズが可能なバックエンドからの起動引数を使用してゲームを起動します。 + +フロントエンドはSvelteで記述され、ランチャー全体は9MBに収まり、Windows 7-11をサポートしています。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx new file mode 100644 index 00000000000..eedc72d97e8 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# Wombat + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat)はクロスプラットフォーム対応のgRPCクライアントです。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx new file mode 100644 index 00000000000..94e0e09f2ab --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails)はYouTubeからトラックをダウンロードしたり、オフラインプレイリストを作成して友達と共有するためのアプリです。友達は、あなたのプレイリストを内蔵プレーヤーで再生したり、オフラインで聴くためにダウンロードしたりできます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx new file mode 100644 index 00000000000..3dc92d2fbd9 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# テンプレート + +このページでは、コミュニティがサポートしているテンプレートを紹介しています。 このページに新たにテンプレートを含めたい場合は、このページの下側にある`このページを編集`をクリックして、プルリクエストを出してください。 独自テンプレートの作成方法については、[テンプレート](../guides/templates.mdx)ガイドをご覧ください。 + +これらのテンプレートを使用するには、`wails init -n "プロジェクト名" -t [テンプレートのリンク[@バージョン]]`コマンドを実行してください。 + +バージョンサフィックスが無い場合は、デフォルトで、メインブランチのコードテンプレートが使用されます。 バージョンサフィックスがある場合は、当該バージョンのタグに対応するコードテンプレートが使用されます。 + +例: `wails init -n "プロジェクト名" -t https://github.com/misitebao/wails-template-vue` + +:::warning 注意 + +**Wailsプロジェクトでは、サードパーティ製テンプレートのメンテナンスは行っておらず、責任も負いません!** + +テンプレートについてよく分からない場合は、`package.json`および`wails.json`を確認し、どのようなスクリプトが実行されるのかや、どのようなパッケージがインストールされるのかを調べてください。 + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - VueのエコシステムをベースにしたWailsテンプレート (TypeScript、ダークテーマ、i18n、シングルページルーティング、TailwindCSSを統合) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Viteを使用したVue 3 TypeScript (および機能を追加する手順) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vite、Vuex、Vue Router、SaaS、ESLint + Prettier を使用した Vue 3 TypeScript +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) を使用したテンプレート +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, <script setup>によるComposition API) を使用したテンプレート +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Naive UI(Vue3のコンポーネントライブラリ)をベースにしたWailsテンプレート + +## Angular + +- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - 15以上のAngularアクションを含み、すぐに実装に取り掛かれるテンプレート +- [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - TypeScript、Sass、ホットリロード、コード分割、i18n を使用した Angular + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - reactjsを使用したテンプレート +- [wails-react-template](https://github.com/flin7/wails-react-template) - ライブ開発をサポートしたReactの最小テンプレート +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - Next.js、TypeScript を使用したテンプレート +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - React + TypeScript + Vite + TailwindCSSを使用したテンプレート +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - Vite、React、TypeScript、TailwindCSS、shadcn/uiを使用したテンプレート + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - Svelteを使用したテンプレート +- [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - SvelteおよびViteを使用したテンプレート +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - TailwindCSS v3を含んだ、SvelteおよびViteを使用したテンプレート +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - Svelte v4.2.0、Vite、TailwindCSS v3.3.3を使用したテンプレート +- [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - SvelteKitを使用したテンプレート + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - Solid + Ts + Viteを使用したテンプレート +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - Solid + Js + Viteを使用したテンプレート + +## Elm + +- [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - 関数型プログラミングと**高速な**ホットリロードを使ったGUIアプリ開発 :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Elm + Tailwind CSS + Wailsのパワー:muscle:を組み合わせたテンプレート (ホットリロードサポートあり) + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - インタラクティビティを実現するためのピュアなhtmxに、コンポーネントおよびフォームを作成するためのテンプレートが合わさったテンプレート + +## ピュアJavaScript (バニラ) + +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - 基本的なJavaScript、HTML、CSSのみを含むテンプレート diff --git a/website/versioned_docs/version-v2.5.0/gettingstarted/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/gettingstarted/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx new file mode 100644 index 00000000000..dfab958b4e8 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# プロジェクトのコンパイル + +プロジェクトディレクトリ上で、`wails build`コマンドを実行しましょう。 そうすることで、プロジェクトがコンパイルされ、`build/bin`ディレクトリ内に本番配布用のバイナリが出力されます。 + +バイナリを起動すると、デフォルト仕様のアプリを確認することができます: + +```mdx-code-block +
+ +
+
+``` + +コンパイルオプションについて詳しくは、[CLIリファレンス](../reference/cli.mdx#build)をご覧ください。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx new file mode 100644 index 00000000000..5fbc7ab19cc --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# アプリの開発 + +プロジェクトディレクトリのルート上で`wails dev`コマンドを実行すると、アプリを開発モードで起動することができます。 コマンドを実行すると下記の処理が実行されます: + +- アプリをビルドしたのち、起動する +- Goのコードをフロントエンドにバインドして、JavaScriptから呼び出せるようにする +- [Vite](https://vitejs.dev/)の機能を使用して、Goファイルの変更を監視し、変更された際には自動的にリビルド/再起動する +- ブラウザからアプリを操作できるようにする[Webサーバ](http://localhost:34115)を立ち上げる。 これにより、任意のブラウザ拡張機能を利用できる。 JavascriptのコンソールからGoのコードを呼び出すこともできる + +アプリ開発を始めるときは、プロジェクトディレクトリ上で`wails dev`コマンドを実行しましょう。 詳しくは、[こちら](../reference/cli.mdx#dev)をご覧ください。 + +近日中にチュートリアルを公開予定です。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..5dce70b33ea --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# プロジェクトの開始 + +## プロジェクトの生成 + +CLIのインストールが終わったら、`wails init`コマンドで新しいプロジェクトを生成しましょう。 + +好きなフレームワークを選択してください: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + JavaScriptによるSvelteプロジェクトを生成する場合:

+ + wails init -n myproject -t svelte + +TypeScriptによるプロジェクトを生成する場合:
+ + wails init -n myproject -t svelte-ts + +
+ + JavaScriptによるReactプロジェクトを生成する場合:

+ + wails init -n myproject -t react + +TypeScriptによるプロジェクトを生成する場合:
+ + wails init -n myproject -t react-ts + +
+ + JavaScriptによるVueプロジェクトを生成する場合:

+ + wails init -n myproject -t vue + +TypeScriptによるプロジェクトを生成する場合:
+ + wails init -n myproject -t vue-ts + +
+ + JavaScriptによるPreactプロジェクトを生成する場合:

+ + wails init -n myproject -t preact + +TypeScriptによるプロジェクトを生成する場合:
+ + wails init -n myproject -t preact-ts + +
+ + JavaScriptによるLitプロジェクトを生成する場合:

+ + wails init -n myproject -t lit + +TypeScriptによるプロジェクトを生成する場合:
+ + wails init -n myproject -t lit-ts + +
+ + JavaScriptによるバニラなプロジェクトを生成する場合:

+ + wails init -n myproject -t vanilla + +TypeScriptによるプロジェクトを生成する場合:
+ + wails init -n myproject -t vanilla-ts + +
+
+``` + +
+ +様々な機能やフレームワークを提供する[コミュニティテンプレート](../community/templates.mdx)を利用することもできます。 + +プロジェクト生成時に使用可能なオプションを確認するには、`wails init -help`を実行してください。 詳しくは、[CLIリファレンス](../reference/cli.mdx#init)を参照してください。 + +## プロジェクトのディレクトリ構成 + +Wailsのプロジェクトディレクトリの構成は次のとおりです: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### プロジェクトの構造 + +- `/main.go` - アプリのメインコード +- `/frontend/` - フロントエンドのプロジェクトディレクトリ +- `/build/` - ビルドディレクトリ +- `/build/appicon.png` - アプリアイコン +- `/build/darwin/` - Mac固有のプロジェクトディレクトリ +- `/build/windows/` - Windows固有のプロジェクトディレクトリ +- `/wails.json` - プロジェクト構成ファイル +- `/go.mod` - Goモジュール定義ファイル +- `/go.sum` - Goモジュールチェックサムファイル + +`frontend`ディレクトリ内は、Wailsで決まったファイル構成等は無く、お好きなフロントエンドプロジェクトを配置することができます。 + +`build`ディレクトリは、アプリのビルド時に使用されます。 この中のファイルは、ビルドの挙動をカスタマイズするために、適宜ファイル内容を書き換えることができます。 buildディレクトリ内のファイルを削除すると、デフォルトのファイルが再生成されます。 + +`go.mod`のモジュール名は、最初は"changeme"になっています。 このモジュール名は、あなたのプロジェクトに適切な名前に変更しましょう。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx new file mode 100644 index 00000000000..bb00dbe8fd5 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx @@ -0,0 +1,89 @@ +--- +sidebar_position: 1 +--- + +# インストール + +## サポートされているプラットフォーム + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## 依存関係 + +Wailsをインストールする前に、下記のものを導入しておく必要があります。 + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +[Goのダウンロードページ](https://go.dev/dl/)からGoをダウンロードしてください。 + +公式の[Goインストール手順](https://go.dev/doc/install)に従って、Goをインストールしてください。 その際、`PATH`環境変数に`~/go/bin`ディレクトリへのパスが含まれていることも確認してください。 それらが終わったら、ターミナルを再起動し、以下の確認をしてください: + +- Goが正しくインストールされているかを確認する: `go version` +- "~/go/bin"のディレクトリパスがPATH環境変数に含まれているか確認する: `echo $PATH | grep go/bin` + +### NPM + +[Nodeダウンロードページ](https://nodejs.org/ja/download/)からNPMをダウンロードしてください。 最新版を利用することをお勧めします。なぜなら、私たちは最新版に対してテストを実施しているためです。 + +`npm --version`を実行して、インストールが完了しているかを確認してください。 + +## プラットフォーム固有の依存関係 + +開発作業を行うプラットフォームによって、必要な依存関係が存在します: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wailsを使用するには、xcodeのコマンドラインツールがインストールされている必要があります。 xcode-select --installコマンドを実行することでインストールできます。 + + + Wailsを使用するには、WebView2ランタイムがインストールされている必要があります。 最新のWindowsでは、すでにインストールされている場合もあります。 wails doctorコマンドで、インストール状況を確認できます。 + + + Linuxでは、標準のgccビルドツール、libgtk3libwebkitが必要です。 Wailsは、様々なディストリビューション向けに大量のコマンドを列挙することはせず、現在使用されているディストリビューションのインストールコマンドを自動的に判定します。 Wailsをインストールした後に、wails doctorコマンドを実行して、別途インストールが必要な依存関係を確認してください。 あなたが利用しているディストリビューションやパッケージマネージャーがサポートされていない場合は、Linuxディストリビューションサポートガイドを参照してください。 + + +``` + +## 任意の依存関係 + +- [UPX](https://upx.github.io/)を導入することで、構築したアプリを圧縮できます。 +- [NSIS](https://wails.io/docs/guides/windows-installer/)を導入することで、Windowsのインストーラを生成できます。 + +## Wailsのインストール + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest`を実行して、Wails CLIをインストールしてください。 + +注意: 次のようなエラーが発生した場合: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +下記コマンドで、Go 1.18以上がインストールされているかを確認してください: +```shell +go version +``` + +## システムチェック + +`wails doctor`を実行すると、必要な依存関係が正しくインストールされているかを確認することができます。 正しくインストールされていない場合は、その内容をあなたにお知らせして、どうすれば解決できるかを教えてくれます。 + +## `wails`コマンドが見つからないのですが? + +`wails`コマンドが見つからないとシステムに怒られた場合は、Goが、公式のGoインストール手順に従って導入されているかを確認してください。 コマンドが見つからないほとんどの理由は、あなたのホームディレクトリ配下にある`go/bin`ディレクトリのパスが、`PATH`環境変数に含まれていないからです。 また、インストールによって行われた環境変更を反映させるために、もともと開いていたコマンドプロンプト(ターミナル)がある場合はそれらをいったん閉じて、再度開きなおしてください。 diff --git a/website/versioned_docs/version-v2.5.0/guides/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx new file mode 100644 index 00000000000..2a049fd1e58 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +Wailsでは、Angular向けのテンプレートは用意されていませんが、Angular自体を使用することはできます。 + +## 開発モード + +Angularで開発モードを使用するには、`wails.json`で下記のように設定してください: + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx new file mode 100644 index 00000000000..850d85f382b --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx @@ -0,0 +1,215 @@ +# アプリケーション開発 + +Wailsでアプリケーションを作成する際に、厳格なルールはありませんが、基本的なガイドラインがいくつかあります。 + +## アプリケーションセットアップ + +デフォルトのテンプレートを使用した場合、`main.go`にはアプリケーションの構成および起動コードが記述され、`app.go`にはアプリケーションのロジック定義が記述されています。 + +`app.go`ファイルには、メインアプリケーションのフックとして機能する2つのメソッドを持った構造体を定義します。 + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- startupメソッドは、Wailsが起動時に、必要リソースの割り当てが完了した際にすぐに呼び出されます。このメソッド内は、リソースの作成、イベントリスナーの設定、その他アプリケーションに必要な初期処理を実施するのに適しています。 メソッド呼び出し時に与えられる`context.Context`は、通常は、構造体のフィールドに格納するようにしてください。 なぜならば、このcontextは[runtime](../reference/runtime/intro.mdx)を呼び出す際に必要なためです。 このstartupメソッドがエラーを返却した場合、アプリケーションは終了します。 開発モードの場合、コンソールにエラーが出力されます。 + +- shutdownメソッドは、シャットダウンプロセスの最後に、Wailsによって呼び出されます。 このメソッド内は、メモリを解放し、任意のシャットダウンタスクを実施するのに適しています。 + +通常、`main.go`ファイルでは、アプリケーション構成を受け取る`wails.Run()`メソッドを1回だけ呼び出すようにします。 テンプレートでは、`wails.Run()`を呼び出す前に、`app.go`で定義した構造体のインスタンスを作成して、`app`に格納しています。 アプリケーション構成では、コールバックを追加できます: + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +アプリケーションのライフサイクルフックについて詳しくは、[こちら](../howdoesitwork.mdx#application-lifecycle-callbacks)をご覧ください。 + +## メソッドのバインディング + +フロントエンドからGoのメソッドを呼び出したいことがありますよね。 そのようなときは、`app.go`ですでに定義している構造体に、パブリックメソッドを追加しましょう: + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +アプリケーション構成の`Bind`キーで、Wailsに何をバインドさせたいかを指定できます: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +これにより、`App`構造体のすべてのパブリックメソッドがバインドされます(startupメソッド・shutdownメソッドはバインドされません)。 + +### 複数の構造体をバインドするときのcontextの扱い + +もし、複数の構造体のメソッドをバインドしたくて、かつ、各構造体でランタイム関数を呼び出せるようにcontextを共有したい場合は、`OnStartup`のコールバック内で、各構造体にcontextを渡してあげるのが良いでしょう: + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +バインディングの詳細については、[こちら](../howdoesitwork.mdx#method-binding)をご覧ください。 + +## アプリケーションメニュー + +Wailsでは、アプリケーションにメニューを追加することができます。 追加したい場合は、アプリケーション構成に[Menu](../reference/menus.mdx#menu)構造体を渡してください。 通常は、Menuを返すようなメソッドを使用するようにします。ライフサイクルフックに使用される`App`構造体のメソッドとして用意するのが良いでしょう。 + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## アセット + +Wails v2にアセットを処理させるための方法は複雑ではありません。 Wailsに与える必要があるのは、`embed.FS`だけです。 任意の方法で与えてあげてください。 バニラテンプレートで使用されているように、素のhtml/css/jsファイルをアセットとして使用できます。 もちろん、複雑なビルドシステムを使用することも可能ですが、必須要件ではありません。 + +`wails build`コマンドを実行すると、プロジェクトルートディレクトリに存在するプロジェクト構成ファイルである`wails.json`のチェックが行われます。 プロジェクトファイルには、次の2つのキーを含めることができます: + +- "frontend:install" +- "frontend:build" + +1番目のキーは、`frontend`ディレクトリにおいて、nodeモジュールをインストールするためのコマンドを指定します。 2番目のキーは、`frontend`ディレクトリにおいて、フロントエンドをビルドするためのコマンドを指定します。 + +2つのキーのどちらともが指定されていない場合、Wailsはフロントエンドプロジェクトに対する操作を一切行いません。 `embed.FS`で指定されたディレクトリを参照するのみとなります。 + +### アセットハンドラ + +Wails v2では必要に応じて、`options.App`の中に`http.Handler`を定義することができます。これにより、アセットサーバにフックして、その場でファイルを作成したり、POST/PUTリクエストを処理したりすることができます。 GETリクエストが要求されたときは、まず初めに、`assets` FSにハンドリングされます。 リクエストされたファイルをFSが見つけられなかった場合、そのリクエストは`http.Handler`に転送されます。 GET以外のリクエストは、`AssetsHandler`が指定されていれば、当該ハンドラによって直接処理されます。 なお、`Assets`オプションに`nil`を指定することで、`AssetsHandler`のみを使用することも可能です。 + +## ビルトイン開発サーバ + +`wails dev`コマンドを実行すると、ビルトイン開発サーバが起動し、プロジェクトディレクトリ内のファイル監視が開始されます。 デフォルトでは、ファイルの中身が変更された場合、Wailsはそのファイルがアプリケーションファイルであるかどうかをチェックします(デフォルト: `.go`、 `-e`フラグで制御可能)。 アプリケーションファイルであった場合は、アプリケーションをリビルドして再起動します。 アセットディレクトリ内でファイルが変更された場合は、少し経ってからリロードが実行されます。 + +開発サーバは"デバウンシング"と呼ばれるテクニックを使用しています。これにより、短時間で複数のファイルが更新されたとしても、すぐに再読み込みされることはありません。 トリガーが起動すると、再読み込みを実行する前に一定時間待機します。 待機中に別のトリガーが起動した場合、待機時間はリセットされます。 デフォルトでは、待機時間は`100ms`です。 この待機時間の値があなたのプロジェクトで適切ではない場合、`-debounce`フラグを使用して変更することができます。 このフラグを使用すると、値がプロジェクト構成ファイルに保存され、次回以降のデフォルト値となります。 + +## 外部開発サーバ + +フレームワークによっては独自のライブリロードサーバが付属しているものがありますが、それらはWailsのGoバインディングを使用することができません。 このような場面では、Wailsが監視するビルドディレクトリ内で、プロジェクトをリビルドする監視スクリプトを実行すると良いでしょう。 例としては、[rollup](https://rollupjs.org/guide/en/)を使用するデフォルトのsvelteテンプレートをご覧ください。 + +### Create React App + +The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: + +```html + + + + + +``` + +This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. + +## Goモジュール + +デフォルトのWailsテンプレートは、モジュール名が"changeme"となっている`go.mod`ファイルを生成します。 プロジェクトの生成が完了したら、適切なモジュール名に変更してください。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..a765cff58f6 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# 動的アセット + +[AssetsHandler](../reference/options#assetshandler)オプションを使用することで、フロントエンドアセットを動的に読み込んだり生成させることができます。 AssetsHandlerは、ジェネリックな`http.Handler`です。アセットサーバへのGET以外のすべてのリクエスト、および、アセットファイルとして存在しなかったがために転送されてきたGETリクエストに対して、ハンドラが呼び出されます。 + +カスタムされたAssetsHandlerを導入すると、カスタムアセットサーバを使用して独自のアセットを提供することができます。 + +## 例 + +このサンプルプロジェクトでは、ディスクからファイルを読み込むシンプルなアセットハンドラを作成します: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // App構造体のインスタンスを作成 + app := NewApp() + + // オプション付きでアプリケーションを作成 + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +`wails dev`コマンドでアプリケーションを開発モードで実行すると、次のように出力されます: + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +これを見ると分かるように、アセットハンドラは、デフォルトのアセットサーバが`favicon.ico`を提供できない場合に呼び出されています。 + +メインアプリケーションをクリックし、"検証(開発者ツールで調査する)"を選択して、開発者ツールを起動します。そして、コンソールに次のように入力して実行することで、機能をテストできます: + +``` +let response = await fetch('does-not-exist.txt'); +``` + +これにより、開発者ツールでエラーが発生します。 カスタムアセットハンドラによって、想定どおりのエラーが返却されていることが分かります。 + +```mdx-code-block +

+ +

+``` + +もし、`go.mod`をリクエストした場合は、次のような出力となります: + +```mdx-code-block +

+ +

+``` + +このテクニックを使用して、ページ上に画像を直接読み込ませることができます。 デフォルトのバニラテンプレートで、ロゴ画像を指定している、下記コードを、 + +```html + +``` + +次のように書き換えます: + +```html + +``` + +すると、画面上は次のような表示となります: + +```mdx-code-block +

+ +

+``` + +:::warning + +この例のようにファイルシステムを公開することは、セキュリティ上のリスクとなります。 ファイルシステムへのアクセスは、適切に管理することを推奨します。 + +::: diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..167955b12d7 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,244 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx new file mode 100644 index 00000000000..b6218d81e1e --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx @@ -0,0 +1,87 @@ +# フレームレスアプリケーション + +Wailsでは、フレームのないアプリケーションの作成をサポートしています。 [アプリケーションオプション](../reference/options.mdx#application-options)の[frameless](../reference/options.mdx#frameless)フィールドで設定することが可能です。 + +Wailsでは、フレームレスウィンドウをドラッグできるようにするための簡単な仕組みを提供しています。`--wails-draggable:drag`というCSSスタイルをもつ任意のHTML要素は、"ドラッグハンドル"として機能します。 このプロパティは、すべての子要素にも継承されます。 ネストされている子要素で、ドラッグを明示的に無効化したい場合は、当該子要素で'--wails-draggable:no-drag'を指定します。 + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +一部のプロジェクトでは、動的なスタイリングが原因で、これらのCSSプロパティが使用できない場合があるでしょう。 そのような場合、アプリケーションオプションの`CSSDragProperty`および`CSSDragValue`を使用して、ドラッグ可能な領域を示すプロパティ名と値を独自に定義することができます。 + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // アプリケーション構造体のインスタンスを作成 + app := NewApp() + + // オプション付きでアプリケーションを作成 + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info フルスクリーン + +アプリケーションをフルスクリーン状態にすると、このドラッグ機能は無効になります。 + +::: diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx new file mode 100644 index 00000000000..2d976fbf741 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx @@ -0,0 +1,72 @@ +# フロントエンド + +## スクリプトの注入 + +Wailsが`index.html`を提供するとき、デフォルトでは、``タグ内に`/wails/ipc.js`および`/wails/runtime.js`の2つのスクリプトを読み込む行を注入します。 これらのスクリプトファイルは、バインディングおよびランタイムを提供しています。 + +デフォルトでどの位置にスクリプトが注入されるかを、下記のコードに示します: + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### デフォルトのスクリプト注入の上書き + +開発者がより柔軟に対応できるように、これらの動作をカスタマイズするためのメタタグが用意されています: + +```html + +``` + +オプションは次のとおりです: + +| 値 | 説明 | +| ------------------- | ------------------------------- | +| noautoinjectruntime | `/wails/runtime.js`の自動注入を無効化します | +| noautoinjectipc | `/wails/ipc.js`の自動注入を無効化します | +| noautoinject | すべてのスクリプトの自動注入を無効化します | + +カンマで区切ることで、複数のオプションを指定できます。 + +次のコードは、自動注入版のコードと完全に同じ挙動となります: + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx new file mode 100644 index 00000000000..daf0c4802fa --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx @@ -0,0 +1,127 @@ +# 統合開発環境 + +Wailsでは、優れた開発体験の提供を目指しています。 その一環として、統合開発環境特有の構成設定の生成をサポートしており、より快適にプロジェクトをセットアップできるようになっています。 + +現在は[Visual Studio Code](https://code.visualstudio.com/)をサポートしていますが、今後、Golandなど他の統合開発環境もサポートしていく予定です。 + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +プロジェクト生成時のコマンドに`-ide vscode`フラグを付与すると、統合開発環境用の構成ファイルが、他のプロジェクトファイルと共に作成されます。 これらのファイルは`.vscode`ディレクトリ内に配置され、アプリケーションをデバッグするための正しい構成設定が記述されています。 + +生成されるのは`tasks.json`ファイルおよび`launch.json`ファイルの2つです。 デフォルトのバニラプロジェクトの場合、次のような内容となります: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### インストールおよびビルドステップの構成 + +デフォルトのプロジェクトでは、`npm install`や`npm run build`などのステップが必要ないため、`tasks.json`ファイルの内容なシンプルなものになっています。 Svelteテンプレートのように、フロントエンドのビルドステップが必要なプロジェクトの場合、`tasks.json`を編集して、インストールおよびビルドのステップを追加する必要があります。 + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +今後の機能強化予定 + +将来的に、インストールおよびビルドステップを含んだ`tasks.json`を自動生成できるように計画しています。 + +::: diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..4d9b9640025 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Linuxディストリビューションサポート + +## 概要 + +WailsはLinuxをサポートしていますが、すべてのLinuxディストリビューションに対してインストール手段を提供することは困難であるため、行っていません。 その代わりにWailsでは、アプリケーション開発に必要なパッケージが、システムのパッケージマネージャを介して利用可能かどうかを判断します。 現在、以下のパッケージマネージャをサポートしています: + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## パッケージ名の追加 + +あなたが使用しているディストリビューションが、Wailsがサポートしているパッケージマネージャを使用していたとしても、パッケージ名が想定と異なっている場合があります。 例えば、Ubuntuの派生ディストリビューションを使用している場合、gtkのパッケージ名が異なる場合があります。 Wailsは、パッケージ名のリストを繰り返し処理することで、正しいパッケージを見つけようと試みます。 パッケージ名のリストは、`v2/internal/system/packagemanager`ディレクトリ内に、パッケージマネージャごとのファイルで格納されています。 下記は、`v2/internal/system/packagemanager/apt.go`の例です。 + +このファイルでは、パッケージのリストは`Packages()`メソッドによって定義されています: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +お使いのLinuxディストリビューションが、`libgtk-3`パッケージが`lib-gtk3-dev`という名前で配布されていると仮定しましょう。 その場合、次の行を追加することで、対応することができます: + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## パッケージマネージャの追加 + +新しいパッケージマネージャを追加するには、次の手順を実行します: + +- `v2/internal/system/packagemanager`に、`.go`という名前の新しいファイルを作成します。``はパッケージマネージャの名前です。 +- `pm.go`の中で、PackageManagerインターフェースを実装した構造体を定義します: + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()`メソッドで、パッケージマネージャの名前を返すように実装します。 +- `Packages()`メソッドで、依存関係の候補ファイル名を提供する`packagemap`を返すように実装します。 +- `PackageInstalled()`メソッドで、引数で渡されたパッケージがすでにインストールされている場合は`true`を返すように実装します。 +- `PackageAvailable()`メソッドで、引数で指定されたパッケージがインストールはされていないけれどもインストール可能であるなら`true`を返すように実装します。 +- `InstallCommand()`メソッドで、引数で指定されたパッケージをインストールするための正確なコマンドを返すように実装します。 + +他のパッケージマネージャのコードを参考にして、これらのコードがどのように動作するか把握してください。 + +:::info リマインド + +新しいパッケージマネージャを追加した場合は、このページも忘れずに更新してください。 + +::: diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx new file mode 100644 index 00000000000..e726244fc6e --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +このページでは、Linux向けのWailsアプリケーション開発に関する、様々なガイドを掲載しています。 + +## videoタグが"ended"イベントを発火しない + +videoタグを使用している場合、ビデオの再生が終了しても"ended"イベントが発火しません。 これはWebkitのバグであり、次のような回避策を講じることで修正できます: + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +出典: [ディスカッションボード](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275)における[Lyimmi](https://github.com/Lyimmi)のコメント diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx new file mode 100644 index 00000000000..f01fba42878 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx @@ -0,0 +1,55 @@ +# ローカル開発 + +## 概要 + +Wailsは現在も継続的に開発されており、定期的に、新しいリリースに対して"タグ付け"が行われています。 "タグ付け"は通常、`master`ブランチに存在する新しいコードがすべてテストされ、正常な動作が確認されたときに行われます。 まだリリースされていない新機能やバグフィックスを利用したい場合、以下の手順で"ブリーディング・エッジ"版を使用できます: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +注意: このページでは、Wailsプロジェクトをクローンしたディレクトリのことを"clonedir"と記します。 + +この手順により、Wails CLIは一番最新のバージョンとなります。 + +### 自身のプロジェクトの更新 + +クローンした最新バージョンのWailsライブラリを自身のプロジェクトで使用させたい場合、`go.mod`ファイルの末尾に次の行を追記してください: + +`replace github.com/wailsapp/wails/v2 => ` + +例: + +Windowsの場合: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +'nixの場合: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +安定バージョンに戻す場合は、次のコマンドを実行します: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## ブランチのテスト + +先述の手順に従えば、ブランチをテストすることができますが、インストールを実行する前に、テスト対象のブランチに切り替えるようにしてください: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +そして、先述のとおり[自身のプロジェクトを更新](#updating-your-project)してください。 + +## プルリクエストのテスト + +先述の手順に従えば、プルリクエストをテストすることができますが、インストールを実行する前に、テスト対象のプルリクエストを取得してブランチを切り替えるようにしてください。 下記手順の`[IDofThePR]`の部分は、github.comで表示されているプルリクエストのIDに置き換えてください: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +そして、先述のとおり[自身のプロジェクトを更新](#updating-your-project)してください。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx new file mode 100644 index 00000000000..b9cbf912aa2 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Mac App Storeガイド + +このページでは、WailsアプリをMac App Storeで公開する方法について、概要を説明します。 + +## 前提条件 + +- Apple Developerアカウントを所持しておく必要があります。 詳しくは、[Apple Developer Program](https://developer.apple.com/support/compare-memberships/)のサイトを確認してください。 +- 開発者ポータルで、証明書、ID、アプリを作成しておく必要があります。 詳しくは以下の手順に従ってください。 +- Xcodeコマンドラインツールを、あなたのローカルマシンにインストールしておく必要があります。 + +#### 証明書とIDの作成 + +1. [Apple Developerアカウント](https://developer.apple.com/account/)へアクセスします。 +2. `Certificates, Identifiers & Profiles`内にある`Identifiers`をクリックし、新しいApp IDを登録します。 (com.example.app) のような形式にしてください。 +3. 同じページで`Certificates`をクリックし、Mac App Storeディストリビューション用の新しい証明書を生成します。 それらの証明書をダウンロードし、ローカルマシンのKeychainにインポートしてください。 + +#### アプリ提出物の作成 + +1. [App Store Connectサイト](https://appstoreconnect.apple.com/apps)へアクセスします。 +2. 新しいアプリケーションを登録し、前のステップで作成済みのバンドルIDをリンクします。 +3. Appleの要求する正規のスクリーンショットや説明文などを、アプリに追加します。 +4. アプリの新しいバージョンを作成します。 + +#### プロビジョニングプロファイルの作成 +1. [Apple Developerプロフィール](https://developer.apple.com/account/resources/profiles/list)ページへアクセスします。 +2. Mac App Storeへの配布用に、新しいプロビジョニングプロファイルを追加します。 +3. プロファイルタイプをMacに設定し、先ほど作成したアプリケーションのApp IDを選択します。 +4. Mac App Distribution証明書を選択します。 +5. 埋め込みプロビジョニングプロファイルに名前をつけ、作成されたプロファイルをダウンロードします。 + +## Mac App Storeへの公開手順 + +#### AppleのApp Sandboxの有効化 + +Mac App Storeへ提出されるアプリは、Appleの[App Sandbox](https://developer.apple.com/app-sandboxing/)内で動作する必要があります。 そのためには、`entitlements.plist`ファイルを作成しておく必要があります。 ファイルは、`{PROJECT_DIR}/build/darwin/entitlements.plist`配下に作成することを推奨します。 + +**entitlementsファイルの例** + +ここでは、[RiftShare](https://github.com/achhabra2/riftshare)というアプリのentitlementsファイルを例として示します。 参考として、アプリが必要とする権限を入力してください。 詳しくは、[こちらのサイト](https://developer.apple.com/documentation/bundleresources/entitlements)をご覧ください。 チーム ID とアプリケーション名を上記で登録したものに置き換える必要があります。 + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +**組み込みプロビジョニング プロファイルの追加** 上記で作成したプロビジョニング プロファイルは、アプリケーションのルートに追加する必要があります。 embedded.provisionprofile という名前にする必要があります。 + +#### アプリパッケージのビルドと署名 + +以下は、Mac App Storeへの提出用に、アプリをビルドおよび署名するスクリプトの例です。 プロジェクトのルートディレクトリからスクリプトを実行することを前提としています。 + +アプリへ署名するための証明書と、インストーラへ署名するための証明書は異なりますので、ご注意ください。 両方の証明書がKeychainにインポートされていることを確認してください。 Keychainで文字列を探し、下記スクリプトに挿入してください。 具体的には、証明書名およびアプリ名を入力します。 以下のスクリプトを実行すると、アプリケーションのルートディレクトリに、署名済みの`app.pkg`ファイルが生成されます。 + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### アプリバンドルのアップロード + +審査向けに提出が可能となる前に、生成されたパッケージファイルをアップロードし、アプリケーションに関連付ける必要があります。 + +1. Mac App Storeから[Transporter App](https://apps.apple.com/us/app/transporter/id1450874784)をダウンロードします。 +2. アプリを起動し、Apple IDでサインインします。 +3. +マークをクリックし、前のステップで生成済みの`APP_NAME.pkg`ファイルを選択します。 そしてアップロードします。 +4. [App Store Connect](https://appstoreconnect.apple.com/apps)サイトへ戻り、アプリ提出物ページへ戻ります。 App Storeで公開する準備できているバージョンを選択します。 `Build`内で、Transporter経由でアップロードしたパッケージを選択します。 + +以上で終了です! サイトで、アプリを審査のために提出できるようになりました。 数営業日後、審査結果に問題が無ければ、アプリがMac App Storeに公開されます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx new file mode 100644 index 00000000000..884f9e8c3a0 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# 手動ビルド + +Wails CLIは、プロジェクトに関する様々な面倒な作業を担ってくれていますが、ときにはプロジェクトを手動でビルドできたほうが望ましい場合もあるでしょう。 このドキュメントでは、CLIが実行する多くの操作や、これらの操作を別の手段で実現するための方法について説明します。 + +## ビルドプロセス + +Wails CLIは、`wails build`コマンドまたは`wails dev`コマンドが使用されると、共通のビルドプロセスを実行します: + + - フロントエンド依存関係のインストール + - フロントエンドのビルド + - ビルドアセットの生成 + - アプリケーションのコンパイル + - [任意] アプリケーションの圧縮 + +### フロントエンド依存関係のインストール + +#### CLIが実行する手順 + +- `-s`フラグが指定された場合、この手順をスキップします。 +- `wails.json`ファイル内で、`frontend:install`キーにインストールコマンドが記述されているかを確認します。 +- 記述されていない場合、この手順をスキップします。 +- 記述されている場合、フロントエンドディレクトリ内に`package.json`ファイルが存在するか確認します。 存在しない場合、この手順をスキップします。 +- `package.json`ファイルの内容をもとに、MD5チェックサムを生成します。 +- `package.json.md5`ファイルが存在するか確認し、存在する場合は、さきほど生成したMD5チェックサムと比較して、内容が変更されていないかどうかを確認します。 内容が同じ場合、この手順をスキップします。 +- `package.json.md5`ファイルが存在しない場合、ファイルを作成し、さきほど生成したMD5チェックサムを書き込みます。 +- この時点でビルドが必要と判断された場合、`node_modules`ディレクトリが存在しない場合、または`-f`フラグが指定された場合は、フロントエンドディレクトリ内でインストールコマンドを実行します。 + +#### 手動で実行する手順 + +コマンドライン、または`npm install`のスクリプトを用いて、この手順を実行してください。 + +### フロントエンドのビルド + +#### Wails CLIでの手順 + +- `-s`フラグが指定された場合、この手順をスキップします。 +- `wails.json`ファイル内で、`frontend:build`キーにビルドコマンドが記述されているかを確認します。 +- 記述されていない場合、この手順をスキップします。 +- 記述されている場合、フロントエンドディレクトリ内でコマンドが実行されます。 + +#### 手動で実行する手順 + +コマンドライン、`npm run build`のスクリプト、またはフロントエンドのビルドスクリプトを用いて、この手順を実行してください。 + +### アセットの生成 + +#### Wails CLIでの手順 + +- `-nopackage`フラグが指定された場合、この手順をスキップします。 +- `build/appicon.png`ファイルが存在しない場合、デフォルトのファイルを作成します。 +- Windowsの場合、[Windowsバンドル](#windows)の節を参照してください。 +- `build/windows/icon.ico`ファイルが存在しない場合、`build/appicon.png`画像ファイルから新規作成します。 + +##### Windows + +- `build/windows/icon.ico`ファイルが存在しない場合、`build/appicon.png`ファイルをもとに、256、128、64、48、32、16サイズのアイコンを新規作成します。 この処理は[winicon](https://github.com/leaanthony/winicon)によって実現しています。 +- `build/windows/.manifest`ファイルが存在しない場合、デフォルトバージョンから新規作成します。 +- アプリケーションを本番ビルドとしてコンパイルします(上記のとおり)。 +- [winres](https://github.com/tc-hib/winres)を使用して、アイコンとマニフェストをリンクできる`.syso`ファイルにバンドルします。 + +#### 手動で実行する手順 + +- [winicon](https://github.com/leaanthony/winicon)のCLIツール(または他の任意ツール)を使用して、`icon.ico`を作成します。 +- アプリケーションの`.manifest`ファイルを作成または更新します。 +- [winres CLI](https://github.com/tc-hib/go-winres)を使用して、`.syso`ファイルを生成します。 + +### アプリケーションのコンパイル + +#### Wails CLIでの手順 + +- `-clean`フラグが指定された場合、`build`ディレクトリを削除して再作成します。 +- `wails dev`コマンドの場合、デフォルトで次のGoフラグを使用するようにします: `-tags dev -gcflags "all=-N -l"` +- `wails build`コマンドの場合、デフォルトで次のGoフラグを使用するようにします: `-tags desktop,production -ldflags "-w -s"` + - Windowsの場合: `-ldflags "-w -h -H windowsgui"` +- `-tags`フラグを使用してCLIに指定された追加タグを、デフォルトのタグに追記します。 +- `-ldflags`フラグを使用してCLIに指定された追加のldflagsを、デフォルトのタグに追記します。 +- `-o`フラグのパラメータをパススルーさせます。 +- `-compiler`フラグで指定されたGoコンパイラを、コンパイル時に使用するようにします。 + +#### 手動で実行する手順 + +- 開発ビルドの場合、最小のコマンドは次のとおりです: `go build -tags dev -gcflags "all=-N -l"` +- 本番ビルドの場合、最小のコマンドは次のとおりです: `go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- `.syso`ファイルと同じディレクトリでコンパイルするようにしてください。 + +### アプリケーションの圧縮 + +#### Wails CLIでの手順 + +- `-upx`フラグが指定された場合、アプリケーションを圧縮するために、`upx`プログラムをデフォルト設定で実行します。 +- `-upxflags`フラグも指定された場合、デフォルト設定の代わりにこれらのフラグを使用します。 + +#### 手動で実行する手順 + +- `upx [flags]`を手動で実行して、アプリケーションを圧縮します。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx new file mode 100644 index 00000000000..68319030e81 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx @@ -0,0 +1,191 @@ +# v1からの移行 + +## 概要 + +Wails v2は、v1から大幅に変更されています。 このドキュメントでは、変更点、および既存のプロジェクトを移行する方法に焦点をあてて説明します。 + +### アプリケーションの作成 + +v1では、`wails.CreateApp`でメインアプリケーションを作成して、`app.Bind`でバインディングを追加し、`app.Run()`でアプリケーションを起動していました。 + +例: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +v2では、`wails.Run()`メソッドだけが、[アプリケーションオプション](../reference/options.mdx#application-options)を指定することができます。 + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### バインディング + +v1では、任意の関数と構造体の両方をバインドすることが可能でした。 v2では、構造体のみバインドできるようになり、シンプルになりました。 v1において`Bind()`メソッドに渡していた構造体のインスタンスは、[アプリケーションオプション](../reference/options.mdx#application-options)の`Bind`フィールドで指定できます。 + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +v1では、バインドされたメソッドをフロントエンドから利用するには`window.backend`を使用していました。 これは`window.go`に変更されました。 + +### アプリケーションライフサイクル + +v1では、バインドされた構造体の中に、`WailsInit()`および`WailsShutdown()`という2つの特別なメソッドが存在しました。 これらは、[アプリケーションオプション](../reference/options.mdx#application-options)の一部として、3つのライフサイクルフックに置き換えられています: + +- [OnStartup](../reference/options.mdx#onstartup) +- [OnShutdown](../reference/options.mdx#onshutdown) +- [OnDomReady](../reference/options.mdx#ondomready) + +注意: [OnDomReady](../reference/options.mdx#ondomready)は、v1での`wails:ready`システムイベントに代わるものです。 + +これらのメソッドは標準的な関数を指定できますが、一般的には、それらのメソッドを構造体の一部として保持しておきます: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### ランタイム + +v2のランタイムはv1よりも非常に充実しており、メニュー、ウィンドウ操作、優れたダイアログをサポートしています。 メソッドのシグネチャが若干変更されていますので、詳しくは[ランタイムのリファレンス](../reference/runtime/intro.mdx)を参照してください。 + +v1では、`WailsInit()`に渡された構造体を介して[ランタイム](../reference/runtime/intro.mdx)を利用できていました。 v2では、ランタイムは独自のパッケージとして移植されています。 ランタイムの各メソッドは、[OnStartup](../reference/options.mdx#onstartup)メソッドで受け渡される`context.Context`を引数に取ります。 + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startupはアプリケーション起動時に呼び出される +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} + +``` + +### アセット + +v2での_最も大きな_変更は、アセットの取り扱い方です。 + +v1では、2つのアプリケーションオプションを介してアセットを指定していました: + +- `JS` - アプリケーションのJavaScript +- `CSS` - アプリケーションのCSS + +これは、単一のJSファイルおよびCSSファイルを、開発者が責任をもって作成する必要があることを意味していました。 基本的には、webpackのような複雑なパッカーを使用する必要がありました。 + +v2では、Wailsは一般的なWebサーバと同じように、フロントエンドアセットは何でも構いません。 すべてのアプリケーションアセットは、`embed.FS`インスタンスとしてアプリケーションオプションに渡されます。 + +**つまりアセットをバンドルしたり、画像をBase64でエンコードしたり、カスタムフォントを使用するためにバンドラーの設定を工夫したりする必要は一切ありません**。 + +アプリケーション起動時に、Wailsは、あらかじめ`embed.FS`で指定されたディレクトリ内をスキャンして`index.html`を探し、Webサーバの挙動のように、その場所をすべてのアプリケーションアセットのルートパスとみなします。 + +例: とあるアプリケーションのプロジェクトディレクトリの構成が次のようになっているとします。 この場合、最終的に必要なすべてのアセットは`frontend/dist`ディレクトリ配下に配置される必要があります: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +これらのアセットは、`embed.FS`を定義するだけで、アプリケーションから使用することができます: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +もちろん、必要に応じてバンドラを使用することもできます。 唯一行わなければいけないことは、[アプリケーションオプション](../reference/options.mdx#application-options)の`Assets`キーに`embed.FS`インスタンスを指定して、Wailsにアプリケーションアセットディレクトリを教えてあげることです。 + +### プロジェクト構成 + +v1では、プロジェクトルートに存在する`project.json`ファイルで、プロジェクト構成を管理していました。 v2では、プロジェクトルートにある`wails.json`ファイルでプロジェクト構成を管理します。 + +フォーマットは若干変更されています。 比較表は次のとおりです: + +

+ +| v1 | v2 | 備考 | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | 消去されました | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | 変更されました | +| frontend / dir | | 消去されました | +| frontend / install | frontend:install | 変更されました | +| frontend / build | frontend:build | 変更されました | +| frontend / bridge | | 消去されました | +| frontend / serve | | 消去されました | +| tags | | 消去されました | +| | wailsjsdir | wailsjsモジュールを生成するディレクトリ | +| | assetdir | `dev`モードでコンパイルされたフロントエンドアセットのディレクトリ。 通常は自動推定されるため、空のままで構いません。 | +| | reloaddirs | `dev`モードでリロードをトリガーさせたい追加のディレクトリを指定するための、カンマ区切りのリスト。 高度なアセット構成をとる場合にのみ必要です。 | + +

diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..58ee21c687a --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# マウスボタン + +Wailsランタイムでは、マウスクリックを途中で捕捉し、フレームレスウィンドウのサイズ変更が必要かどうかや、ウィンドウの移動が必要かどうかを判断しています。 `window.onclick`はマウスボタンを正しく報告しないため、どのようにマウスクリックが発生したことを検出しているのかを尋ねられました。 以下のコードは、マウスクリックを検出する方法を示しています: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +参考: https://developer.mozilla.org/ja/docs/Web/API/MouseEvent/button diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx new file mode 100644 index 00000000000..9b5bab1fe7f --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# 難読化されたビルド + +Wailsでは、[garble](https://github.com/burrowers/garble)を使用したアプリケーションの難読化をサポートしています。 + +`wails build`コマンドで`-obfuscate`フラグを使用することで、ビルド時に難読化を有効にすることができます: + +```bash +wails build -obfuscated +``` + +また、難読化に関する設定をカスタマイズするために、`-garbleargs`フラグを使用できます: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +これらの設定は、[プロジェクト構成](../reference/project-config)で記述されている場合もあります。 + +## 動作の仕組み + +標準のビルドでは、バインドされたすべてのメソッドは、フロントエンドの`window.go`変数から利用することができます。 これらのメソッドが呼び出されると、対応するバックエンドメソッドが完全修飾関数名で呼び出されます。 難読化ビルドの場合、メソッドは名前の代わりにIDでバインドされます。 `wailsjs`ディレクトリに生成されたバインディングは、このIDを使用してバックエンドの関数を呼び出します。 + +:::note + +アプリケーションを難読化した状態で動作させるためには、`wailsjs`ディレクトリの配下に生成されたバインディングを使用する必要があります。 + +::: + +## 例 + +Importing the "Greet" method from the bindings like this: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +これは、バインディングがIDで再生成され、呼び出しメカニズムが更新されるため、難読化モードでメソッドが正しく動作することを保証します。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx new file mode 100644 index 00000000000..4ee1db0e4ff --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# オーバースクロール + +[オーバースクロール](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior)は、ページコンテンツの境界を越えてスクロールした際に発生する"バウンス効果"です。 この効果はモバイルアプリでよく見かけます。 CSSを使用することでこの効果を無効化できます: + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx new file mode 100644 index 00000000000..ba9195d2bb2 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx @@ -0,0 +1,47 @@ +# ルーティング + +アプリケーションでビューを切り替える一般的な方法は、ルーティングです。 このページでは、ルーティングを実現する方法をいくつか紹介します。 + +## Vue + +Vueにおいてルーティングで推奨されるアプローチは、[ハッシュモード](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode)となります: + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +Angularにおいてルーティングで推奨されるアプローチは、[HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies#_hashlocationstrategy)となります: + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +Reactにおいてルーティングで推奨されるアプローチは、[HashRouter](https://reactrouter.com/en/main/router-components/hash-router)となります: + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx new file mode 100644 index 00000000000..8b57d7e0fe3 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx @@ -0,0 +1,387 @@ +# コード署名 + +このガイドでは、MacOSおよびWindowsでWailsによって生成されたバイナリに署名するための方法を解説します。 このガイドは、CI環境、とりわけGitHub Actionsを対象としています。 + +## Windows + +まず初めに、コード署名証明書が必要となります。 まだ証明書を持っていない場合は、[こちら](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate)のMicrosoftのページに、いくつかのプロバイダーが紹介されていますので入手してください、 デバイスドライバといったカーネルレベルのソフトウェアを作成する必要がない限り、EV証明書を入手する必要はありません。 Wailsアプリに署名する場合、標準のコード証明書で問題ありません。 + +自動ビルドシステムを対象に作業する前に、ローカルマシン上でバイナリに署名する方法を証明書プロバイダに確認し、特別な要件がないかを確認することを推奨します。 例えば、[こちら](https://www.ssl.com/how-to/using-your-code-signing-certificate/)はSSL.comのWindows向けコード署名ガイドです。 ローカル環境での署名方法が分かっていれば、CI環境で発生するかもしれない問題のトラブルシューティングが容易になることでしょう。 例えば、SSL.comのコード署名証明書は[SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool)において`/tr`フラグを指定する必要がありますが、他の証明書プロバイダでは、タイムスタンプサーバの提供のために、`/t`フラグのみ必要である場合もあります。 [こちら](https://github.com/Dana-Prajea/code-sign-action)のような、Windowsバイナリの署名によく使われるGitHub Actionsのアクションは、SignTool.exeの`/tr`フラグに対応していません。 したがって、このガイドではPowerShellコマンドを使用して手動でアプリケーションに署名する方法に焦点をあてます。ただし、必要であれば[code-sign-action](https://github.com/Dana-Prajea/code-sign-action)アクションといったアクションを使用することも可能です。 + +まずは、GitHub CIでWailsアプリをビルドできるようにしておきましょう。 簡単なワークフローテンプレートは次のとおりです: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # wails.jsonでフロントエンドのビルドおよびインストールコマンドを設定していない場合、ここで手動でフロントエンドをビルドする必要があるかもしれません。 + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +次に、GitHubワークフローに署名証明書へのアクセス権を付与する必要があります。 これは、.pfxまたは.p12形式の証明書をbase64文字列にエンコードすることで実現できます。 PowerShellの場合、証明書名が仮に'my-cert.p12'だとすると、次のコマンドが使用できます: + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +これで、base64でエンコードされた証明書が含まれる.txtファイルが作成されます。 このファイルの中身は、_-----BEGIN CERTIFICATE-----_で始まり、_-----END CERTIFICATE-----_で終わる必要があります。 次に、GitHubで2つのActions secretsを作成します。 _Settings -> Secrets -> Actions_のページに移動し、2つのシークレットを作成してください: + +- **WIN_SIGNING_CERT** には、base64エンコードされた証明書テキストの内容を設定します。 +- **WIN_SIGNING_CERT_PASSWORD** には、証明書のパスワードを設定します。 + +ここまでの作業で、次の2つのいずれかの方法を使用して、ワークフローによる署名処理を実装する準備が整いました。 + +### 方法1: コマンドを使用して署名 + +この方法では、Power Shellコマンドを使用してアプリに署名し、署名プロセス全体をコントロールします。 + +`"Build Wails app"`ステップのあとに、下記のステップをワークフローに追加してください。 + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +このスクリプトでは、証明書ファイル用に新しいディレクトリを作成して、base64シークレットから証明書ファイルを作成します。その証明書を.pfxファイルに変換し、最後にバイナリへ署名します。 下記の変数は、最後の行内で置換してください: + +- **signing algorithm**: 通常は、sha256。 +- **timestamping server**: 証明書で使用するタイムスタンプサーバのURL。 +- **path to binary**: 署名したいバイナリへのパス。 + +Wailsの構成において`outputfilename`に"app.exe"という値が設定されており、SSL.comでの証明書が用意されている場合、次のようなワークフローとなります: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # wails.jsonでフロントエンドのビルドおよびインストールコマンドを設定していない場合、ここで手動でフロントエンドをビルドする必要があるかもしれません。 + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### 方法2: アクションを使用して自動的に署名 + +[こちら](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate)のようなWindowsコード署名のためのアクションを使用することはできますが、証明書用のSHA1ハッシュや証明書名が必要であることに注意してください。 詳しくは、アクションの[マーケットプレイス](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate)で、使用方法を確認してください。 + +--- + +## MacOS + +まずは、Appleのコード署名証明書が必要です。 もし所有していない場合は、Googleで検索すれば取得方法が分るでしょう。 証明書を入手できたら、それをエクスポートし、base64でエンコードする必要があります。 [このチュートリアル](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions)をご覧いただければ、簡単にエンコードする方法が分ります。 .p12証明書ファイルをエクスポートした場合、次のコマンドを使用して、チュートリアルで示されているようにbase64にエンコードできます: + +```bash +base64 Certificates.p12 | pbcopy +``` + +これにより、Windowsと同様に、GitHubプロジェクトにいくつかのシークレットを設定する準備が整いました: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64**には、新しくコピーしたbase64証明書の内容を設定します。 +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD**には、証明書のパスワードを設定します。 +- **APPLE_PASSWORD**には、[こちら](https://appleid.apple.com/account/manage)で生成できるアプリ固有のパスワードを設定します。 + +GitHub Actionのワークフローで、Wailsアプリをビルドできるようにしましょう。 簡単な例は次のとおりです: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # wails.jsonでフロントエンドのビルドおよびインストールコマンドを設定していない場合、ここで手動でフロントエンドをビルドする必要があるかもしれません。 + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +macOSでコード署名をするときは、[gon](https://github.com/mitchellh/gon)という非常に便利なツールを使うことで、コード署名およびAppleサーバとの通信ができます。これはGoで記述されており、このガイドでも使用されています。 + +`Build Wails app`ステップの後に、次のワークフローを追加します: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +ここで、`build/darwin`ディレクトリ内に、gonの構成設定ファイルを用意する必要があります: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +`source`にはWailsのバイナリファイルの場所、`bundle_id`にはバンドルID、`apple_id`にはあらかじめ作成したApple IDのユーザ名とアプリ固有のパスワード、`sign.application_identity`には次のコマンドを実行することで分かるIDを指定します: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +このファイルでは、アプリに必要な権限(たとえば、アプリがカメラを使用する場合、カメラへのアクセス権限)を設定します。 権限について詳しくは[こちら](https://developer.apple.com/documentation/bundleresources/entitlements)をご覧ください。 + +`Info.plist`ファイルのバンドルIDを、`gon-sign.json`に入力したものと同じIDに書き換えてください。 次は、`Info.plist`ファイルの例です: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +これで、Wailsアプリをビルドしたあとに、ワークフローに署名のステップを追加する準備ができました: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +Appleでバイナリに署名する場合、数分から数時間程度の所要時間がかかることがありますのでご注意ください。 + +## ワークフローファイルの組み合わせ + +WindowsとmacOSを組み合わせたGitHubワークフローファイルの例は次のとおりです: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # wails.jsonでフロントエンドのビルドおよびインストールコマンドを設定していない場合、ここで手動でフロントエンドをビルドする必要があるかもしれません。 + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# おわりに + +このガイドは、RiftShareプロジェクトとそのワークフローに触発されて作成されたものであり、 [こちら](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml)を併せて確認することを強く推奨します。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..4651c422ed1 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +This guide will go into: + +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. + +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. + +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. + +- `npm create svelte@latest frontend` + +##### Modify wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. + +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Install/remove dependencies using your favorite package manager. + +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js + +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. + +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. + +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. + +- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! + +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html + +``` + +... + + + +... + +``` + +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. + +- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling + +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions + +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx new file mode 100644 index 00000000000..8e1f8304557 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx @@ -0,0 +1,97 @@ +# テンプレート + +Wailsは、事前に作成されたテンプレートからプロジェクトを生成します。 v1では、テンプレートがメンテナンス困難なプロジェクトとなり、時代遅れとなってしまう可能性がありました。 v2では、コミュニティを強化するために、テンプレートにいくつかの新機能が追加されました: + +- [リモート点テンプレート](../reference/cli.mdx#remote-templates)からプロジェクトを生成する機能 +- 独自のテンプレートの作成に役立つツール + +## テンプレートの作成 + +テンプレートを作成するには、`wails generate template`コマンドを使用します。 デフォルトのテンプレートを生成する場合は、次のコマンドを実行してください: + +`wails generate template -name mytemplate` + +これにより、デフォルトのファイルが含まれた"mytemplate"ディレクトリが作成されます: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### テンプレートの概要 + +デフォルトのテンプレートは、次のファイルおよびディレクトリで構成されています: + +| ファイル名 / ディレクトリ名 | 説明 | +| --------------- | ----------------------- | +| NEXTSTEPS.md | テンプレートを完成させる手順を記した説明 | +| README.md | テンプレートとともに公開されるREADME | +| app.tmpl.go | `app.go`のテンプレートファイル | +| frontend/ | フロントエンドアセットを含むディレクトリ | +| go.mod.tmpl | `go.mod`のテンプレートファイル | +| main.tmpl.go | `main.go`のテンプレートファイル | +| template.json | テンプレートのメタデータ | +| wails.tmpl.json | `wails.json`のテンプレートファイル | + +このあとは、`NEXTSTEPS.md`に記述されている手順に従うことを推奨します。 + +## 既存プロジェクトからのテンプレート作成 + +テンプレートの生成時にプロジェクトへのパスを渡すことで、既存のフロントエンドプロジェクトから、テンプレートを作成することができます。 ここでは、Vue3テンプレートの作成方法を説明します: + +- Vue Cliをインストールする: `npm install -g @vue/cli` +- デフォルトのプロジェクトを作成する: `vue create vue3-base` + - その際、`Default (Vue 3) ([Vue 3] babel, eslint)`を選択します +- プロジェクトが作成されたあとに、次のコマンドを実行する: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- `NEXTSTEPS.md`ファイルに記述されているように、テンプレートをカスタマイズする +- ファイルの準備ができたら、次のコマンドを実行してテストする: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- 新しいプロジェクトの動作をテストするために、次のコマンドを実行する: `cd my-vue3-project`および`wails build` +- プロジェクトがコンパイルされたら、実行する: `.\build\bin\my-vue3-project.exe` +- Vue3アプリケーションが完全に動作していることを確認する + +```mdx-code-block +
+ +
+``` + +## テンプレートの公開 + +テンプレートを公開するためにすることは、ファイルをGitHubにプッシュすることだけです。 以下のベストプラクティスを実施することを推奨します: + +- 不要なファイルやディレクトリ(`.git`など)をフロントエンドディレクトリから削除する +- `template.json`がきちんと記述されていること、とくに`helpurl`が記述されていることを確認する +- ファイルをGitHubにプッシュする +- [コミュニティのテンプレート](../community/templates.mdx)ページにプルリクエストを作成する +- [Template Announcement](https://github.com/wailsapp/wails/discussions/825)のディスカッションボードでテンプレートをアナウンスする diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..eb93618657c --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# トラブルシューティング + +このページは、トラブルシューティングを手助けするヒント集です。 + +## `wails`コマンドが見つからないのですが? + +`wails`コマンドが見つからないとシステムに怒られた場合は、Goが、公式のGoインストール手順に従って導入されているかを確認してください。 コマンドが見つからないほとんどの理由は、あなたのホームディレクトリ配下にある`go/bin`ディレクトリのパスが、`PATH`環境変数に含まれていないからです。 また、インストールによって行われた環境変更を反映させるために、もともと開いていたコマンドプロンプト(ターミナル)がある場合はそれらをいったん閉じて、再度開きなおしてください。 + +## アプリケーションがホワイト/ブラックスクリーン表示になってしまいます + +アプリケーション内で、正しいディレクトリでアセットが指定されていることを確認してください。 `main.go`ファイル内には、以下のようなコードが存在します: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +また、`frontend/dist`ディレクトリ内に、アプリケーションアセットがきちんと含まれているか確認してください。 + +### Mac + +この事象がMacで発生した場合は、`Info.plist`に以下の記述を追加してみてください: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +参考: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Macのアプリケーションが無効になっています + +ビルドしたアプリケーションが次のように表示される場合: + +```mdx-code-block +

+ +

+``` + +アプリケーションの`info.plist`が無効である可能性があります。 `build/.app/Contents/info.plist`ファイルを更新し、データが有効かどうかを確認します。たとえば、バイナリ名が正しいかどうかを確認してください。 変更を反映するには、ファイルを`build/darwin`ディレクトリにコピーします。 + +## Windowsのエクスプローラにアプリケーションアイコンが正しく表示されません + +アプリケーションアイコンが正しく表示されない場合は、`C:\Users\<あなたのユーザ名>\AppData\Local`ディレクトリ内にある、`IconCache.db`という隠しファイルを削除してみてください。 これにより、Windowsのアイコンキャッシュが強制的に再作成されます。 + +出典: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## 可変長引数を持つバックエンドメソッドをフロントエンドから呼び出せません + +次のように、可変長引数を持つバックエンドメソッドが定義されている場合: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +このメソッドを、次のようにフロントエンドから呼び出すと失敗します: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +回避方法: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +出典: https://github.com/wailsapp/wails/issues/1186 + +## Wailsのインストール時にプロキシエラーが発生します + +次のようなエラーが発生する場合: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +おそらく、公式のGo Proxyがブロックされています (中国のユーザより報告されています)。 解決するには、次のように、プロキシを手動で設定します: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +出典: https://github.com/wailsapp/wails/issues/1233 + +## 生成されたTypeScriptの型定義が正しくありません + +場合によっては、生成されるTypeScriptの型定義が誤っていることがあります。 この事象を軽減させるために、`ts_type`という構造体のタグを使用して、生成する型を指定することができます。 詳しくは、[こちら](https://github.com/tkrajina/typescriptify-golang-structs#custom-types)をご覧ください。 + +## `index.html`から移動するとフロントエンド上でメソッドを呼び出すことができません + +`index.html`から新しいhtmlファイルへ遷移すると、コンテキスト情報は失われてしまいます。 この事象を修正するには、遷移先の新しいページの``セクションに、次のインポートコードを追加します: + +```html + + + + +``` + +出典: https://github.com/wailsapp/wails/discussions/1512 + +## Macで`wails dev`を実行すると`too many open files`エラーが発生します + +デフォルトでは、macOSは最大で256個のファイルしか開くことができません。 この制限により、`wails dev`コマンド実行時に影響が発生する場合があります。 この制限は、ターミナルで次のようなコマンドを実行することで、緩和することができます: `ulimit -n 1024`。 + +FSNotifyは、Macのために、[Appleのfseventsへの移行を検討しています](https://github.com/fsnotify/fsnotify/issues/11)。 この移行が完了しない間は、[こちら](https://github.com/wailsapp/wails/issues/1733)で言及されているように、独自で実装を行います。 + +## Macアプリで不可解なコンパイルエラーが発生します + +一部のユーザより、次のようなコンパイルエラーが発生することがあるという報告を受けています: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +これは_一般的に_、実行されているOSのバージョンと、インストールされているXCodeコマンドラインツールのバージョンが不一致であることが原因です。 このようなエラーが発生した場合は、XCodeコマンドラインツールを最新バージョンに更新してみてください。 + +XCodeコマンドラインツールの再インストールが引き続き失敗する場合は、次のコマンドを使用して、ツールキットがどこに配置されているかを確認できます: + +`xcode-select -p` + +`/Applications/Xcode.app/Contents/Developer`と表示された場合、`sudo xcode-select --switch /Library/Developer/CommandLineTools`というコマンドを実行してください。 + +出典: https://github.com/wailsapp/wails/issues/1806 および https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +次のようなエラーが発生する場合: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" + +`frontend/node_modules`と`frontend/package-lock.json`を`.gitignore`に追加することをお勧めします。 そうしないと、異なるバージョンのNodeがインストールされている別のマシンでリポジトリを開いた際に、アプリケーションが実行できなくなる場合があります。 + +この事象が発生した場合は、単純に`frontend/node_modules`と`frontend/package-lock.json`を削除し、`wails build`コマンドおよび`wails dev`コマンドを再実行してください。 + +## ビルドプロセスが"Generating bindings"で停止します + +バインディングの生成プロセスは、アプリケーションを特別なモードで実行します。 アプリケーションに、意図に有無にかかわらず無限ループ(`wails.Run()`のあとに終了されないコード)が含まれている場合、バインディングの生成の段階でビルドプロセスが停止する可能性があります。 コードが正しく終了していることを確認してください。 + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx new file mode 100644 index 00000000000..86b7c16a487 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +このページでは、Wailsでの開発でVisual Studio Codeを使用する際のさまざまなヒントやコツをご紹介します。 + +## Veturの構成設定 + +このヒントを提供してくれた[@Lyimmi](https://github.com/Lyimmi)に感謝します。 このヒントは[こちら](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349)に投稿されたものです。 + +Veturは、Vueプロジェクトのシンタックスハイライトおよびコード補完を提供してくれる、Visual Studio Codeでのポピュラーな拡張機能です。 VSCodeでWailsのプロジェクトを読み込むと、Veturはルートディレクトリでフロントエンドのプロジェクトを見つけようとするため、エラーが発生します。 これを解消するためには、次の手順を実施します: + +プロジェクトのルートディレクトリに、`vetur.config.js`という名前のファイルを作成します。 + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +次に、`frontend/tsconfig.json`の構成を修正します: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +以上により、Veturを期待どおりに使用できるようになります。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx new file mode 100644 index 00000000000..3e3ae9b5e70 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# NSISインストーラ + +```mdx-code-block +

+ +
+

+``` + +Wailsでは、[NSISインストーラ](https://nsis.sourceforge.io/)を使用したWindowsインストーラの生成をサポートしています。 + +## NSISのインストール + +### Windows + +インストーラは、[NSISダウンロード](https://nsis.sourceforge.io/Download)ページから入手できます。 + +Chocolateyパッケージマネージャを使用している場合は、次のスクリプトを実行することでインストールできます: + +``` +choco install nsis +``` + +NSISを手動でインストールする場合、NSISインストール時に作成される`makensis.exe`ファイルが含まれた_Bin_フォルダを、パスに追加する必要があります。 Windows上でパスを追加する方法については、[こちら](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/)の優れたチュートリアルをご覧ください。 + +### Linux + +`nsis`パッケージは、ディストリビューションのパッケージマネージャから入手できます。 + +### MacOS + +NSISは、homebrew経由でインストールできます: `brew install nsis`。 + +## インストーラの生成 + +新しくプロジェクトが作成されると、Wailsは、`build/windows/installer`内に、NSIS構成ファイルを生成します。 構成データは`installer/info.json`から読み込まれますが、当該データはプロジェクトの`wails.json`のInfo§の情報を使用するように設定されています: + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +アプリケーションのインストーラを生成するには、`wails build`コマンド実行時に、`-nsis`フラグを使用します: + +``` +wails build -nsis +``` + +これにより、`build/bin`ディレクトリにインストーラが生成されます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx new file mode 100644 index 00000000000..0b47b152967 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +このページでは、Windows向けのWailsアプリケーション開発に関する、様々なガイドを掲載しています。 + +## WebView2ランタイム依存関係のハンドリング + +Windows用にビルドされたWailsアプリケーションは、Microsoft [WebView2ランタイム](https://developer.microsoft.com/en-us/microsoft-edge/webview2/)がランタイム要件となっています。 Windows 11ではデフォルトでこのランタイムがインストールされていますが、一部のマシンではインストールされていません。 Wailsでは、この依存関係に対処するための簡単なアプローチを提供しています。 + +ビルド時に`-webview2`フラグを使用することで、アプリ起動時に適切なランタイムが検出されない場合(インストールされているランタイムが古すぎる場合を含む) に、アプリケーションがどのように動作するかを指定できます。 次の4つのオプションがあります: + +1. Download +2. Embed +3. Browser +4. Error + +### Download + +このオプションでは、適切なランタイムが見つからない旨をユーザに知らせ、MicrosoftのWebView2のサイトからダウンロードされる公式のブートストラッパを実行する提案をします。 ユーザが続行を選択した場合、公式のブートストラッパがダウンロードおよび実行されます。 + +### Embed + +このオプションでは、アプリケーション内に公式のブートストラッパの埋め込みます。 適切なランタイムが見つからない場合、アプリケーションはブートストラッパの実行を提案します。 これにより、バイナリサイズが150k程度増えます。 + +### Browser + +このオプションでは、適切なランタイムが見つからない旨をユーザに知らせ、ブートストラッパをダウンロードしてインストールすることのできるWebView2の公式ページをブラウザで開く提案をします。 続行した場合、アプリケーションは終了し、インストールをユーザに委ねます。 + +### Error + +このオプションでは、適切なランタイムが見つからない場合、ユーザにエラーが表示され、それ以上何の処理も実行しません。 + +## フィックスドバージョンランタイム + +WebView2の依存関係に対処するための別手段として、自身でランタイムを運ぶという方法が挙げられます。 [フィックスドバージョンランタイム](https://developer.microsoft.com/microsoft-edge/webview2/#download-section)をダウンロードしてバンドルするか、アプリケーション内でランタイムをダウンロードするようにします。 + +また、Wailsの起動時に`windows.Options`において、フィックスドバージョンWebView2ランタイムのパスを指定する必要があります。 + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +注意: `WebviewBrowserPath`が指定されている場合、必要な最低バージョンを満たしていなかったり、ランタイムへのパスが無効であったりすると、強制的に`error`オプションの挙動となります。 + +## 他のプログラムの起動 + +スクリプトなどの他のプログラムを起動すると、当該プログラムのウィンドウが画面に表示されます。 このウィンドウを表示させたくない場合は、次のコードを使用してください: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +この解決策は、[ディスカッションボード](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172)内で[sithembiso](https://github.com/sithembiso)により示されました。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx new file mode 100644 index 00000000000..00fe1d68b73 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# どうやって動いているの? + +Wailsは、webkitフロントエンドを備えた、何の変哲もないGoアプリです。 アプリ全体のうちGoの部分は、アプリのコードと、ウィンドウ制御などの便利な機能を提供するランタイムライブラリで構成されています。 フロントエンドはwebkitウィンドウであり、フロンドエンドアセットをウィンドウ上に表示します。 フロントエンドからも、JavaScriptでランタイムライブラリを呼び出すことができます。 そして最終的に、Goのメソッドはフロントエンドにバインドされ、ローカルのJavaScriptメソッドであるかのように、フロントエンドから呼び出すことができます。 + +```mdx-code-block +
+ +
+``` + +## アプリのメインコード + +### 概要 + +アプリは、`wails.Run()`メソッドを1回呼び出すことで、構成することができます。 このメソッドで、アプリのウィンドウサイズやウィンドウタイトル、使用アセットなどを指定することができます。 基本的なアプリを作るコードは次のとおりです: + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### オプション + +上記のコードでは、次のオプションが指定されています: + +- `Title` - ウィンドウのタイトルバーに表示されるテキスト +- `Width` & `Height` - ウィンドウの大きさ +- `Assets` - アプリのフロントエンドアセット +- `OnStartup` - ウィンドウが作成され、フロントエンドの読み込みを開始しようとする時のコールバック +- `OnShutdown` - アプリを終了しようとするときのコールバック +- `Bind` - フロントエンドにバインドさせたい構造体インスタンスのスライス + +設定可能なすべてのオプションについては、[オプションのリファレンス](reference/options)をご覧ください。 + +#### アセット + +Wailsでフロントエンドアセット無しのアプリを作成することはできないため、`Assets`オプションは必須オプションです。 アセットには、一般的なWebアプリケーションでよく見かけるような、html、js、css、svg、pngなどのファイルを含めることができます。**アセットバンドルを生成する必要は一切なく**、そのままのファイルを使用できます。 アプリが起動すると、アセット内の`index.html`が読み込まれます。この時点で、フロントエンドはブラウザとして動作するようになります。 `embed.FS`を使ってアセットファイルが格納されたディレクトリを指定しますが、ディレクトリの場所はどこでも構いません。 embedで指定するパスは、`frontend/dist`のように、アプリのメインコードから相対的に見たディレクトリパスになります: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +起動時に、Wailsは`index.html`が含まれるディレクトリを再帰的に探します。 他のすべてのアセットは、当該ディレクトリから相対的に読み込まれます。 + +本番用のバイナリファイルには、`embed.FS`で指定されたアセットファイルが含まれるため、アプリ配布時に、バイナリファイルとは別にアセットファイルを付加させる必要はありません。 + +`wails dev`コマンドを使って開発モードでアプリを起動した場合、アセットはディスクから読み込まれ、アセットファイルが更新されると、自動的にアプリがライブリロードされます。 アセットの場所は、`embed.FS`での指定値から推定されます。 + +詳細は、[アプリ開発ガイド](guides/application-development.mdx)をご覧ください。 + +#### アプリのライフサイクル + +フロントエンドが`index.html`を読み込もうとする直前に、[OnStartup](reference/options.mdx#onstartup)で指定されたメソッドがコールバックされます。 Goの標準的なcontextがこのメソッドに渡されます。 このメソッドに引数で渡されるContextは、今後、Wailsのラインタイムを呼び出すときに必要になるため、通常は、このContextへの参照を保持しておいてください。 同様に、アプリがシャットダウンされる直前には、[OnShutdown](reference/options.mdx#onshutdown)で指定されたコールバックが呼び出され、Contextも渡されます。 `index.html`に含まれるすべてのアセットが読み込み終わったときに呼び出される[OnDomReady](reference/options.mdx#ondomready)コールバックもあります。これは、JavaScriptの[`body onload`](https://www.w3schools.com/jsref/event_onload.asp)イベントと同等のものです。 また、[OnBeforeClose](reference/options.mdx#onbeforeclose)を指定すると、ウィンドウを閉じる(またはアプリを終了する)イベントにフックさせることもできます。 + +#### メソッドのバインド + +`Bind`オプションは、Wailsアプリで最も重要なオプションの1つです。 このオプションでは、フロントエンドに公開する、構造体のメソッドを指定することができます。 構造体は、従来のWebアプリにおける"コントローラ"の立ち位置であるとお考えください。 アプリが起動すると、`Bind`オプションで指定されている構造体を対象に、その中にあるパブリックメソッド(大文字で始まるメソッド名)を探します。そして、フロントエンドのコードからそれらのメソッドを呼び出せるJavaScriptが生成されます。 + +:::info 備考 + +Wailsで構造体を正しくバインドするためには、構造体の*インスタンス*をオプションで指定してください。 + +::: + +下記のコードでは、新しく`App`インスタンスを作成し、`wails.Run`関数の`Bind`オプションの中で、そのインスタンスを追加しています: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +構造体は、好きな数だけバインドできます。 `Bind`には、構造体のインスタンスを渡すようにしてください: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +`wails dev`コマンド(または、`wails generate module`コマンド)を実行すると、以下のものを含むフロントエンドモジュールが生成されます: + +- バインドされたすべてのメソッドのJavaScript +- バインドされたすべてのメソッドのTypeScript宣言 +- バインドされたメソッドの引数または返り値で使用されているGoの構造体のTypeScript宣言 + +これにより、強力な型付けがなされたデータ構造を使用して、フロントエンドから簡単にGoのコードを呼び出すことができます。 + +## フロントエンド + +### 概要 + +フロントエンドは、webkitによってレンダリングされるファイル群です。 ブラウザとWebサーバが一体となったような動きをします。 使用できるフレームワークやライブラリの制限はほぼありません[^1]。 フロントエンドとGoコードとの相互のやり取りについて、主なポイントは次のとおりです: + +- バインドされたGoメソッドの呼び出し +- ランタイムメソッドの呼び出し + +### バインドされたGoメソッドの呼び出し + +`wails dev`コマンドでアプリを起動すると、Go構造体のJavaScriptバインディングが`wailsjs/go`ディレクトリに自動的に生成されます(`wails generate module`コマンドでも生成可能)。 生成されたファイルには、アプリ内のGoパッケージ名が反映されています。 先の例では、`Greet`というパブリックメソッドを持つ`app`をバインドしていました。 この場合、次のようなファイルが生成されることになります: + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +ご覧のとおり、`main`パッケージ内の`App`構造体JavaScriptバインディングが、それらのメソッドのTypeScript型定義と並んで生成されていることが分かります。 フロントエンドから`Greet`メソッドを呼び出すには、単純にメソッドをインポートし、通常のJavaScript関数と同じように呼び出してください: + +```javascript +// ... +import {Greet} from '../wailsjs/go/main/App' + + function doGreeting(name) { + Greet(name).then((result) => { + // resultを使って何かする + }) + } +``` + +TypeScript型定義ファイルは、バインドされたメソッドの正しい型を提供します: + +```ts +export function Greet(arg1: string): Promise; +``` + +生成されたメソッドはPromiseを返すようになっています。 呼び出しが成功すると、Goからの1番目の返り値が`resolve`ハンドラに渡されます。 呼び出しに失敗したとみなされるのは、Goメソッドの2番目の返り値がerror型で、それがerrorインスタンスを返したときです。 これは、`reject`ハンドラを介して返されます。 先の例では、`Greet`メソッドは`string`型の返り値のみであるため、無効なデータが渡されない限り、JavaScript側でrejectハンドラが呼ばれることはありません。 + +すべてのデータ型は、GoとJavaScriptの間で正しく解釈されます。 もちろん構造体も正しく解釈されます。 Goから構造体が返された場合、フロントエンドにはJavaScriptのクラスとして返されます。 + +:::info 備考 + +TypeScript型定義を正しく自動生成するために、構造体のフィールドには、有効な`json`タグを_必ず_付与するようにしてください。 + +ネストされた匿名構造体(無名構造体) は、現時点ではサポートされていません。 + +::: + +Goに対して引数として構造体を渡すこともできます。 構造体として取り扱ってほしいJavaScriptのマップやクラスを渡すと、構造体に変換されます。 あなたが簡単にこれらのことを把握できるように、`dev`モードでは、バウンドされたGoメソッドで使用されている全構造体の型が定義された、Typescriptモジュールが生成されます。 このモジュールを使用すると、JavaScriptネイティブなオブジェクトを構築し、Goコードへ送信することができます。 + +また、シグネチャに構造体を使用するGoメソッドもサポートされています。 バインドされたメソッドで、引数または返り値として指定されているすべてのGo構造体は、Goのコードラッパーモジュールの一部として生成されたTypeScript定義を持っています。 れらを使用することで、GoとJavaScriptの間で、同じデータモデルを共有できます。 + +例: `Greet`メソッドを更新して、文字列型の代わりに`Person`型を引数で受け付けてみる: + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +`wailsjs/go/main/App.js`ファイルには、次のコードが出力されます: + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +しかし、`wailsjs/go/main/App.d.ts`ファイルは次のコードが出力されます: + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +見ると分かるように、"main"名前空間は、新しく生成された"models.ts"ファイルからインポートされています。 このファイルには、バインドされたメソッドで使用されるすべての構造体の型定義が含まれています。 この例では、`Person`構造体の型定義が含まれています。 `models.ts`を確認すれば、モデルがどのように定義されているかが分かります。 + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +フロントエンドのビルド構成にTypescriptを使用している限り、これらのモデルを次のように使用できます: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +生成されたバインディングとTypescriptモデルの組み合わせによって、強力な開発環境を実現させています。 + +バインディングの詳細については、[アプリ開発ガイド](guides/application-development.mdx)の[バインディングメソッド](guides/application-development.mdx#binding-methods)をご覧ください。 + +### ランタイムメソッドの呼び出し + +Javascriptランタイムは`window.runtime`に存在し、イベント発行やロギングなど、さまざまなタスクを実行するためのメソッドが含まれています: + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +Javascriptランタイムの詳細については、[ランタイムリファレンス](reference/runtime/intro)をご覧ください。 + +[^1]: まれに、WebViewでサポートされていない機能を使用するライブラリがあります。 ほとんどの場合、それらは代替手段や回避方法がありますので、それらを検討してください。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx new file mode 100644 index 00000000000..174ffb4f88f --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# イントロダクション + +Wailsは、Go言語とWeb技術を使用して、デスクトップアプリの構築を可能にするプロジェクトです。 + +"Goの力によって、Electronが軽量かつ高速になったようなもの"、と考えるとよいでしょう。 Goの柔軟性とパワーに、リッチでモダンなフロントエンドを組み合わせたアプリを、簡単に構築することができます。 + +### 特徴 + +- ネイティブなメニュー、ダイアログ、テーマ、透過処理を使用可能です +- Windows、macOS、Linuxをサポートしています +- Svelte、React、Preact、Vue、Lit、バニラJS向けにビルトインテンプレートを用意しています +- GoメソッドをJavaScriptから簡単に呼び出せます +- Go構造体に対応するTypeScript型定義を自動生成します +- WindowsにおいてCGOや外部DLLは必要ありません +- [Vite](https://vitejs.dev/)の力を利用したライブ開発が可能です +- アプリケーションを簡単に作成、ビルド、パッケージングするための強力なCLIを備えています +- 豊富な[ランタイムライブラリ](/docs/reference/runtime/intro)を用意しています +- Wailsで構築されたアプリケーションは、Apple StoreおよびMicrosoft Storeに準拠しています + +例えば [varly](https://varly.app) は、Wailsで構築されたMacOS・Windows向けのデスクトップアプリです。 見栄えが良いだけではなく、ネイティブメニューや半透明効果を使用しています。これらは、モダンなネイティブアプリに期待されるものです。 + +```mdx-code-block +

+ + + +

+``` + +### クイックスタートテンプレート + +Wailsには、アプリの開発をすばやく始められるように、多数のテンプレートが用意されています。 Svelte、React、Vue、Preact、Lit向け、およびバニラなJavaScript用で、それぞれのテンプレートがあります。 各テンプレートには、Javascript版とTypescript版が用意されています。 + +### ネイティブ要素 + +Wailsは、ウィンドウ、メニュー、ダイアログなどのネイティブ要素を処理する専用ライブラリを使用するため、見栄えが良く、リッチな機能を備えたデスクトップアプリを構築できます。 + +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. 例えばWindowsの場合、Chromium上でビルトされているMicrosoft Webview2ライブラリを使用します。 + +### GoとJavascriptのやり取り + +Wailsは自動的に、GoのメソッドをJavascriptから利用できるようにするので、フロントエンドからGoのメソッド名を呼び出すことができます。 Goメソッドが使用する構造体を表すTypeScritpt型定義も自動生成されるため、GoとJavaScriptの間で同じデータ構造をやり取りすることができます。 + +### ランタイムライブラリ + +WailsはGoとJavascriptの両方にランタイムライブラリを提供し、イベント、ロギング、ダイアログなど、モダンなアプリに必要な多くの機能を使うことができます。 + +### ライブ開発 + +#### 自動リビルド + +アプリを"dev"モードで起動すると、Wailsはあなたが書いたコードをネイティブデスクトップアプリとしてビルドしますが、プログラムアセットは常にディスクから読み込む状態になります。 その仕組みにより、アプリ起動中にGoコードが書き換えられると、変更を検出して自動的にアプリがリビルドされ、再起動します。 + +#### 自動リロード + +フロントエンドアセットの変更が検出された場合、起動中のアプリは自動的にリロードされ、変更がすぐに反映されます。 + +#### ブラウザを使った開発 + +ブラウザでのデバッグや開発も、Wailsにお任せください。 起動中のアプリはWebサーバを兼ねており、お好きなブラウザから接続してアプリを操作することができます。 プログラムアセットが書き換わった時はすぐに更新されます。 + +### 本番用のネイティブバイナリ + +アプリを本番用にビルドする準備ができたら、CLIが、アプリを単一の実行可能ファイルへコンパイルし、すべてのアセットをバンドルしてくれます。 WindowsおよびMacOSでは、配布用のネイティブパッケージを作成できます。 パッケージ化に必要なアイコン、info.plist、マニフェストファイルなどは、プロジェクトの一部としてカスタマイズできるため、アプリのビルド方法をフルコントロールできます。 + +### ツール + +Wails CLIを使うことで、簡単にアプリを生成、ビルド、バンドルすることができます。 アイコンの作成、最適なコンパイル構成の設定、本番向けのバイナリ配布など、面倒なことをあなたの代わりに引き受けてくれます。 多数のテンプレートの中から、あなたに合ったものを選んで、すぐに開発を始めましょう! diff --git a/website/versioned_docs/version-v2.5.0/reference/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx new file mode 100644 index 00000000000..3a7ff0a1ae5 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx @@ -0,0 +1,240 @@ +--- +sidebar_position: 2 +--- + +# CLI + +WailsのCLIには、プロジェクトの管理に使用できるコマンドが多数あります。 すべてのコマンドは次の文法で実行できます: + +`wails <コマンド> <フラグ>` + +## init + +`wails init`はプロジェクトの生成に使用します。 + +| フラグ | 説明 | デフォルト | +|:----------------- |:----------------------------------------------------------------------------- |:-------:| +| -n "プロジェクト名" | プロジェクトの名前。 **必須項目**。 | | +| -d "プロジェクトディレクトリ" | 作成するプロジェクトディレクトリ | プロジェクト名 | +| -g | gitリポジトリを初期化する | | +| -l | 利用可能なプロジェクトテンプレート一覧を表示 | | +| -q | コンソールへの出力を抑止 | | +| -t "テンプレート名" | 使用するプロジェクトテンプレート。 ここで指定する値は、デフォルトテンプレート名、または、GitHubでホストされているリモートテンプレートのURLです。 | vanilla | +| -ide | IDEプロジェクトファイルを生成 | | +| -f | アプリケーションを強制的にビルド | false | + +例: `wails init -n test -d mytestproject -g -ide vscode -q` + +この例では、サイレントモードで、"mytestproject"ディレクトリに"test"という名前のプロジェクトが生成されるとともに、gitの初期化、vscodeプロジェクトファイルの生成が行われます。 + +WailsでIDEを使用する場合の詳細については、[こちら](../guides/ides.mdx)をご覧ください。 + +### リモートテンプレート + +WailsではGitHubでホストされているリモートテンプレートをサポートしており、テンプレートのプロジェクトURLを使用してインストールできます。 + +例: `wails init -n test -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +コミュニティがメンテナンスしているテンプレートの一覧は[こちら](../community/templates.mdx)をご覧ください。 + +:::warning 注意 + +**Wailsプロジェクトでは、サードパーティ製テンプレートのメンテナンスは行っておらず、責任も負いません!** + +テンプレートについてよく分からない場合は、`package.json`および`wails.json`を確認し、どのようなスクリプトが実行されるのかや、どのようなパッケージがインストールされるのかを調べてください。 + +::: + +## build + +`wails build`は、プロジェクトを本番配布用のバイナリにコンパイルするときに使用します。 + +| フラグ | 説明 | デフォルト | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | `build/bin`ディレクトリをクリーンする | | +| -compiler "compiler" | 違うGoコンパイラを使用する。例: go1.15beta1 | go | +| -debug | アプリケーションのデバッグ情報を保持し、デバッグコンソールを表示する。 これにより、アプリケーションウィンドウで開発者ツールを使用することを許可できます。 | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | 実際には実行せずにbuildコマンドの結果を表示する | | +| -f | アプリケーションを強制的にビルド | | +| -garbleargs | garbleへ渡す引数 | `-literals -tiny -seed=random` | +| -ldflags "flags" | コンパイラに渡す追加のldflags | | +| -m | コンパイル前のmod tidyの実行をスキップする | | +| -nopackage | アプリケーションをパッケージ化しない | | +| -nocolour | 出力文字に色をつけない | | +| -nosyncgomod | go.modとWailsのバージョンを同期させない | | +| -nsis | Windows向けのNSISインストーラを生成する | | +| -o filename | 出力ファイル名 | | +| -obfuscated | [garble](https://github.com/burrowers/garble)を使用してアプリケーションを難読化する | | +| -platform | 指定された[プラットフォーム](../reference/cli.mdx#platforms)(カンマ区切り) 向けにビルドする。例: `windows/arm64`。 アーキテクチャを指定しない場合は、`runtime.GOARCH`の値が使用されます。 | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Goのrace detectorを使用してビルドする | | +| -s | フロントエンドのビルドをスキップ | | +| -skipbindings | バインディングの生成をスキップする | | +| -tags "extra tags" | Goコンパイラに渡すビルドタグ。 値は引用符で囲んでください。 また、スペースまたはカンマで区切ってください(両方は使用しないでください)。 | | +| -trimpath | 実行可能ファイルから、すべてのファイルシステムパスを削除する | | +| -u | プロジェクトの`go.mod`を更新し、CLIと同じバージョンのWailsを使用する | | +| -upx | "upx"を使用して最終的にバイナリを圧縮する | | +| -upxflags | upxに渡すフラグ | | +| -v int | 詳細度レベル (0 - サイレント, 1 - デフォルト, 2 - 詳細) | 1 | +| -webview2 | WebView2インストーラーのストラテジ: download,embed,browser,error | download | +| -windowsconsole | Windiws向けビルドでコンソールウィンドウを維持する | | + +`webview2`フラグの詳細については、[Windows](../guides/windows.mdx)ガイドをご覧ください。 + +標準のGoツールを使用してビルドしたい場合は、[手動ビルド](../guides/manual-builds.mdx)ガイドをご覧ください。 + +例: + +`wails build -clean -o myproject.exe` + +:::info + +Macの場合、`Info.dev.plist`ではなく`Info.plist`がアプリケーションにバンドルされます。 + +::: + +:::info AppleシリコンでのUPX + +AppleシリコンにおけるUPXの使用には[既知の問題](https://github.com/upx/upx/issues/446)が確認されています。 + +::: + +:::info WindowsでのUPX + +いくつかのアンチウイルスソフトでは、`upx`で圧縮されたバイナリをウイルスとして検知することが確認されています。詳しくは、[upxのIssue](https://github.com/upx/upx/issues/437)をご覧ください。 + +::: + +### プラットフォーム + +サポートされているプラットフォームは次のとおりです: + +| プラットフォーム | 説明 | +|:---------------- |:-------------------------------- | +| darwin | MacOS + ビルドマシンのアーキテクチャ | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | MacOS AMD64+ARM64 ユニバーサルアプリケーション | +| windows | Windows 10/11 + ビルドマシンのアーキテクチャ | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + ビルドマシンのアーキテクチャ | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## doctor + +`wails doctor`は、あなたのコンピュータで開発の準備が整っているかを診断します。 + +例: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## dev + +`wails dev`は、アプリケーションを"ライブ開発"モードで実行させたいときに使用します。 このコマンドにより、次のことが行われます: + +- アプリケーションの`go.mod`が、CLIと同じバージョンのWailsを使用するように更新されます +- アプリケーションがコンパイルされた後、自動的に実行されます +- ウォッチャーが起動し、Goファイルの変更を検出した際には、アプリがリビルドされます +- `http://localhost:34115`でWebサーバが起動し、HTTP経由でアプリケーション(フロントエンドだけではありません)が提供されます。 これにより、任意のブラウザ拡張機能を使用することができます +- すべてのアプリケーションアセットはディスクから読み込まれます。 アセットが変更された場合、アプリケーションは自動的に、リビルドではなくリロードされます。 接続されているすべてのブラウザもリロードされます +- 以下のものを含むJSモジュールが生成されます: +- GoメソッドのJavaScriptラッパー (コードヒントに有用なJSDocも自動付与されています) +- Goの構造体のTypeScriptバージョン (構造体のインスタンスを生成したり、Goメソッドの引数として渡したりすることができます) +- 別のJSモジュールとして、ランタイムのラッパーおよびTS定義も生成されます +- macOSの場合、アプリケーションは`.app`ファイルにバンドルされて実行されます。 これには、開発用の`build/darwin/Info.dev.plist`を使用します。 + +| フラグ | 説明 | デフォルト | +|:---------------------------- |:----------------------------------------------------------------------------------------------------------------------------- |:--------------------- | +| -appargs "args" | シェル形式でアプリケーションに渡される引数 | | +| -assetdir "./path/to/assets" | 通常のアセットFSを使用する代わりに、指定されたディレクトリからアセットを提供する | `wails.json`で指定されている値 | +| -browser | 起動時にブラウザで`http://localhost:34115`を開く | | +| -compiler "compiler" | 違うGoコンパイラを使用する。例: go1.15beta1 | go | +| -debounce | アセットの変更が検出されたあと、リロードするまでの時間 | 100 (ミリ秒) | +| -devserver "host:port" | Wails開発サーバをバインドするアドレス | "localhost:34115" | +| -extensions | リビルドをトリガーする拡張子 (カンマ区切り) | go | +| -forcebuild | アプリケーションを強制的にビルドする | | +| -frontenddevserverurl "url" | アセットを提供するサードパーティ製の開発サーバ(例: Vite) を使用する | "" | +| -ldflags "flags" | コンパイラに渡す追加のldflags | | +| -loglevel "loglevel" | 使用するログレベル - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | CLIのカラー出力を無効にする | false | +| -noreload | アセットが変更されたときの自動リロードを無効にする | | +| -nosyncgomod | go.modとWailsのバージョンを同期させない | false | +| -race | Goのrace detectorを使用してビルドする | false | +| -reloaddirs | リロードをトリガーする追加ディレクトリ (カンマ区切り) | `wails.json`で指定されている値 | +| -s | フロントエンドのビルドをスキップ | false | +| -save | 指定された`assetdir`、`reloaddirs`、`wailsjsdir`、`debounce`、`devserver`、`frontenddevserverurl`フラグの値を、`wails.json`へ保存し、次回以降のデフォルト値にする | | +| -skipbindings | バインディングの生成をスキップする | | +| -tags "extra tags" | コンパイラへ渡すビルドタグ (引用符およびスペース区切り) | | +| -v | 詳細度レベル (0 - サイレント, 1 - デフォルト, 2 - 詳細) | 1 | +| -wailsjsdir | 生成されたWailsのJSモジュールを格納するディレクトリ | `wails.json`で指定されている値 | + +例: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +この例では次のことが行われます: + +- アプリケーションをビルドして実行する (詳しくは[こちら](../guides/manual-builds.mdx)をご覧ください) +- WailsのJSモジュールを`./frontend/src`ディレクトリ内に生成する +- `./frontend/dist`ディレクトリ内のファイルの更新を監視し、変更されたときにリロードする +- ブラウザを開き、アプリケーションへ接続する + +既存のフレームワークスクリプトで本機能を使用する方法について詳しくは[こちら](../guides/application-development.mdx#live-reloading)をご覧ください。 + +## generate + +### template + +Wailsは、プロジェクトの生成に必ずテンプレートを使用します。 `wails generate template`コマンドは、プロジェクト生成時に使用できるテンプレートの作成を支援します。 + +| フラグ | 説明 | +|:---------------- |:---------------------------- | +| -name | テンプレート名 (必須項目) | +| -frontend "path" | テンプレートで使用するフロントエンドプロジェクトへのパス | + +テンプレートの作成について詳しくは、[テンプレートガイド](../guides/templates.mdx)をご覧ください。 + +### module + +`wails generate module`コマンドを使用すると、アプリケーションの`wailsjs`ディレクトリを手動で生成できます。 + +## update + +`wails update`コマンドを実行すると、Wails CLIのバージョンをアップデートできます。 + +| フラグ | 説明 | +|:------------------ |:----------------------- | +| -pre | 最新のプレリリースバージョンにアップデートする | +| -version "version" | 特定のバージョンのCLIをインストールする | + +## version + +`wails version`は、現在のCLIバージョンを出力するだけのコマンドです。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx new file mode 100644 index 00000000000..5e1f1e21808 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# メニュー + +Wailsプロジェクトに、アプリケーションメニューを追加することができます。 [Menu](#menu)構造体を定義し、アプリケーション設定の[`Menu`](../reference/options.mdx#menu)オプションへ設定するか、ランタイムの[MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu)メソッドを呼び出すことで、メニューを表示させることができます。 + +メニューを作成する例: + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +Menu構造体を更新し、[MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu)メソッドを呼び出すことで、メニューを動的に更新することも可能です。 + +上記の例では、ヘルパーメソッドを使用していますが、Menu構造体を手動で構築することも可能です。 + +## Menu + +Menuは、MenuItemのコレクションです。 + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +アプリケーションメニューにおいて、各MenuItemは、"編集"などの単一メニューを表します。 + +メニューを構築するするためのシンプルなヘルパーメソッドが提供されています: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +これにより、コードのレイアウトが、実際のメニューと似たレイアウトになるため、メニュー項目を作成した後にそれを手動で追加するといった作業は必要なくなります。 ヘルパーを使用せず、メニュー項目を作成して手動でメニューに追加することもできます。 + +## MenuItem + +MenuItemは、メニュー内の項目を表します。 + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| フィールド | 型 | 内容 | +| ----------- | ------------------------------------ | -------------------------------------------- | +| Label | string | メニューのテキスト | +| Accelerator | [\*keys.Accelerator](#accelerator) | このメニュー項目のキーバインディング | +| 型 | [型](#type) | メニュー項目の種類 | +| Disabled | bool | メニュー項目を無効化する | +| Hidden | bool | メニュー項目を非表示にする | +| Checked | bool | 項目にチェックを追加する (チェックボックス & ラジオタイプ) | +| SubMenu | [\*Menu](#menu) | サブメニューを設定する | +| Click | [Callback](#callback) | メニューがクリックされたときのコールバック関数 | +| Role | string | メニュー項目に[ロール](#role)を定義する。 現在のところ、Macでのみ有効です。 | + +### Accelerator + +Accelerator(キーボードショートカット) は、キーストロークとメニュー項目とのバインドを定義します。 Wailsでは、Acceleratorを、キー + [Modifier](#modifier)の組み合わせとして定義しています。 これらは`"github.com/wailsapp/wails/v2/pkg/menu/keys"`パッケージから利用可能です。 + +例: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +キーは、`+`を除いて、キーボード上の任意の一文字です(`+`は、`plus`で定義されています)。 いくつかのキーは一文字で表現できないため、名前付き文字のセットがあります: + +| | | | | +|:-----------:|:-----:|:-----:|:---------:| +| `backspace` | `f1` | `f16` | `f31` | +| `tab` | `f2` | `f17` | `f32` | +| `return` | `f3` | `f18` | `f33` | +| `enter` | `f4` | `f19` | `f34` | +| `escape` | `f5` | `f20` | `f35` | +| `left` | `f6` | `f21` | `numlock` | +| `right` | `f7` | `f22` | | +| `up` | `f8` | `f23` | | +| `down` | `f9` | `f24` | | +| `space` | `f10` | `f25` | | +| `delete` | `f11` | `f36` | | +| `home` | `f12` | `f37` | | +| `end` | `f13` | `f38` | | +| `page up` | `f14` | `f39` | | +| `page down` | `f15` | `f30` | | + +またWailsでは、Electronと同じ構文のAcceleratorを解析することもできます。 設定ファイルにAcceleratorを保存する際に便利です。 + +例: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### Modifier + +ModifierはAcceleratorキーと組み合わせて使用できるキーです: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +Modifierを使用してAcceleratorを作成するためのヘルパーメソッドがあります: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +Modifierは`keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)`を使用して組み合わせることができます: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### 型 + +各メニュー項目は5種類のうちの1タイプを指定する必要があります: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +使い勝手の良い、メニュー項目をすばやく作成するためのヘルパーメソッドが提供されています: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +"Add"ヘルパーメソッドを使用して、メニューに対して直接メニュー項目を追加することも可能です: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +ラジオグループに関する注意事項: ラジオグループは、メニュー内でお互いに隣接しているラジオメニュー項目同士で1つのグループとして定義されます。 つまり、手動でグループ化する必要はありません。 逆に言うと、2つのラジオグループを隣接させることはできません。ラジオグループの間には、ラジオ項目ではないメニューアイテムが必要です。 + +### Callback + +各メニュー項目には、項目がクリックされたときに呼び出されるコールバックがあります: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +このコールバックには、どのメニュー項目がコールバックをトリガーしたのかを識別できる、`CallbackData`構造体が渡されます。 これは、同じコールバックを呼び出すラジオグループを作成する際などに便利です。 + +### Role + +:::info ロール + +現在、ロールはMacでのみサポートされています。 + +::: + +メニュー項目には、もともと事前に定義されたメニュー項目のロールを設定することができます。 現在サポートされているロールは次のとおりです: + +| ロール | 説明 | +| ------------ | -------------------------------------------- | +| AppMenuRole | Macアプリケーションの標準メニュー。 `menu.AppMenu()`で作成できます。 | +| EditMenuRole | Macの標準的な編集メニュー。 `menu.EditMenu()`で作成できます。 | diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx new file mode 100644 index 00000000000..2f53915a0ea --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx @@ -0,0 +1,853 @@ +--- +sidebar_position: 3 +--- + +# オプション + +## アプリケーションオプション + +`Options.App`構造体には、アプリケーションの構成設定が含まれています。 これを`wails.Run()`メソッドに渡してください: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // ユーザへのメッセージをカスタマイズします + Messages *windows.Messages + // OnSuspendはWindowsが省電力モードに移行した場合に呼び出されます + OnSuspend func() + // OnResumeはWindowsが省電力モードから復帰した場合に呼び出されます + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### Title + +ウィンドウのタイトルバーに表示されるテキストです。 + +名前: Title
データ型: `string` + +### Width + +ウィンドウの初期幅です。 + +名前: Width
データ型: `int`
デフォルト値: 1024 + +### Height + +ウィンドウの初期の高さです。 + +名前: Height
データ型: `int`
デフォルト値: 768 + +### DisableResize + +デフォルトでは、メインウィンドウのサイズは自在に変更することができます。 この設定を`true`にすると、サイズが固定されます。 + +名前: DisableResize
データ型: `bool` + +### Fullscreen + +非推奨: 代わりに[WindowStartState](#windowstartstate)を使用してください。 + +### WindowStartState + +起動時にウィンドウをどのように表示させるかを定義します。 + +| 値 | Win | Mac | Lin | +| ---------- | --- | --- | --- | +| Fullscreen | ✅ | ✅ | ✅ | +| Maximised | ✅ | ✅ | ✅ | +| Minimised | ✅ | ❌ | ✅ | + +名前: WindowStartState
データ型: `options.WindowStartState` + +### Frameless + +この設定を`true`にすると、ウィンドウに境界線やタイトルバーが表示されなくなります。 詳しくは[フレームレスウィンドウ](../guides/frameless.mdx)をご覧ください。 + +名前: Frameless
データ型: `bool` + +### MinWidth + +ウィンドウの最小幅を設定します。 `Width`の値がこの値より小さい場合、ウィンドウ幅はデフォルトで`MinWidth`の値となります。 + +名前: MinWidth
データ型: `int` + +### MinHeight + +ウィンドウの最小の高さを設定します。 `Height`の値がこの値より小さい場合、ウィンドウの高さはデフォルトで`MinHeight`の値となります。 + +名前: MinHeight
データ型: `int` + +### MaxWidth + +ウィンドウの最大幅を設定します。 `Width`の値がこの値より大きい場合、ウィンドウ幅はデフォルトで`MaxWidth`の値となります。 + +名前: MaxWidth
データ型: `int` + +### MaxHeight + +ウィンドウの最大の高さを設定します。 `Height`の値がこの値より大きい場合、ウィンドウの高さはデフォルトで`MaxHeight`の値となります。 + +名前: MaxHeight
データ型: `int` + +### StartHidden + +`true`に設定すると、アプリケーションは[WindowShow](../reference/runtime/window.mdx#windowshow)が呼び出されるまで非表示となります。 + +名前: StartHidden
データ型: `bool` + +### HideWindowOnClose + +デフォルトでは、ウィンドウを閉じるとアプリケーションが終了します。 この設定を`true`にすると、ウィンドウを閉じる操作をした際に、 + +ウィンドウが非表示の状態になります。 + +名前: HideWindowOnClose
データ型: `bool` + +### BackgroundColour + +ウィンドウのデフォルトの背景色です。 例: options.NewRGBA(255,0,0,128) - 50%透過された赤色 + +名前: BackgroundColour
データ型: `*options.RGBA`
デフォルト値: white + +### AlwaysOnTop + +ウィンドウへのフォーカスが無くなっても、他ウィンドウより手前側にウィンドウを表示させるかを設定します。 + +名前: AlwaysOnTop
データ型: `bool` + +### Assets + +非推奨: 代わりに[AssetServerの固有オプション](#assetserver)であるAssetsオプションを使用してください。 + +### AssetsHandler + +非推奨: 代わりに[AssetServerの固有オプション](#assetserver)であるAssetsHandlerオプションを使用してください。 + +### AssetServer + +AssetServerの固有オプションを定義します。 静的なアセットでAssetServerをカスタマイズしたり、`http.Handler`で動的なアセットを提供したり、`assetserver.Middleware`でリクエストチェーンにフックしたりすることができます。 + +現在のところ、`http.Request`のすべての機能がサポートされているわけではありません。次の機能マトリクスを確認してください: + +| 機能 | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +名前: AssetServer
データ型: `*assetserver.Options` + +#### Assets + +アプリケーションのフロントエンドで使用される静的アセットです。 + +GETリクエストが要求された場合、まず初めに、この`fs.FS`からのアセット提供を試みます。 `fs.FS`から、当該ファイルが存在しない旨の`os.ErrNotExist`エラーが返された場合、このリクエストは[Handler](#handler)へフォールバックされ、ハンドラ側でGETリクエストへの応答を試みます。 + +nilがセットされている場合、GETリクエストに常に[Handler](#handler)へ転送されます。 + +名前: Assets
データ型: `fs.FS` + +#### Handler + +アセットハンドラは、アセットが見つからなかった場合のフォールバック処理を担う、ジェネリックな`http.Handler`です。 + +[Assets](#assets)内で`os.ErrNotExist`エラーが発生したことにより、アセットを提供できないと判断されたすべてのGETリクエストによって、このハンドラが呼び出されます。 また、GET以外のすべての種類のリクエストは、常にこのハンドラから応答が返されます。 ハンドラが定義されていない状態で、ハンドラが呼び出された場合、次の応答が返されます: + +- GETリクエスト: `http.StatusNotFound` +- その他のリクエスト: `http.StatusMethodNotAllowed` + +注意: フロントエンド側の開発サーバと組み合わせて使用すると、一部制限がかかる場合があります。 Viteは、拡張子を含まないパスに対して、常にindex.htmlを返します。 + +名前: AssetsHandler
データ型: `http.Handler` + +#### Middleware + +MiddlewareはHTTPミドルウェアで、AssetServerのリクエストチェーンをフックすることができます。 例えば、特殊なルーティングを実装したいときなど、デフォルトのリクエストハンドラを動的にスキップさせることができます。 Middlewareは、AssetSeverが使用する新しい`http.Handler`を生成するために呼び出されます。引数では、AssetServerがデフォルトで使用するハンドラを受け取ります。 + +Middlewareが定義されていない場合、デフォルトのAssetServerのリクエストチェーンが実行されます。 + +名前: Middleware
データ型: `assetserver.Middleware` + +### Menu + +アプリケーションで使用されるメニューです。 メニューについて詳しくは、[メニューのリファレンス](../reference/runtime/menu.mdx)をご覧ください。 + +:::note + +Macでは、メニューが指定されていない場合、デフォルトメニューが作成されます。 + +::: + +名前: Menu
データ型: `*menu.Menu` + +### Logger + +アプリケーションで使用するロガーです。 ロギングについて詳しくは、[ログのリファレンス](../reference/runtime/log.mdx)をご覧ください。 + +名前: Logger
データ型: `logger.Logger`
デフォルト値: 標準出力へのロガー + +### LogLevel + +デフォルトのログレベルです。 ロギングについて詳しくは、[ログのリファレンス](../reference/runtime/log.mdx)をご覧ください。 + +名前: LogLevel
データ型: `logger.LogLevel`
デフォルト値: 開発モードの場合は`Info`、本番モードの場合は`Error` + +### LogLevelProduction + +本番ビルド時のデフォルトのログレベルです。 ロギングについて詳しくは、[ログのリファレンス](../reference/runtime/log.mdx)をご覧ください。 + +名前: LogLevelProduction
データ型: `logger.LogLevel`
デフォルト値: `Error` + +### OnStartup + +フロントエンド作成後、`index.html`が読み込まれる前に呼び出されるコールバックです。 アプリケーションのcontextが渡されます。 + +名前: OnStartup
データ型: `func(ctx context.Context)` + +### OnDomReady + +フロントエンドが`index.html`とそのリソースを読み込んだ後に呼び出されるコールバックです。 アプリケーションのcontextが渡されます。 + +名前: OnDomReady
データ型: `func(ctx context.Context)` + +### OnShutdown + +フロントエンドが破棄され、アプリケーションが終了する直前に呼び出されるコールバックです。 アプリケーションのcontextが渡されます。 + +名前: OnShutdown
データ型: `func(ctx context.Context)` + +### OnBeforeClose + +ウィンドウの閉じるボタンをクリックするか、`runtime.Quit`が呼ばれて、アプリケーションが終了されそうになっているときに呼び出されるコールバックです。 trueを返すとアプリケーションはそのまま維持され、falseを返すと通常どおりシャットダウンされます。 ユーザに対してプログラムの終了を確認したいときは、このコールバックを使うのが良いでしょう。 + +例: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +名前: OnBeforeClose
データ型: `func(ctx context.Context) bool` + +### CSSDragProperty + +ウィンドウをドラッグできる要素を特定するためのCSSプロパティ名を設定します。 デフォルト値: `--wails-draggable` + +名前: CSSDragProperty
データ型: `string` + +### CSSDragValue + +ウィンドウのドラッグを有効にするために、`CSSDragProperty`スタイルが持つべき値を設定します。 デフォルト値: `drag` + +名前: CSSDragValue
データ型: `string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenuは、本番環境において、ブラウザのデフォルトコンテキストメニューを有効にします。 + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +このオプションを有効にすると、デフォルトでは、テキストに関するコンテキスト(切り取り/コピー/貼り付け) のみがコンテキストメニューに表示されます。この動作をオーバーライドするには、`--default-contextmenu`というCSSプロパティを任意のHTML要素(`body`含む) で以下の値と共に使用してください: + +| CSSスタイル | 動作 | +| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------ | +| `--default-contextmenu: auto;` | (**デフォルト**) 次の場合にのみデフォルトのコンテキストメニューを表示します:
contentEditableがtrueである、またはテキストが選択されている、またはinput要素/textarea要素であるとき | +| `--default-contextmenu: show;` | 常にデフォルトのコンテキストメニューを表示します | +| `--default-contextmenu: hide;` | 常にデフォルトのコンテキストメニューを非表示にします | + +このルールは通常のCSSルールと同様に継承されるため、ネストも期待どおりに動作します。 + +:::note +このフィルタリング機能は本番環境でのみ有効であり、開発・デバッグビルドでは、フルコンテキストメニューが常に使用できます。 +::: + +:::warning +このフィルタリング機能はセキュリティ対策として使用することはできません。開発者は、画像のダウンロード、リロード、ウェブページの保存といったコマンドを含むフルコンテキストメニューが常にリークされる可能性を考慮すべきです。この点が気になる場合、開発者はデフォルトのコンテキストメニューを有効にするべきではありません。 +::: + + +名前: EnableDefaultContextMenu
データ型: `bool` + +### EnableFraudulentWebsiteDetection + +EnableFraudulentWebsiteDetectionは、マルウェアやフィッシング詐欺などの不正コンテンツのスキャンサービスを有効にします。 これらのサービスは、ナビゲートされたURLやその他コンテンツ情報を、アプリからAppleおよびMicrosoftのクラウドサービスに送信する可能性があります。 + +名前: EnableFraudulentWebsiteDetection
データ型: `bool` + +### ZoomFactor + +名前: ZoomFactor
データ型: `float64` + +WebView2の拡大率を定義します。 これは、Edgeのユーザによるズームインまたはズームアウトに対応するオプションです。 + +### IsZoomControlEnabled + +名前: IsZoomControlEnabled
データ型: `bool` + +このオプションを有効にすると、拡大率をユーザによって変更することができます。 拡大率の変更がユーザに許可されていない間は、オプションで拡大率を設定することができますのでご注意ください (例: キオスクアプリケーションなど)。 + +### Bind + +フロントエンドにバインドする必要があるメソッドが定義された、構造体インスタンスのスライスです。 + +名前: Bind
データ型: `[]interface{}` + +### ErrorFormatter + +JSからGoへ呼び出されたメソッドがエラーを返す際に、エラーをフォーマットする関数です。 返り値はJSONとして変換されます。 + +名前: ErrorFormatter
データ型: `func (error) any` + +### Windows + +[Windows固有のオプション](#windows)を定義します。 + +名前: Windows
データ型: `*windows.Options` + +#### WebviewIsTransparent + +この値を`true`に設定すると、アルファ値が`0`の際に、WebViewの背景が透明になります。 つまり、CSSで`background-color`に`rgba(0,0,0,0)`を設定すると、ホストウィンドウが透けて見えるようになります。 多くの場合、[WindowIsTranslucent](#WindowIsTranslucent)と組み合わて、霜のように見えるアプリケーションを作成する際に使用します。 + +名前: WebviewIsTransparent
データ型: `bool` + +#### WindowIsTranslucent + +この値を`true`に設定すると、ウィンドウの背景が半透明になります。 多くの場合、[WebviewIsTransparent](#WebviewIsTransparent)と組み合わせて使用されます。 + +ビルド22621より前のWindows 11の場合、半透明を実現させるために[BlurBehind](https://learn.microsoft.com/ja-jp/windows/win32/dwm/blur-ovw)メソッドを使用するため、処理が遅くなる可能性があります。 ビルド22621以降のWindows 11では、より高速な、新しい半透明タイプが有効になります。 デフォルトで使用される半透明タイプは、Windowsにより決定されます。 このタイプを設定するには、[BackdropType](#BackdropType)オプションを使用してください。 + +名前: WindowIsTranslucent
データ型: `bool` + +#### BackdropType + +:::note + +この設定を適用するには、Windows 11 ビルド22621以降が必要です。 + +::: + +ウィンドウの半透明タイプを設定します。 この設定は、[WindowIsTranslucent](#WindowIsTranslucent)が`true`に設定されている場合にのみ適用されます。 + +名前: BackdropType
データ型: `windows.BackdropType` + +値は次のいずれかを指定してください: + +| 値 | 説明 | +| ------- | ----------------------------------------------------------------------------------- | +| Auto | Windowsに背景を決定させる | +| None | 半透明にしない | +| Acrylic | [アクリル](https://learn.microsoft.com/ja-jp/windows/apps/design/style/acrylic)の効果を使用する | +| Mica | [マイカ](https://learn.microsoft.com/ja-jp/windows/apps/design/style/mica)の効果を使用する | +| Tabbed | タブを使用する。 これはマイカに似ている背景です。 | + +#### DisableWindowIcon + +この設定を`true`にすると、タイトルバーの左上隅のアイコンが消去されます。 + +名前: DisableWindowIcon
データ型: `bool` + +#### DisableFramelessWindowDecorations + +この設定を`true`にすると、[フレームレス](#Frameless)モードでのウィンドウデコレーションが消去されます。 つまり、'Aero Shadow'および'Rounded Corners'がウィンドウに適用されなくなります。 なお、'Rounded Corners'はWindows 11でのみサポートされていますのでご注意ください。 + +名前: DisableFramelessWindowDecorations
データ型: `bool` + +#### WebviewUserDataPath + +WebView2が、ユーザデータを格納するパスを設定します。 空の場合は、`%APPDATA%\[BinaryName.exe]`が使用されます。 + +名前: WebviewUserDataPath
データ型: `string` + +#### WebviewBrowserPath + +WebView2の実行ファイルおよびライブラリが存在するディレクトリへのパスを設定します。 空の場合、システムにインストールされているWebView2が使用されます。 + +固定バージョンランタイムディストリビューションに関する重要情報: + +- [ランタイムの取得および展開方法](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [固定バージョンに関する既知の問題](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [WenView2ランタイムの固定バージョンのパスに、\Edge\Application\ を含めてはいけません](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +名前: WebviewBrowserPath
データ型: `string` + +#### Theme + +サポートされるWindowsの最小バージョン: Windows 10 2004/20H1 + +アプリケーションが使用するテーマを設定します: + +| 値 | 説明 | +| ------------- | ---------------------------------------------------------------------------------------- | +| SystemDefault | _デフォルト値_です。 テーマは、システムのデフォルト設定に基づきます。 ユーザがシステムのテーマ設定を変更した場合、アプリケーションは新しい設定を使用するように更新されます。 | +| Dark | アプリケーションはダークテーマのみを使用します。 | +| Light | アプリケーションはライトテーマのみを使用します。 | + +名前: Theme
データ型: `windows.Theme` + +#### CustomTheme + +:::note + +サポートされるWindowsの最小バージョン: Windows 10/11 2009/21H2 ビルド22000 + +::: + +ウィンドウがアクティブまたは非アクティブのときに、ライトモード・ダークモードのそれぞれにおいて、タイトルバー、タイトルテキスト、ボーダーのカスタムカラーを設定できます。 + +名前: CustomTheme
データ型: `windows.CustomTheme` + +##### CustomTheme 型 + +CustomTheme構造体は、`int32`型で色の値を指定します。 これは`0x00BBGGAA`で表されるWindowsの標準フォーマットで指定します。 RGBの値をこのフォーマットに変換するために、`windows.RGB(r,g,b uint8)`のようなヘルパー関数が用意されています。 + +注意: 指定されていない値はデフォルトで黒色になります。 + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +例: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### Messages + +利用可能なWebView2ランタイムが見つからなかったときに表示する、WebView2インストーラに関するメッセージ文字列を設定します。 + +名前: Messages
データ型: `*windows.Messages` + +プロジェクトがサポートする言語にあわせて、この設定をカスタマイズしてください。 + +#### ResizeDebounceMS + +ResizeDebounceMSは、ウィンドウサイズが変更されたときに、WebView2の再描画を実行するまでの時間です。 デフォルト値(0) は、できるだけ早く再描画を実行します。 + +名前: ResizeDebounceMS
データ型: `uint16` + +#### OnSuspend + +Windowsがローパワーモード(サスペンド/休止状態) に切り替わると呼び出されるコールバックを設定します。 + +名前: OnSuspend
データ型: `func()` + +#### OnResume + +Windowsがローパワーモード(サスペンド/休止状態) から復帰したときに呼び出されるコールバックを設定します。 + +名前: OnResume
データ型: `func()` + +#### WebviewGpuIsDisabled + +`true`に設定すると、webviewのGPUハードウェアアクセラレーションが無効化されます。 + +名前: WebviewGpuIsDisabled
データ型: `bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +[Mac固有のオプション](#mac)を定義します。 + +名前: Mac
データ定義: `*mac.Options` + +#### TitleBar + +TitleBar構造体は、タイトルバーのルック・アンド・フィールを設定する機能を提供します。 + +名前: TitleBar
データ型: [`*mac.TitleBar`](#titlebar-struct) + +##### Titlebar 構造体 + +アプリケーションのタイトルバーは、TitleBarオプションを使用することでカスタマイズできます: + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| 名前 | 説明 | +| -------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TitlebarAppearsTransparent | タイトルバーを透明にします。 これにより、タイトルバーが非表示になり、コンテンツがウィンドウ全体に表示されます。 [Appleドキュメント](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | ウィンドウタイトルを非表示にします。 [Appleドキュメント](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | スタイルマスクから[NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/)を消去します。 | +| FullSizeContent | WebViewをウィンドウ全体に表示します。 [Appleドキュメント](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | ウィンドウにデフォルトツールバーを追加します。 [Appleドキュメント](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | ツールバーの下側の線を消去します。 [Appleドキュメント](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +事前設定されたタイトルバー構成を利用することも可能です: + +| 設定 | 例 | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +例: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +タイトルバーのカスタマイズに関してインスピレーションを得たい場合は[こちら](https://github.com/lukakerr/NSWindowStyles)をご覧ください。 + +#### Appearance + +Appearanceは、Appleの[NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc)Nameに従って、アプリケーションのスタイルを設定するために使用します。 + +名前: Appearance
データ型: [`mac.AppearanceType`](#appearance-type) + +##### Appearance 型 + +アプリケーションの[外観](https://developer.apple.com/documentation/appkit/nsappearance?language=objc)を指定します。 + +| 値 | 説明 | +| ----------------------------------------------------- | ------------------------------------- | +| DefaultAppearance | DefaultAppearanceは、システムのデフォルト値を使用します。 | +| NSAppearanceNameAqua | システムの標準的なライト外観 | +| NSAppearanceNameDarkAqua | システムの標準的なダーク外観 | +| NSAppearanceNameVibrantLight | より鮮やかなライト外観 | +| NSAppearanceNameAccessibilityHighContrastAqua | システムの標準的なライト外観のハイコントラスト版 | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | システムの標準的なダーク外観のハイコントラスト版 | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | より鮮やかなライト外観のハイコントラスト版 | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | より鮮やかなダーク外観のハイコントラスト版 | + +例: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### WebviewIsTransparent + +この値を`true`に設定すると、アルファ値が`0`の際に、WebViewの背景が透明になります。 つまり、CSSで`background-color`に`rgba(0,0,0,0)`を設定すると、ホストウィンドウが透けて見えるようになります。 多くの場合、[WindowIsTranslucent](#WindowIsTranslucent)と組み合わて、霜のように見えるアプリケーションを作成する際に使用します。 + +名前: WebviewIsTransparent
データ型: `bool` + +#### WindowIsTranslucent + +この値を`true`に設定すると、ウィンドウの背景が半透明になります。 多くの場合、[WebviewIsTransparent](#WebviewIsTransparent)と組み合わて、霜のように見えるアプリケーションを作成する際に使用します。 + +名前: WindowIsTranslucent
データ型: `bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| 名前 | 説明 | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +例: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### About + +"AppMenu"ロールで作成されたアプリケーションメニューの中にある"About"メニュー項目において、タイトル、メッセージ、アイコンを設定できます。 + +名前: About
データ型: [`*mac.AboutInfo`](#about-struct) + +##### About 構造体 + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +これらの設定が定義されている場合、`AppMenu`ロールが使用されていれば、アプリケーションメニュー内に"About"メニューが表示されます。 例えば以下のように設定すると: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +アプリケーションメニューに"About"メニュー項目が表示されます: + +```mdx-code-block +
+ +
+
+``` + +クリックすると、アプリケーション概要メッセージボックスが開かれます: + +```mdx-code-block +
+ +
+
+``` + +### Linux + +[Linux固有のオプション](#linux)を定義します。 + +名前: Linux
データ型: `*linux.Options` + +#### Icon + +ウィンドウを示すアイコンを設定します。 このアイコンは、ウィンドウが最小化されたときに使用されます(iconifiedと呼ばれます)。 + +名前: Icon
データ型: `[]byte` + +一部のウィンドウマネージャやデスクトップ環境では、ウィンドウフレームに配置されたり、他のコンテキストで表示される場合もあります。 反対に、環境によっては全くアイコンが使用されないこともありますのでご注意ください。 + +注意: 少なくともWayland上のGnomeでは、このアイコンは表示されません。 アプリケーションアイコンを表示させるには、`.desktop`ファイルを使用する必要があります。 KDEの場合は正常に表示されるはずです。 + +アイコンは、何の加工もされていないサイズで用意してください。つまり、拡大/縮小された画像は使用しないでください。 最高品質を確保するために、拡大/縮小は、最終的な目的サイズがはっきりするまで待ってください。 + +#### WindowIsTranslucent + +この値を`true`に設定すると、ウィンドウの背景が半透明になります。 ウィンドウマネージャによっては、この設定を無視したり、ブラックウィンドウになる場合があります。 + +名前: WindowIsTranslucent
データ型: `bool` + +#### WebviewGpuPolicy + +このオプションでは、webviewのハードウェアアクセラレーションポリシーを指定することができます。 + +名前: WebviewGpuPolicy
データ型: [`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
デフォルト値: `WebviewGpuPolicyAlways` + +##### WebviewGpuPolicy型 + +| 値 | 説明 | +| ------------------------ | --------------------------------------------- | +| WebviewGpuPolicyAlways | ハードウェアアクセラレーションを常に有効にする | +| WebviewGpuPolicyOnDemand | Webコンテンツからの要求に応じて、ハードウェアアクセラレーションの有効/無効を切り替える | +| WebviewGpuPolicyNever | ハードウェアアクセラレーションを常に無効にする | + +#### ProgramName + +このオプションでは、GTKのg_set_prgname() を使用し、ウィンドウマネージャのプログラム名を設定することができます。 ただし、ローカライズされた名前を設定すべきではありません。詳しくは[ドキュメント](https://docs.gtk.org/glib/func.set_prgname.html)をご覧ください。 + +.desktopファイルが作成される際、.desktopファイルの`Name`オプションと実行ファイルのファイル名が異なる場合に、ウィンドウのグループ化およびデスクトップアイコンの表示に、本オプションは役立ちます。 + +名前: ProgramName
データ型: string
+ +### Debug + +デバッグビルド時に適用される[デバッグ固有のオプション](#Debug)を定義します。 + +名前: Debug
データ型: `options.Debug` + +#### OpenInspectorOnStartup + +このオプションを`true`に設定すると、アプリケーション起動時にWeb開発者ツールが表示されます。 + +名前: OpenInspectorOnStartup
データ型: `bool` + +[^1]: この機能の動作には、WebKit2GTK 2.36以上が必要となるため、機能を確実に対応させたい場合は、アプリビルド時にビルドタグ`webkit2_36`を付与してください。 これにより、アプリに必要なWebKit2GTKの最小バージョン要件が2.36に引き上げられます。 +[^2]: この機能の動作には、WebKit2GTK 2.40以上が必要となるため、機能を確実に対応させたい場合は、アプリビルド時にビルドタグ`webkit2_40`を付与してください。 これにより、アプリに必要なWebKit2GTKの最小バージョン要件が2.40に引き上げられます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx new file mode 100644 index 00000000000..514c48c4ca9 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx @@ -0,0 +1,91 @@ +--- +sidebar_position: 5 +--- + +# プロジェクト構成 + +プロジェクト構成は、プロジェクトディレクトリ内の`wails.json`ファイルで設定します。 ファイルの構造は次のとおりです: + +```json5 +{ + // プロジェクト構成のバージョン。 + "version": "", + // プロジェクト名。 + "name": "", + // コンパイルされたアセットディレクトリへの相対パス。通常は自動的に推測されるため、空で構いません。 + "assetdir": "", + // 再読み込みのトリガーとなる追加のディレクトリ(カンマ区切り)。高度なアセット構成をとる場合にのみ使用します。 + "reloaddirs": "", + // ビルドファイルが存在するディレクトリ。 デフォルトは'build'です。 + "build:dir": "", + // フロントエンドディレクトリの相対パス。 デフォルトは'frontend'です。 + "frontend:dir": "", + // node依存関係をインストールするために、フロントエンドディレクトリで実行するコマンド。一般的には`npm install`です。 + "frontend:install": "", + // アセットをビルドするために、フロントエンドディレクトリで実行するコマンド。一般的には`npm run build`です。 + "frontend:build": "", + // このコマンドはfrontend:dev:buildへ置換されました。 frontend:dev:buildが指定されていない場合は、代わりにこのコマンドが実行されます。 このコマンドも指定されていない場合は、代わりにfrontend:buildが実行されます。 + "frontend:dev": "", + // 開発モードにおけるfrontend:buildと同様のコマンド。 指定されていない場合は、代わりにfrontend:devが実行されます。 + "frontend:dev:build": "", + // 開発モードにおけるfrontend:installと同様のコマンド。 指定されていない場合は、代わりにfrontend:installが実行されます。 + "frontend:dev:install": "", + // `wails dev`実行時に別プロセスで実行するコマンド。 サードパーティ製のウォッチャや開発サーバを起動したい場合に便利です。 + "frontend:dev:watcher": "", + // Viteなど、アセットを提供するサードパーティ製の開発サーバのURL。 'auto'に設定すると、Viteの出力から自動的に開発サーバURLを推測します。 + "frontend:dev:serverUrl": "", + // 自動生成されるJSモジュールを出力するディレクトリへの相対パス。 + "wailsjsdir": "", + // 出力バイナリのファイル名 + "outputfilename": "", + // アセットファイルに変更があった場合に、開発サーバが再読み込みを行うまでのデフォルトの待ち時間。 + "debounceMS": 100, + // Wailsの開発サーバをバインドするアドレス。 デフォルト値: localhost:34115 + "devServer": "", + // 開発モードのときに、アプリケーションに渡されるシェルスタイルの引数。 + "appargs": "", + // ホストOS以外のOS用にビルドフックが定義されている場合、それらを実行するかどうか。 + "runNonNativeBuildHooks": false, + "preBuildHooks": { + // 指定されたGOOS/GOARCHのビルドの前に実行されるコマンド。${platform}は'GOOS/GOARCH'に置換されます。 'GOOS/GOARCH'フックは、'GOOS/*'および'*/*'フックの前に実行されます。 + "GOOS/GOARCH": "", + // 指定されたGOOSのビルド前に実行されるコマンド。${platform}は'GOOS/GOARCH'に置換されます。 'GOOS/*'フックは、'*/*'フックの前に実行されます。 + "GOOS/*": "", + // 毎回のビルドの前に実行されるコマンド。${platform}は'GOOS/GOARCH'に置換されます。 + "*/*": "" + }, + "postBuildHooks": { + // 指定されたGOOS/GOARCHのビルドの後に実行されるコマンド。${platform}は'GOOS/GOARCH'に置換され、${bin}はコンパイル済みバイナリへのパスに置換されます。 'GOOS/GOARCH'フックは、'GOOS/*'および'*/*'フックの前に実行されます。 + "GOOS/GOARCH": "", + // 指定されたGOOSのビルド後に実行されるコマンド。${platform}は'GOOS/GOARCH'に置換され、${bin}はコンパイル済みバイナリへのパスに置換されます。 'GOOS/*'フックは、'*/*'フックの前に実行されます。 + "GOOS/*": "", + // 毎回のビルドの後に実行されるコマンド。${platform}は'GOOS/GOARCH'に置換され、${bin}はコンパイル済みバイナリへのパスに置換されます。 + "*/*": "" + }, + // マニフェストやバージョン情報で使用されるデータ。 + "info": { + // 会社名。 デフォルト値: [プロジェクト名] + "companyName": "", + // 製品名。 デフォルト値: [プロジェクト名] + "productName": "", + // 製品バージョン。 デフォルト値: '1.0.0'] + "productVersion": "", + // 製品の著作権。 デフォルト値: 'Copyright.........' + "copyright": "", + // アプリケーションの説明。 デフォルト値: 'Built using Wails (https://wails.app)' + "comments": "" + }, + // 'multiple': アーキテクチャごとに1つのインストーラ。 'single': ビルドした全アーキテクチャに対応する単一ユニバーサルインストーラ。 デフォルト値: 'multiple' + "nsisType": "", + // アプリ難読化を実行するかどうか。 デフォルト値: false + "obfuscated": "", + // obfuscatedフラグがtrueの際に、garbleコマンドへ渡す引数。 + "garbleargs": "" +} +``` + +このファイルは、Wails CLIで `wails build` コマンドまたは `wails dev` コマンドを実行した際に読み込まれます。 + +`wails build/dev` コマンドを実行時に、`assetdir`、`reloaddirs`、`wailsjsdir`、`debounceMS`、`devserver`、`frontenddevserverurl`フラグを使用すると、プロジェクト構成が更新され、次回以降コマンドを実行する際のデフォルト値となります。 + +このファイルのJSONスキーマは、[こちら](https://wails.io/schemas/config.v2.json)にあります。 diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx new file mode 100644 index 00000000000..9e146a3da00 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# ブラウザ + +これらは、システムブラウザに関連したメソッドです。 + +### BrowserOpenURL + +指定されたURLをシステムブラウザで開きます。 + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..9e632294ba2 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# クリップボード + +ランタイム機能の一部として、オペレーティングシステムのクリップボードへのアクセスを提供します。
現在は、テキストの処理のみに対応しています。 + +### ClipboardGetText + +このメソッドは、クリップボードに現在保存されているテキストを読み込みます。 + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
返り値: 文字列(クリップボードが空の場合は、空文字)またはエラー。 + +JS: `ClipboardGetText(): Promise`
返り値: 文字列を待機するPromise(クリップボードが空の場合は、空文字)。 + +### ClipboardSetText + +このメソッドは、クリップボードにテキストを書き込みます。 + +Go: `ClipboardSetText(ctx context.Context, text string) error`
返り値: anyの場合はエラー。 + +JS: `ClipboardSetText(text: string): Promise`
返り値: クリップボードにテキストが正常に書き込まれた場合はtrue、そうでない場合はfalseを返すようなPromise。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..721ba5f31d4 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# ダイアログ + +ランタイムでは、ファイルセレクターやメッセージボックスといったネイティブダイアログへのアクセスを提供しています。 + +:::info JavaScript + +現在、Javascriptランタイムではダイアログをサポートしていません。 + +::: + +### OpenDirectoryDialog + +ユーザにディレクトリの選択を求めるダイアログを開きます。 [OpenDialogOptions](#opendialogoptions)を使用してカスタマイズできます。 + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +返り値: 選択されたディレクトリ(キャンセルされた場合は空) またはエラー + +### OpenFileDialog + +ユーザにファイルの選択を求めるダイアログを開きます。 [OpenDialogOptions](#opendialogoptions)を使用してカスタマイズできます。 + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +返り値: 選択されたファイル(キャンセルされた場合は空) またはエラー + +### OpenMultipleFilesDialog + +ユーザに複数ファイルの選択を求めるダイアログを開きます。 [OpenDialogOptions](#opendialogoptions)を使用してカスタマイズできます。 + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +返り値: 選択された複数ファイル(キャンセルされた場合はnil) またはエラー + +### SaveFileDialog + +保存の目的でユーザにファイル名を入力選択させるダイアログを開きます。 [SaveDialogOptions](#savedialogoptions)を使用してカスタマイズできます。 + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +返り値: 入力選択されたファイル(キャンセルされた場合は空) またはエラー + +### MessageDialog + +メッセージダイアログを使用してメッセージを表示します。 [MessageDialogOptions](#messagedialogoptions)を使用してカスタマイズできます。 + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +返り値: 選択されたボタンのテキストまたはエラー + +## オプション + +### OpenDialogOptions + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| フィールド | 説明 | Win | Mac | Lin | +| -------------------------- | ------------------------- | --- | --- | --- | +| DefaultDirectory | ダイアログが開かれたときに初期表示するディレクトリ | ✅ | ✅ | ✅ | +| DefaultFilename | デフォルトファイル名 | ✅ | ✅ | ✅ | +| Title | ダイアログのタイトル | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | ファイルフィルタのリスト | ✅ | ✅ | ✅ | +| ShowHiddenFiles | システムの隠しファイルを表示 | | ✅ | ✅ | +| CanCreateDirectories | ユーザによるディレクトリの作成を許可する | | ✅ | | +| ResolvesAliases | エイリアスではなくファイルパスを返す | | ✅ | | +| TreatPackagesAsDirectories | パッケージへのナビゲーションを許可 | | ✅ | | + +### SaveDialogOptions + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| フィールド | 説明 | Win | Mac | Lin | +| -------------------------- | ------------------------- | --- | --- | --- | +| DefaultDirectory | ダイアログが開かれたときに初期表示するディレクトリ | ✅ | ✅ | ✅ | +| DefaultFilename | デフォルトファイル名 | ✅ | ✅ | ✅ | +| Title | ダイアログのタイトル | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | ファイルフィルタのリスト | ✅ | ✅ | ✅ | +| ShowHiddenFiles | システムの隠しファイルを表示 | | ✅ | ✅ | +| CanCreateDirectories | ユーザによるディレクトリの作成を許可する | | ✅ | | +| TreatPackagesAsDirectories | パッケージへのナビゲーションを許可 | | ✅ | | + +### MessageDialogOptions + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| フィールド | 説明 | Win | Mac | Lin | +| ------------- | ------------------------------------------------- | -------------- | --- | --- | +| Type | メッセージダイアログの種類 (質問、情報など) | ✅ | ✅ | ✅ | +| Title | ダイアログのタイトル | ✅ | ✅ | ✅ | +| Message | ユーザに表示するメッセージ | ✅ | ✅ | ✅ | +| Buttons | ボタンテキストのリスト | | ✅ | | +| DefaultButton | 指定されたテキストのボタンをデフォルトボタンとして扱う。 `return`キーにバインドされます。 | ✅[*](#windows) | ✅ | | +| CancelButton | 指定されたテキストのボタンをキャンセルボタンとして扱う。 `escape`キーにバインドされます。 | | ✅ | | + +#### Windows + +Windowsでは、ボタンのカスタマイズができない標準ダイアログタイプがあります。 返却される値は次のいずれかとなります: "Ok"、"Cancel"、"Abort"、"Retry"、"Ignore"、"Yes"、"No"、"Try Again"、"Continue"。 + +質問ダイアログの場合、デフォルトボタンは"Yes"、キャンセルボタンは"No"となります。 この設定は、`DefaultButton`の値を`"No"`にすることで、変更できます。 + +例: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linuxでは、ボタンのカスタマイズができない標準ダイアログタイプがあります。 返り値は次のいずれかになります: "Ok"、"Cancel"、"Yes"、"No" + +#### Mac + +Macのメッセージダイアログでは、最大4つまでのボタンを指定できます。 `DefaultButton`や`CancelButton`が指定されていない場合、1番目のボタンがデフォルトボタンとして扱われ、`return`キーにバインドされます。 + +例えば次のようなコードの場合: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, +}) +``` + +1番目のボタンがデフォルトになります: + +```mdx-code-block +
+ +
+
+``` + +そして`DefaultButton`を"two"に設定した場合: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +2番目のボタンがデフォルトになります。 このとき`return`キーが押されると、返り値として"two"が返却されます: + +```mdx-code-block +
+ +
+
+``` + +また、`CancelButton`を"three"に設定した場合 + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +ダイアログの下部に"three"ボタンが表示されるようになります。 このとき`escape`キーが押されると、返り値として"three"が返却されます: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### DialogType + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### FileFilter + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +Windowsでは、ダイアログボックスで複数のファイルフィルタを使用できます。 それぞれのFileFiltersは、ダイアログ上で個別のエントリーとして表示されます: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linuxでは、ダイアログボックスで複数のファイルフィルタを使用できます。 それぞれのFileFiltersは、ダイアログ上で個別のエントリーとして表示されます: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +Macのダイアログでは、ファイルをフィルタするためのパターンセットは1つしか持たせることができません。 もし複数のFileFiltersを定義した場合、Wailsはそれらすべてのパターンを使用します。 + +例: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +このとき、ファイル選択ダイアログを開くと、`*.png,*.jpg,*.mov,*.mp4`がフィルタとして使用されます。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx new file mode 100644 index 00000000000..ac4f40eba00 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# イベント + +Wailsでは、GoまたはJavaScriptによって発行および受信できる、一元化されたイベントシステムが用意されています。 必要に応じて、イベント発行時にデータを渡すことも可能です。 イベントリスナーは、そのデータをローカルデータ型で受け取ります。 + +### EventsOn + +このメソッドは、指定されたイベント名のイベントリスナーを新たに設定します。 `eventName`という名前のイベントが[発行](#EventsEmit)されると、コールバックがトリガーされます。 イベント発行時にデータも付与されていた場合、そのデータはコールバックに渡されます。 このメソッドは、リスナーをキャンセルするための関数を返します。 + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff + +このメソッドは、指定されたイベント名のイベントリスナー設定を解除します。引数の`additionalEventNames`を使用することで、複数のリスナーを一度に解除できます。 + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce + +このメソッドは、指定されたイベント名のイベントリスナーを新たに設定し、一度だけトリガーさせます。 このメソッドは、リスナーをキャンセルするための関数を返します。 + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple + +このメソッドは、指定されたイベント名のイベントリスナーを新たに設定し、最大`counter`回だけトリガーします。 このメソッドは、リスナーをキャンセルするための関数を返します。 + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit + +このメソッドは、指定されたイベントを発行します。 必要に応じて、イベント発行時にデータを渡すこともできます。 このメソッドによって、任意のイベントリスナーをトリガーさせることができます。 + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx new file mode 100644 index 00000000000..01e62ff1af0 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# イントロダクション + +ランタイムは、アプリケーションにユーティリティメソッドを提供するライブラリです。 GoとJavaScriptの両方にランタイムがあり、どちらにもほぼ同じメソッドが提供されています。 + +ユーティリティメソッドには次のようなものがあります: + +- [ウィンドウ](window.mdx) +- [メニュー](menu.mdx) +- [ダイアログ](dialog.mdx) +- [イベント](events.mdx) +- [ブラウザ](browser.mdx) +- [ログ](log.mdx) +- [クリップボード](clipboard.mdx) + +Goのランタイムは、`github.com/wailsapp/wails/v2/pkg/runtime`をインポートすることで利用できます。 このパッケージのすべてのメソッドは、1番目の引数でContextを渡す必要があります。 このContextは、[OnStartup](../options.mdx#onstartup)フック、または[OnDomReady](../options.mdx#ondomready)フックからあらかじめ取得しておいてください。 + +:::info 備考 + +[OnStartup](../options.mdx#onstartup)で提供されるContextは、ウィンドウが別のスレッドで初期化されているため、ランタイムが機能する保証がありません。 起動時にランタイムメソッドを呼び出したい場合は、[OnDomReady](../options.mdx#ondomready)を使用してください。 + +::: + +JavaScriptのランタイムは、`window.runtime`マップを介してフロントエンド上で利用できます。 `dev`モードでは、TypeScript型定義を提供するランタイムパッケージが生成されます。 これらは、フロントエンドディレクトリの`wailsjs`ディレクトリに配置しておく必要があります。 + +### Hide + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +アプリケーションを非表示にします。 + +:::info 備考 + +Macでこのメソッドを使用すると、標準のMacアプリケーションにおけるメニュー項目の`Hide`と同じ方法で、アプリケーションが非表示になります。 これはウィンドウの非表示とは異なりますが、アプリケーションはフォアグラウンドに残ったままになります。 WindowsおよびLinuxでは、`WindowHide`メソッドと同等です。 + +::: + +### Show + +アプリケーションを表示します。 + +:::info 備考 + +Macでこのメソッドを使用すると、アプリケーションがフォアグラウンドに戻ります。 WindowsおよびLinuxでは、`WindowShow`メソッドと同等です。 + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### Quit + +アプリケーションを終了します。 + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### Environment + +現在の環境の詳細情報を取得します。 + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### EnvironmentInfo + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx new file mode 100644 index 00000000000..924f6257ac1 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# ログ + +Wailsでは、GoまたはJavaScriptから呼び出すことのできるロギングメカニズムを用意しています。 一般的なロガーと同じように、ログにはいくつかのログレベルがあります: + +- Trace +- Debug +- Info +- Warning +- Error +- Fatal + +ロガーは、設定されている出力ログレベル以上のログメッセージを出力します。 例えば、出力ログレベルを`Debug`に設定した場合、`Trace`以外のすべてのレベルのメッセージが出力されます。 + +### LogPrint + +指定されたメッセージをRawメッセージとしてロギングします。 + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf + +指定されたメッセージをRawメッセージとしてロギングします。 + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace + +指定されたメッセージを`Trace`ログレベルでロギングします。 + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef + +指定されたメッセージを`Trace`ログレベルでロギングします。 + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug + +指定されたメッセージを`Debug`ログレベルでロギングします。 + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf + +指定されたメッセージを`Debug`ログレベルでロギングします。 + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo + +指定されたメッセージを`Info`ログレベルでロギングします。 + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof + +指定されたメッセージを`Info`ログレベルでロギングします。 + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning + +指定されたメッセージを`Warning`ログレベルでロギングします。 + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf + +指定されたメッセージを`Warning`ログレベルでロギングします。 + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError + +指定されたメッセージを`Error`ログレベルでロギングします。 + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf + +指定されたメッセージを`Error`ログレベルでロギングします。 + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal + +指定されたメッセージを`Fatal`ログレベルでロギングします。 + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf + +指定されたメッセージを`Fatal`ログレベルでロギングします。 + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel + +出力ログレベルを設定します。 JavaScriptでは、数値が次のログレベルに対応しています: + +| 値 | ログレベル | +| - | ------- | +| 1 | Trace | +| 2 | Debug | +| 3 | Info | +| 4 | Warning | +| 5 | Error | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## カスタムロガーの使用 + +カスタムロガーは、アプリケーションオプションの1つである[Logger](../options.mdx#logger)で指定してあげることで、使用することができます。 カスタムロガーを使用する際の唯一の要件は、`github.com/wailsapp/wails/v2/pkg/logger`で定義されている`logger.Logger`インターフェースを、ロガーに実装することです: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx new file mode 100644 index 00000000000..9d38fcd62a7 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# メニュー + +これらは、アプリケーションメニューに関連したメソッドです。 + +:::info JavaScript + +現在、Javascriptランタイムではメニューをサポートしていません。 + +::: + +### MenuSetApplicationMenu + +指定された[menu](../menus.mdx)をアプリケーションメニューとして設定します。 + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu + +`MenuSetApplicationMenu`に渡されたメニューへの変更を検知し、アプリケーションメニューを更新します。 + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx new file mode 100644 index 00000000000..954c407d5ca --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# スクリーン + +これらのメソッドは、現在接続されているスクリーンに関する情報を提供します。 + +### ScreenGetAll + +現在接続されているスクリーンのリストを返します。 + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### スクリーン + +Go構造体: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript型定義: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx new file mode 100644 index 00000000000..ea6dd358d5b --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# ウィンドウ + +アプリケーションウィンドウを制御できるメソッド群です。 + +### WindowSetTitle + +ウィンドウのタイトルバーにテキストを設定します。 + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen + +ウィンドウをフルスクリーンにします。 + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen + +フルスクリーンにする前のウィンドウサイズおよび位置に戻します。 + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen + +ウィンドウがフルスクリーンの場合は、trueを返します。 + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter + +ウィンドウが現在表示されているモニターの中央に、ウィンドウを配置させます。 + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS + +ウィンドウ内で、任意のJSコードを実行します。 + +このメソッドは、ブラウザ上で非同期にコードを実行し、すぐにリターンされます。 スクリプトでエラーが発生した場合、エラーログはブラウザコンソールでのみ確認できます。 + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload + +リロードします。(現在表示されているページをリロード) + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp + +アプリケーションフロントエンドをリロードします。 + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme + +Windowsのみ使用可能。 + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +ウィンドウのテーマをシステムデフォルト(ダーク/ライト) に設定します。 + +### WindowSetLightTheme + +Windowsのみ使用可能。 + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +ウィンドウのテーマをライトに設定します。 + +### WindowSetDarkTheme + +Windowsのみ使用可能。 + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +ウィンドウのテーマをダークに設定します。 + +### WindowShow + +ウィンドウが非表示になっている場合は、表示させます。 + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowHide + +現在表示されているウィンドウを非表示にします。 + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal + +ウィンドウが最小化、最大化、またはフルスクリーンになっていない場合、trueを返します。 + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize + +ウィンドウの幅と高さを設定します。 + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowGetSize + +ウィンドウの幅と高さを取得します。 + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize + +ウィンドウの最小サイズを設定します。 現在のウィンドウサイズが、指定された最小サイズよりも小さい場合、現在のウィンドウサイズは変更されます。 + +サイズを`0,0`に設定すると、サイズの制約が無効化されます。 + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize + +ウィンドウの最大サイズを設定します。 現在のウィンドウサイズが、指定された最大サイズよりも大きい場合、現在のウィンドウサイズは変更されます。 + +サイズを`0,0`に設定すると、サイズの制約が無効化されます。 + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop + +ウィンドウを常に最前面に表示するかを切り替えます。 + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition + +現在ウィンドウが表示されているモニターに対する、相対的なウィンドウ位置を設定します。 + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition + +現在ウィンドウが表示されているモニターに対する、相対的なウィンドウ位置を取得します。 + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximise + +ウィンドウを最大化します。 + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise + +ウィンドウの最大化を解除し、最大化する前のサイズおよび位置に戻します。 + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised + +ウィンドウが最大化している場合はtrueを返します。 + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise + +最大化の状態を切り替えます。 + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise + +ウィンドウを最小化します。 + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise + +ウィンドウの最小化を解除し、最小化する前のサイズおよび位置に戻します。 + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised + +ウィンドウが最小化している場合はtrueを返します。 + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour + +ウィンドウの背景色をRGBAカラー定義で設定します。 この色は、すべての透過ピクセルに対して表示されます。 + +R、G、B、Aの有効な値の範囲は0~255です。 + +:::info Windows + +Windowsの場合、0または255のアルファ値(A) のみがサポートされています。 0以外の値を指定すると、すべて255とみなされます。 + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +ネイティブな印刷ダイアログを開きます。 + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## TypeScript型定義 + +### Position + +```ts +interface Position { + x: number; + y: number; +} +``` + +### Size + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/versioned_docs/version-v2.5.0/tutorials/_category_.json b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/_category_.json similarity index 100% rename from website/versioned_docs/version-v2.5.0/tutorials/_category_.json rename to website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/_category_.json diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..8422c83f715 --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# ドッグAPI + +```mdx-code-block +
+ +
+
+``` + +:::note + +このチュートリアルは[@tatadan](https://twitter.com/tatadan)より提供されたもので、[Wails Examples Repository](https://github.com/tataDan/wails-v2-examples)の一部になります。 + +::: + +このチュートリアルでは、Webから犬の写真を取得して表示するためのアプリケーションを開発していきます。 + +### プロジェクトの作成 + +アプリケーションを作成していきましょう。 ターミナルで次のように入力します: `wails init -n dogs-api -t svelte` + +注意: IDEサポートが欲しい場合は、コマンドの末尾に、`-ide vscode`オプションまたは`-ide goland`オプションを付与してください。 + +では、`cd dogs-api`で移動し、プロジェクトファイルの編集を始めましょう。 + +### 未使用コードの削除 + +まず、使用しないと分かっているいくつかの要素を削除していきます: + +- `app.go`を開き、次の行を削除します: + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- `frontend/src/App.svelte`を開き、すべての行を削除します。 +- `frontend/src/assets/images/logo-universal.png`ファイルを削除します。 + +### アプリケーションの作成 + +では、新しいGoコードを書いていきましょう。 + +関数を定義する前に、`app.go`に下記の構造体の宣言を追加します: + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +下記の関数を`app.go`に追加します(おそらく既存の関数定義の後ろ側に追加することになります): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +`app.go`の`import`セクションを次のように変更します: + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +`frontend/src/App.svelte`に次の行を追加します: + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### アプリケーションのテスト + +バインディングを生成してアプリケーションをテストするには、`wails dev`コマンドを実行してください。 + +### アプリケーションのコンパイル + +アプリケーションを本番用の単一バイナリにコンパイルするには、`wails build`コマンドを実行してください。 diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx new file mode 100644 index 00000000000..55c52342d7b --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# Hello World + +このチュートリアルの目的は、Wailsを使用して、最も基礎的なアプリケーションを起動および実行することです。 あなたは、次のことができるようになります: + +- 新しいWailsアプリケーションを作成する +- アプリケーションをビルドする +- アプリケーションを実行する + +:::note + +このチュートリアルでは、Windowsをターゲットプラットフォームとして使用します。 出力される内容は、オペレーティングシステムによって多少異なります。 + +::: + +## 新しいWailsアプリケーションを作成する + +デフォルトのバニラJSテンプレートを使用して、新しいWailsアプリケーションを作成するには、次のコマンドを実行します: + +```bash +wails init -n helloworld +``` + +すると、次のように表示されます: + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +これにより、カレントディレクトリに`helloworld`という新しいディレクトリが作成されます。 このディレクトリには、いくつかのファイルがあります: + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## アプリケーションをビルドする + +アプリケーションをビルドするには、新しくできた`helloworld`プロジェクトディレクトリに移動し、次のコマンドを実行します: + +```bash +wails build +``` + +すると、次のように表示されます: + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +これにより、アプリケーションがコンパイルされ、`build/bin`ディレクトリに保存されます。 + +## アプリケーションを実行する + +Windowsのエクスプローラで`build/bin`ディレクトリを閲覧すると、プロジェクトのバイナリファイルが存在するはずです: + +```mdx-code-block +
+ +
+
+``` + +`helloworld.exe`ファイルをダブルクリックするだけで起動できます。 + +Macの場合、Wailsは`helloworld.app`ファイルを生成します。このファイルをダブルクリックすることで起動できます。 + +Linuxの場合、`build/bin`ディレクトリ内の`./helloworld`を呼び出すことでアプリケーションを起動できます。 + +アプリケーションが期待したとおりに動作することでしょう: + +```mdx-code-block +
+ +
+
+``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx new file mode 100644 index 00000000000..e36a6edfe8c --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# Links + +This page serves as a list for community related links. Please submit a PR (click `Edit this page` at the bottom) to submit links. + +## Awesome Wails + +The [definitive list](https://github.com/wailsapp/awesome-wails) of links related to Wails. + +## Support Channels + +- [Wails Discord Server](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [v2 Beta Discussion Board](https://github.com/wailsapp/wails/discussions/828) + +## Social Media + +- [Twitter](https://twitter.com/wailsapp) +- [Wails Chinese Community QQ Group](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - Group number: 1067173054 + +## Other Tutorials and Articles + +- [Building of Bulletin Board](https://blog.customct.com/building-bulletin-board) diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/bulletinboard.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/bulletinboard.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/emailit.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/emailit.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..4c8cac235a3 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app)는 간단하고 사용하기 쉬운 PGP 암호화 도구로 모든 키들을 관리합니다. 암호화는 간단해야 합니다. 그리고 Wails로 개발되었습니다.** + +PGP를 사용한 메시지 암호화는 업계 표준입니다. 모두 개인 키와 공개 키를 가지고 있습니다. 귀하의 개인 키는 귀하만 메시지를 읽을 수 있도록 비공개로 유지되어야 합니다. 귀하의 공개 키는 귀하에게 암호화된 비밀 메시지를 보내려는 모든 사람에게 배포됩니다. 키 관리, 메시지 암호화 및 메시지 해독은 매끄러운 경험이어야 합니다. EncryptEasy는 모든 것을 쉽게 만드는 것입니다. EncryptEasy는 모든 것을 쉽게 만듭니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx new file mode 100644 index 00000000000..0881bfe9e1e --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# FileHound Export Utility + +```mdx-code-block +

+ +
+

+``` + +[FileHound Export Utility](https://www.filehound.co.uk/) FileHound는 안전한 파일 보존, 비즈니스 프로세스 자동화 및 SmartCapture 기능을 위해 만들어진 클라우드 문서 관리 플랫폼입니다. + +FileHound Export Utility를 사용하면 FileHound 관리자가 대체 백업 및 복구 목적으로 보안 문서 및 데이터 추출 작업을 실행할 수 있습니다. 이 응용 프로그램은 선택한 필터를 기반으로 FileHound에 저장된 모든 문서 및/또는 메타 데이터를 다운로드합니다. 메타데이터는 JSON 및 XML 형식으로 내보낼 수 있습니다. + +백엔드의 구성은 다음과 같습니다: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +사용된 프론트엔드: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..87e5837d32f --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter) is a simple and efficient http API testing client tool. Based on Wails, Go and sveltejs. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..d31eac0d5f8 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Minecraft Updater + +```mdx-code-block +

+ +
+

+``` + +[Minecraft Updater](https://github.com/Gurkengewuerz/MinecraftModUpdater)는 사용자 기반의 Minecraft 모드를 업데이트하고 동기화하는 유틸리티 도구입니다. Wails2와 [antd](https://ant.design/)를 프런트엔드 프레임워크로 사용한 React를 사용하여 구축되었습니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..9c98dc86473 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Modal File Manager + +```mdx-code-block +

+ +
+

+``` + +[Modal File Manager](https://github.com/raguay/ModalFileManager) 는 웹 기술을 사용하는 이중 창 모양의 파일 관리자입니다. 나의 원래 디자인은 NW.js를 기반으로 했으며 [여기](https://github.com/raguay/ModalFileManager-NWjs)에서 찾을 수 있습니다. 이 버전은 동일한 Svelte 기반 프론트엔드 코드를 사용하지만(그러나 NW.js에서 출발한 이후 크게 수정됨) 백엔드는 [Wails 2](https://wails.io/) 구현입니다. By using this implementation, I no longer use command line `rm`, `cp`, etc. commands, but a git install has to be on the system to download themes and extensions. Go를 사용하여 코딩되었으며, 이전 버전보다 훨씬 빠르게 실행됩니다. + +이 파일 관리자는 Vim과 동일한 원칙, 즉 상태 제어 키보드 동작을 기반으로 설계되었습니다. 상태의 수는 고정이 안되어 있지만, 그러나 매우 프로그래머블합니다. 그러므로, 키보드 설정들의 무한한 수는 생성 될 수 있고 사용될 수 있습니다. 이것은 다른 파일 매니저들과 다른 주요점입니다. There are themes and extensions available to download from GitHub. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..57eda38f981 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molly Wallet](https://github.com/grvlle/constellation_wallet/) 은 Constellation Network의 공식 $DAG 지갑입니다. 이를 통해 사용자는 $DAG 트랜잭션 생성에 국한되지 않고 다양한 방식으로 Hypergraph Network와 상호 작용할 수 있습니다. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/october.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/october.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx new file mode 100644 index 00000000000..6bc28218bc5 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# 옵티머스(Optimus) + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus) 데스크탑 이미지 최적화 애플리케이션입니다. WebP, JPEG, PNG 이미지 포맷 형식간의 변환 및 압축을 지원합니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx new file mode 100644 index 00000000000..f174364c13d --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# 포트폴(Portfall) + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - 모든 클러스터 UI에 쉽게 액세스할 수 있는 데스크탑 k8s 포트 포워딩 포털입니다. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/restic-browser.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/restic-browser.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/riftshare.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/riftshare.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/scriptbar.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/scriptbar.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/surge.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/surge.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/wally.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/wally.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx new file mode 100644 index 00000000000..950dc3f3db1 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# Minecraft launcher for WarMine + +```mdx-code-block +

+ + +
+

+``` + +[Minecraft launcher for WarMine](https://warmine.ru/) is a Wails application, that allows you to easily join modded game servers and manage your game accounts. + +The Launcher downloads the game files, checks their integrity and launches the game with a wide range of customization options for the launch arguments from the backend. + +Frontend is written in Svelte, whole launcher fits in 9MB and supports Windows 7-11. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx new file mode 100644 index 00000000000..b2dc9302a9c --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# 웜뱃(Wombat) + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat) 은 크로스 플랫폼인 gRPC 클라이언트 입니다. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/ytd.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/ytd.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx new file mode 100644 index 00000000000..6ecbac37ee8 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# Templates + +This page serves as a list for community supported templates. Please submit a PR (click `Edit this page` at the bottom) to include your templates. To build your own template, please see the [Templates](../guides/templates.mdx) guide. + +To use these templates, run `wails init -n "Your Project Name" -t [the link below[@version]]` + +If there is no version suffix, the main branch code template is used by default. If there is a version suffix, the code template corresponding to the tag of this version is used. + +Example: `wails init -n "Your Project Name" -t https://github.com/misitebao/wails-template-vue` + +:::warning Attention + +**The Wails project does not maintain, is not responsible nor liable for 3rd party templates!** + +If you are unsure about a template, inspect `package.json` and `wails.json` for what scripts are run and what packages are installed. + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - Wails template based on Vue ecology (Integrated TypeScript, Dark theme, Internationalization, Single page routing, TailwindCSS) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Vue 3 TypeScript with Vite (and instructions to add features) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vue 3 TypeScript with Vite, Vuex, Vue Router, Sass, and ESLint + Prettier +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - A template using JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - A template using TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, Composition API with <script setup>) +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Wails template based on Naive UI (A Vue 3 Component Library) + +## Angular + +- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - Angular 15+ action packed & ready to roll to production. +- [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - Angular with TypeScript, Sass, Hot-Reload, Code-Splitting and i18n + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - A template using reactjs +- [wails-react-template](https://github.com/flin7/wails-react-template) - A minimal template for React that supports live development +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - A template using Next.js and TypeScript +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - A template for React + TypeScript + Vite + TailwindCSS +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - A template using Svelte +- [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - A template using Svelte and Vite +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - A template using Svelte and Vite with TailwindCSS v3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - A template using SvelteKit + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - A template using Solid + Ts + Vite +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - A template using Solid + Js + Vite + +## Elm + +- [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + +## Pure JavaScript (Vanilla) + +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx new file mode 100644 index 00000000000..57d509dfef7 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# 프로젝트 컴파일 + +프로젝트 디렉토리에서 `wails build` 명령을 실행합니다. 이것은 프로젝트를 컴파일하고 `build/bin` 디렉토리에 프로덕션-준비(production-ready) 바이너리를 저장합니다. + +바이너리를 실행하면 기본 애플리케이션이 표시되어야 합니다: + +```mdx-code-block +
+ +
+
+``` + +컴파일 옵션에 대한 자세한 내용은 [CLI Reference](../reference/cli.mdx#build)를 참조하세요. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx new file mode 100644 index 00000000000..5d0d9185cf4 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# 애플리케이션 개발하기 + +프로젝트 디렉토리에서 `wails dev` 명령을 실행하여 개발 모드에서 애플리케이션을 실행할 수 있습니다. 이렇게 하면 다음 작업이 수행됩니다. + +- 애플리케이션 빌드 및 실행 +- JavaScript에서 호출할 수 있도록 Go 코드를 프론트엔드에 바인딩 +- [Vite](https://vitejs.dev/)의 기능을 사용하여 Go 파일의 수정 사항을 감시하고 변경 시 다시 빌드/재실행 합니다. +- 브라우저를 통해 애플리케이션을 제공할 [Webserver](http://localhost:34115) 를 설정합니다. 이렇게 하면 브라우저 확장을 사용할 수 있습니다. 콘솔에서 Go 코드를 호출할 수도 있습니다. + +시작하려면, 프로젝트 디렉토리 내에서 `wails dev` 명령을 실행하세요. 더많은 정보는 [here](../reference/cli.mdx#dev) 에서 찾을 수 있습니다. + +출시 예정: 튜토리얼 diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..45f2aa8a57e --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# 프로젝트 생성하기 + +## 프로젝트 생성 + +이제 CLI가 설치되었습니다, `wails init` 명령을 사용하여 새로운 프로젝트를 생성할 수 있습니다. + +원하는 프레임워크를 선택하세요: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Generate a Svelte project using JavaScript with:

+ + wails init -n myproject -t svelte + +If you would rather use TypeScript:
+ + wails init -n myproject -t svelte-ts + +
+ + Generate a React project using JavaScript with:

+ + wails init -n myproject -t react + +If you would rather use TypeScript:
+ + wails init -n myproject -t react-ts + +
+ + Generate a Vue project using JavaScript with:

+ + wails init -n myproject -t vue + +If you would rather use TypeScript:
+ + wails init -n myproject -t vue-ts + +
+ + Generate a Preact project using JavaScript with:

+ + wails init -n myproject -t preact + +If you would rather use TypeScript:
+ + wails init -n myproject -t preact-ts + +
+ + Generate a Lit project using JavaScript with:

+ + wails init -n myproject -t lit + +If you would rather use TypeScript:
+ + wails init -n myproject -t lit-ts + +
+ + Generate a Vanilla project using JavaScript with:

+ + wails init -n myproject -t vanilla + +If you would rather use TypeScript:
+ + wails init -n myproject -t vanilla-ts + +
+
+``` + +
+ +다양한 기능과 프레임워크를 제공하는 [커뮤니티 템플릿](../community/templates.mdx)도 있습니다. + +사용 가능한 다른 옵션을 보려면 `wails init -help` 명령을 실행할 수 있습니다. 자세한 내용은 [CLI Reference](../reference/cli.mdx#init)에서 찾을 수 있습니다. + +## 프로젝트 레이아웃 + +Wails 프로젝트는 다음 레이아웃을 따릅니다: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### 프로젝트 구조 개요 + +- `/main.go` - 메인 애플리케이션 +- `/frontend/` - Frontend 프로젝트 파일들 +- `/build/` - 프로젝트 빌드 디렉토리 +- `/build/appicon.png` - 애플리케이션 아이콘 +- `/build/darwin/` - Mac 특정 프로젝트 파일 +- `/build/windows/` - Windows 특정 프로젝트 파일 +- `/wails.json` - 프로젝트 구성 파일 +- `/go.mod` - Go 모듈 파일 +- `/go.sum` - Go 모듈 체크섬 파일 + +`frontend` 프론트엔드 디렉토리에는 Wails와 관련된 것이 없으며 선택한 모든 프론트엔드 프로젝트가 될 수 있습니다. + +`build` 디렉토리는 빌드가 진행되는 동안 사용됩니다. 이 파일들을 업데이트 하여 빌드를 커스텀마이징할 수 있습니다. 만약 빌드 디렉토리에서 파일이 제거되면 기본 버전이 재생성됩니다. + +`go.mod`의 기본 모듈 이름은 "changeme" 입니다. 이것을 더 적절한 것으로 변경해야 합니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx new file mode 100644 index 00000000000..1d911d11802 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx @@ -0,0 +1,89 @@ +--- +sidebar_position: 1 +--- + +# 설치하기 + +## 지원되는 플랫폼 + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## 의존성 + +Wails는 설치 전에 아래와 같은 몇 가지 공통적인 의존성이 필요합니다. + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +[Go 다운로드 페이지](https://go.dev/dl/)에서 Go를 다운로드 합니다. + +공식 [Go 설치 지침](https://go.dev/doc/install)에 따라 진행하세요. 또한 `PATH` 환경 변수에 `~/go/bin` 디렉토리에 대한 경로도 포함되어 있는지 확인해야 합니다. 터미널을 다시 시작하고 아래 내용을 확인하세요. + +- Go가 정상적으로 설치되었는지 확인: `go version` +- PATH 변수에 "~/go/bin" 확인: `echo $PATH | grep go/bin` + +### NPM + +[Node 다운로드 페이지](https://nodejs.org/en/download/)에서 NPM을 다운로드 합니다. 우리는 일반적으로 최신 버전에서 테스트를 진행하기 때문에 최신 버전 사용을 권장합니다. + +정상적으로 설치된 것을 확인하기 위해 `npm --version`을 실행합니다. + +## 플랫폼에 따른 의존성 + +플랫폼별 의존 항목도 설치해야 합니다: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wails 는 xcode line tools 설치를 필요로 합니다. xcode-select --install. 를 실행하여 수행할 수 있습니다. + + + Wails 는 WebView2 런타임 설치를 필요로 합니다. 몇몇 Windows는 이미 설치되어 있습니다. wails doctor 명령을 사용하여 확인할 수 있습니다. + + + Linux requires the standard gcc build tools plus libgtk3 and libwebkit. 다양한 배포판에 대한 수 많은 명령을 나열하는 대신 Wail는 특정 배포판에 대한 설치 명령이 무엇인지 결정할 수 있습니다. wails doctor 명령을 실행하면 설치 후 종속성을 설치하는 방법을 보여줍니다. 만약 배포판/패키지 매니저가 지원하지 않는다면, 를 통해 문의하세요. 리눅스 배포판 추가 가이드 + + +``` + +## 선택 설치 + +- [UPX](https://upx.github.io/) 는 애플리케이션 압축을 위함. +- [NSIS](https://wails.io/docs/guides/windows-installer/) 는 윈도우 인스톨러를 생성하기 위함. + +## Wails 설치 + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` 명령을 실행하여 Wails CLI를 설치합니다. + +참고: 다음과 비슷한 오류가 발생하는 경우: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +Go 1.18 이상의 버전이 설치되어 있는지 확인하세요: +```shell +go version +``` + +## 시스템 점검 + +`wails doctor`를 실행하면 의존성이 올바르게 설치되어 있는지 점검할 수 있습니다. 문제가 있는 의존성에 대해서는 문제 해결을 위한 도움을 줄 수 있습니다. + +## `wails` 명령이 사라졌다고 나오나요? + +시스템에서 `wails` 명령이 누락되었다고 뜨는 경우 Go 설치 가이드를 바르게 따랐는지 확인하세요. 일반적으로, 사용자의 홈 디렉토리에 있는 `go/bin` 디렉토리가 `PATH` 환경 변수에 없음을 의미합니다. 또한, 설치 프로그램이 수행한 환경 변경 사항이 명령 프롬프트에 반영되도록 일반적으로 열려 있는 모든 명령 프롬프트를 닫았다가 다시 열어야합니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx new file mode 100644 index 00000000000..1b21f9daa7a --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +Wails에는 Angular 템플릿이 없지만 Wails와 함께 Angular를 사용할 수 있습니다. + +## 개발자 모드 + +Angular에서 작동하는 개발 모드를 사용하려면 `wails.json`에 다음을 추가해야 합니다. + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx new file mode 100644 index 00000000000..e4251eba409 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx @@ -0,0 +1,215 @@ +# 애플리케이션 개발 + +Wails로 애플리케이션을 개발하기 위한 엄격한 규칙은 없으나 몇 가지의 기본 가이드가 있습니다. + +## 애플리케이션 셋업 + +기본 템플릿에서 사용되는 패턴은 `main.go`가 애플리케이션 구성 및 실행에 사용되는 반면 `app.go`는 애플리케이션 로직 정의에 사용된다는 것입니다. + +`app.go` 파일은 기본 애플리케이션에 hook 역할을 하는 2개의 메소드가 있는 구조체를 정의합니다. + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- 시작 메서드는 Wails가 필요한 리소스를 할당하는 즉시 호출되며 리소스 생성, 이벤트 리스너 설정 및 시작 시 응용 프로그램에 필요한 모든 것을 설정하기에 좋은 위치입니다. 일반적으로 구조체 필드에 저장되는 `context.Context`가 제공됩니다. 이 컨텍스트는 [런타임](../reference/runtime/intro.mdx)을 호출하는 데 필요합니다. 이 메서드가 오류를 반환하면 응용 프로그램이 종료됩니다. 개발 모드에서는 오류가 콘솔에 출력됩니다. + +- Shutdown 메소드는 shutdown 프로세스가 끝날 때 바로 Wails에 의해 호출됩니다. 이것은 메모리 할당을 해제하고 모든 shutodown 작업을 수행하기에 좋은 위치입니다. + +`main.go` 파일은 일반적으로 애플리케이션 구성을 허용하는 `wails.Run()`에 대한 단일 호출로 구성됩니다. 템플릿에서 사용하는 패턴은 `wails.Run()`을 호출하기 전에 `app.go`에서 정의한 구조체의 인스턴스가 생성되어 변수에 저장된다는 것입니다. 이것은 `app`이라고 합니다. 이 configuration 은 콜백을 추가하는 위치입니다: + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +애플리케이션 lifecycle hook에 대한 자세한 내용은 [여기](../howdoesitwork.mdx#application-lifecycle-callbacks)에서 확인할 수 있습니다. + +## 바인딩 메소드 + +프론트엔드에서 Go 메서드를 호출할 가능성이 높습니다. 이는 일반적으로 `app.go`에서 이미 정의된 구조체에 public method를 추가하여 수행됩니다. + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +메인 어플리케이션 구성에서, `Bind` 키는 Wails에 무엇을 바인딩할지 알릴 수 있는 부분입니다. + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +이렇게 하면 `App` 구조체의 모든 공개 메서드가 바인딩됩니다(시작 및 종료 메서드는 바인딩되지 않음). + +### 여러 구조체를 바인딩할 때의 컨텍스트 처리 + +여러 구조체에 대한 메서드를 바인딩하고 싶지만 런타임을 사용할 수 있도록 각 구조체가 컨텍스트에 대한 참조를 유지하기를 원하는 경우, 함수에서 좋은 패턴은 `OnStartup` 메서드에서 구조체 인스턴스로 컨텍스트를 전달하는 것입니다. + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +바인딩에 대한 더 많은 정보는 [여기](../howdoesitwork.mdx#method-binding)에서 확인할 수 있습니다. + +## Application Menu + +Wails는 당신의 애플리케이션에 메뉴 추가를 지원합니다. 이것은 [Menu](../reference/menus.mdx#menu) 구조체를 애플리케이션 구성에 전달하여 수행됩니다. Menu를 반환하는 메서드를 사용하는 것이 일반적이며 lifecycle hooks에 사용되는 `App` 구조체의 메서드인 경우에는 훨씬 더 일반적입니다. + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## Assets + +Wails v2가 assets을 처리하는 방식의 장점은 그렇지 않다는 것입니다! Wails에 제공해야 하는 유일한 것은 `embed.FS`입니다. 당신이 어떻게 얻어내는가는 전적으로 당신에게 달렸습니다. 바닐라 템플릿과 같은 바닐라 html/css/js를 사용할 수 있습니다. 복잡한 빌드 시스템을 가질 수 있으나, 중요하지 않습니다. + +`wails build`가 실행되면 프로젝트 최상위에서 `wails.json` 프로젝트 파일을 확인합니다. 프로젝트 파일에는 2개의 키가 있습니다. + +- "frontend:install" +- "frontend:build" + +첫 번째는 노드 모듈을 설치하기 위해 `frontend` 디렉토리에서 실행됩니다. 두 번째는 프런트엔드 프로젝트를 빌드하기 위해 `frontend` 디렉토리에서 실행됩니다. + +이 2개의 키가 주어지지 않으면 Wails는 프런트엔드에서 아무 작업도 수행하지 않습니다. 그것은 `embed.FS`에서만 기대합니다. + +### AssetsHandler + +Wails v2 앱은 선택적으로 `options.App`에서 `http.Handler`를 정의할 수 있습니다. 즉석에서 파일을 생성하거나 POST/PUT 요청을 처리합니다. GET 요청은 항상 `assets` FS에서 먼저 처리됩니다. FS가 요청된 파일을 찾지 못하면 요청은 제공을 위해 `http.Handler`로 전달됩니다. GET 이외의 모든 요청은 지정된 경우 `AssetsHandler`에서 직접 처리됩니다. `Assets` 옵션으로 `nil`을 지정하여 `AssetsHandler`만 사용할 수도 있습니다. + +## Built in Dev Server + +`wails dev`를 실행하면 내장된 dev 서버가 시작되어 프로젝트 디렉토리에서 file watcher를 시작합니다. 기본적으로 파일이 변경되면 wails는 응용 프로그램 파일인지 확인합니다(기본값: `.go`, `-e` 플래그로 구성 가능). 그렇다면 응용 프로그램을 다시 빌드하고 다시 시작합니다. 변경된 파일이 assets에 있는 경우 짧은 시간 후에 다시 로드를 발행합니다. + +개발 서버는 "debouncing"이라는 기술을 사용합니다. 즉, 짧은 시간에 여러 파일이 변경될 수 있으므로 바로 다시 로드되지 않습니다. 트리거가 발생하면 다시 로드하기 전에 설정된 시간 동안 기다립니다. 다른 트리거가 발생하면 다시 대기 시간으로 재설정됩니다. 기본 값은 `100ms` 입니다. 이 값이 프로젝트에서 작동하지 않는 경우 `-debounce` 플래그를 사용하여 구성할 수 있습니다. 사용하는 경우 이 값은 프로젝트 구성에 저장되고 기본값이 됩니다. + +## External Dev Server + +일부 프레임워크는 자체 라이브 리로딩 서버와 함께 제공되지만 Wails Go 바인딩을 활용할 수 없습니다. 이 시나리오에서는 Wails가 감시할 빌드 디렉토리에 프로젝트를 다시 빌드하는 watcher script를 실행하는 것이 가장 좋습니다. 예를 보려면 [rollup](https://rollupjs.org/guide/en/)을 사용하는 기본 svelte 템플릿을 참조하세요. + +### Create React App + +The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: + +```html + + + + + +``` + +This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. + +## Go Module + +기본 Wails 템플릿은 "changeme"라는 모듈 이름이 포함된 `go.mod` 파일을 생성합니다. 프로젝트 생성 후에 이 파일의 이름을 변경해야 합니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..989202e27a6 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# Dynamic Assets + +If you want to load or generate assets for your frontend dynamically, you can achieve that using the [AssetsHandler](../reference/options#assetshandler) option. The AssetsHandler is a generic `http.Handler` which will be called for any non GET request on the assets server and for GET requests which can not be served from the bundled assets because the file is not found. + +By installing a custom AssetsHandler, you can serve your own assets using a custom asset server. + +## Example + +In our example project, we will create a simple assets handler which will load files off disk: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +When we run the application in dev mode using `wails dev`, we will see the following output: + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +As you can see, the assets handler is called when the default assets server is unable to serve the `favicon.ico` file. + +If you right click the main application and select "inspect" to bring up the devtools, you can test this feature out by typing the following into the console: + +``` +let response = await fetch('does-not-exist.txt'); +``` + +This will generate an error in the devtools. We can see that the error is what we expect, returned by our custom assets handler: + +```mdx-code-block +

+ +

+``` + +However, if we request `go.mod`, we will see the following output: + +```mdx-code-block +

+ +

+``` + +This technique can be used to load images directly into the page. If we updated our default vanilla template and replaced the logo image: + +```html + +``` + +with: + +```html + +``` + +Then we would see the following: + +```mdx-code-block +

+ +

+``` + +:::warning + +Exposing your filesystem in this way is a security risk. It is recommended that you properly manage access to your filesystem. + +::: diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..167955b12d7 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,244 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx new file mode 100644 index 00000000000..3845736f427 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx @@ -0,0 +1,87 @@ +# Frameless Applications + +Wails supports application that have no frames. This can be achieved by using the [frameless](../reference/options.mdx#frameless) field in [Application Options](../reference/options.mdx#application-options). + +Wails offers a simple solution for dragging the window: Any HTML element that has the CSS style `--wails-draggable:drag` will act as a "drag handle". This property applies to all child elements. If you need to indicate that a nested element should not drag, then use the attribute '--wails-draggable:no-drag' on that element. + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +For some projects, using a CSS variable may not be possible due to dynamic styling. In this case, you can use the `CSSDragProperty` and `CSSDragValue` application options to define a property and value that will be used to indicate draggable regions: + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info Fullscreen + +If you allow your application to go fullscreen, this drag functionality will be disabled. + +::: diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx new file mode 100644 index 00000000000..ac087ee4514 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx @@ -0,0 +1,72 @@ +# Frontend + +## Script Injection + +When Wails serves your `index.html`, by default, it will inject 2 script entries into the `` tag to load `/wails/ipc.js` and `/wails/runtime.js`. These files install the bindings and runtime respectively. + +The code below shows where these are injected by default: + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### Overriding Default Script Injection + +To provide more flexibility to developers, there is a meta tag that may be used to customise this behaviour: + +```html + +``` + +The options are as follows: + +| Value | Description | +| ------------------- | ------------------------------------------------ | +| noautoinjectruntime | Disable the autoinjection of `/wails/runtime.js` | +| noautoinjectipc | Disable the autoinjection of `/wails/ipc.js` | +| noautoinject | Disable all autoinjection of scripts | + +Multiple options may be used provided they are comma seperated. + +This code is perfectly valid and operates the same as the autoinjection version: + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx new file mode 100644 index 00000000000..f742822832a --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx @@ -0,0 +1,127 @@ +# IDEs + +Wails aims to provide a great development experience. To that aim, we now support generating IDE specific configuration to provide smoother project setup. + +Currently, we support [Visual Studio Code](https://code.visualstudio.com/) but aim to support other IDEs such as Goland. + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +When generating a project using the `-ide vscode` flags, IDE files will be created alongside the other project files. These files are placed into the `.vscode` directory and provide the correct configuration for debugging your application. + +The 2 files generated are `tasks.json` and `launch.json`. Below are the files generated for the default vanilla project: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### Configuring the install and build steps + +The `tasks.json` file is simple for the default project as there is no `npm install` or `npm run build` step needed. For projects that have a frontend build step, such as the svelte template, we would need to edit `tasks.json` to add the install and build steps: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +:::info Future Enhancement + +In the future, we hope to generate a `tasks.json` that includes the install and build steps automatically. + +::: diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..8b25c15754a --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Linux Distro Support + +## Overview + +Wails offers Linux support but providing installation instructions for all available distributions is an impossible task. Instead, Wails tries to determine if the packages you need to develop applications are available via your system's package manager. Currently, we support the following package managers: + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## Adding package names + +There may be circumstances where your distro uses one of the supported package managers but the package name is different. For example, you may use an Ubuntu derivative, but the package name for gtk may be different. Wails attempts to find the correct package by iterating through a list of package names. The list of packages are stored in the packagemanager specific file in the `v2/internal/system/packagemanager` directory. In our example, this would be `v2/internal/system/packagemanager/apt.go`. + +In this file, the list of packages are defined by the `Packages()` method: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +Let's assume that in our linux distro, `libgtk-3` is packaged under the name `lib-gtk3-dev`. We could add support for this by adding the following line: + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## Adding new package managers + +To add a new package manager, perform the following steps: + +- Create a new file in `v2/internal/system/packagemanager` called `.go`, where `` is the name of the package manager. +- Define a struct that conforms to the package manager interface defined in `pm.go`: + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()` should return the name of the package manager +- `Packages()` should return a `packagemap`, that provides candidate filenames for dependencies +- `PackageInstalled()` should return `true` if the given package is installed +- `PackageAvailable()` should return `true` if the given package is not installed but available for installation +- `InstallCommand()` should return the exact command to install the given package name + +Take a look at the other package managers code to get an idea how this works. + +:::info Remember + +If you add support for a new package manager, don't forget to also update this page! + +::: diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx new file mode 100644 index 00000000000..229c282bf55 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +This page has miscellaneous guides related to developing Wails applications for Linux. + +## Video tag doesn't fire "ended" event + +When using a video tag, the "ended" event is not fired when the video is finished playing. This is a bug in WebkitGTK, however you can use the following workaround to fix it: + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +Source: [Lyimmi](https://github.com/Lyimmi) on the [discussions board](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx new file mode 100644 index 00000000000..4ba1f34c37d --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx @@ -0,0 +1,55 @@ +# Local Development + +## Overview + +Wails is in constant development and new releases are regularly "tagged". This usually happens when all the newer code on `master` has been tested and confirmed working. If you need a bugfix or feature that has not yet made it to a release, it's possible to use the latest "bleeding edge" version using the following steps: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +NOTE: The directory that you cloned the project into will now be called "clonedir". + +The Wails CLI will now be at the very latest version. + +### Updating your project + +To update projects to use the latest version of the Wails library, update the project's `go.mod` and ensure the following line is at the bottom of the file: + +`replace github.com/wailsapp/wails/v2 => ` + +Example: + +On Windows: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +On 'nix: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +To revert to a stable version, run: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## Testing a Branch + +If you want to test a branch, follow the instructions above, but ensure you switch the branch you want to test before installing: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +Make sure you [update your project](#updating-your-project) as described above. + +## Testing a PR + +If you want to test a PR, follow the instructions above, but ensure you fetch the PR and switch the branch before installing. Please replace `[IDofThePR]` with the ID of the PR shown on github.com: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +Make sure you [update your project](#updating-your-project) as described above. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx new file mode 100644 index 00000000000..961595711c7 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Mac App Store Guide + +This page gives a brief overview of how to submit your Wails App to the Mac App Store. + +## Prerequisites + +- You will need to have an Apple Developer account. Please find more information on the [Apple Developer Program](https://developer.apple.com/support/compare-memberships/) site +- You will need to have your Certificates, Identifiers, and App created on the developer portal. More on this below +- Xcode command line tools will need to be installed on your local machine + +#### Create Certificates and Identifiers + +1. Go to your [Apple Developer Account](https://developer.apple.com/account/) +2. Under `Certificates, Identifiers & Profiles`, click `Identifiers` and Register a New App ID. Use the format (com.example.app) +3. Under the same page click `Certificates` and generate new Certificates for Mac App Store Distribution. Download them and import the certificates into Keychain on your local machine. + +#### Create App Submission + +1. Go to the [App Store Connect Site](https://appstoreconnect.apple.com/apps) +2. Register a new application and link the bundle ID that you created in the previous step +3. Populate your app with the correct screen shots, descriptions, etc. as required by Apple +4. Create a new version of your app + +#### Create Provisioning Profile +1. Go to the [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) page +2. Add a new provisioning profile for Mac App Store Distribution +3. Set the Profile Type as Mac and select the App ID for the application created above +4. Select the Mac App Distribution certificate +5. Name the Provisioning Profile embedded and download the created profile. + +## Mac App Store Process + +#### Enable Apple's App Sandbox + +Apps submitted to the Mac App Store must run under Apple's [App Sandbox](https://developer.apple.com/app-sandboxing/). You must create an `entitlements.plist` file for this to work. The recommendation is to create this file under this path `{PROJECT_DIR}/build/darwin/entitlements.plist`. + +**Example Entitlements File** + +This is an example entitlements file from the [RiftShare](https://github.com/achhabra2/riftshare) app. For reference please put in the entitlements your app requires. Refer to [this site](https://developer.apple.com/documentation/bundleresources/entitlements) for more information. You will need to replace the Team ID and Application Name with the ones you registered above. + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +**Add the Embedded Provisioning Profile** The Provisioning Profile created above needs to be added to the root of the applicaton. It needs to be named embedded.provisionprofile. + +#### Build and Sign the App Package + +The following is an example script for building and signing your app for Mac App Store submission. It assumes you are running the script from your root project directory. + +Note the certificates for signing the app and signing the installer are different. Please make sure both are imported into Keychain. Find the strings in Keychain and insert them below. Populate your certificate names, and app name below. Running the following script will generate a signed `app.pkg` file in the root directory of your app. + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### Upload App Bundle + +You will need to upload the generated package file and associate it to your Application before you will be able to submit it for review. + +1. Download the [Transporter App](https://apps.apple.com/us/app/transporter/id1450874784) from the Mac App Store +2. Open it and sign in with your Apple ID +3. Click the + sign and select the `APP_NAME.pkg` file that you generated in the previous step. Upload it +4. Go back to the [App Store Connect](https://appstoreconnect.apple.com/apps) site and navigate back into your app submission. Select the version that you are ready to make available on the App Store. Under `Build` select the package that you uploaded via Transporter. + +That's it! You can now use the site to submit your App for review. After a few business days if all goes well you should see your App live on the Mac App Store. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx new file mode 100644 index 00000000000..dcf192d337f --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# Manual Builds + +The Wails CLI does a lot of heavy lifting for the project, but sometimes it's desirable to manually build your project. This document will discuss the different operations the CLI does and how this may be achieved in different ways. + +## Build Process + +When either `wails build` or `wails dev` are used, the Wails CLI performs a common build process: + + - Install frontend dependencies + - Build frontend project + - Generate build assets + - Compile application + - [optional] Compress application + +### Install frontend dependencies + +#### CLI Steps + +- If the `-s` flag is given, this step is skipped +- Checks `wails.json` to see if there is an install command in the key `frontend:install` +- If there isn't, it skips this step +- If there is, it checks if `package.json` exists in the frontend directory. If it doesn't exist, it skips this step +- An MD5 sum is generated from the `package.json` file contents +- It checks for the existence of `package.json.md5` and if it exists, will compare the contents of it (an MD5 sum) with the one generated to see if the contents have changed. If they are the same, this step is skipped +- If `package.json.md5` does not exist, it creates it using the generated MD5 sum +- If a build is now required, or `node_modules` does not exist, or the `-f` flag is given, the install command is executed in the frontend directory + +#### Manual Steps + +This step could be done from the command line or a script with `npm install`. + +### Build frontend project + +#### Wails CLI + +- If the `-s` flag is given, this step is skipped +- Checks `wails.json` to see if there is a build command in the key `frontend:build` +- If there isn't, it skips this step +- If there is, it is executed in the frontend directory + +#### Manual Steps + +This step could be done from the command line or a script with `npm run build` or whatever the frontend build script is. + +### Generate assets + +#### Wails CLI + +- If `-nopackage` flag is set, this stage is skipped +- If the `build/appicon.png` file does not exist, a default one is created +- For Windows, see [Bundling for Windows](#windows) +- If `build/windows/icon.ico` does not exist, it will create it from the `build/appicon.png` image. + +##### Windows + +- If `build/windows/icon.ico` does not exist, it will create it from `build/appicon.png` using icon sizes of 256, 128, 64, 48, 32 and 16. This is done using [winicon](https://github.com/leaanthony/winicon). +- If the `build/windows/.manifest` file does not exist, it creates it from a default version. +- Compiles the application as a production build (above) +- Uses [winres](https://github.com/tc-hib/winres) to bundle the icon and manifest into a `.syso` file ready for linking. + +#### Manual Steps + +- Create `icon.ico` using the [winicon](https://github.com/leaanthony/winicon) CLI tool (or any other tool). +- Create / Update a `.manifest` file for your application +- Use the [winres CLI](https://github.com/tc-hib/go-winres) to generate a `.syso` file. + +### Compile application + +#### Wails CLI + +- If the `-clean` flag is provided, the `build` directory is deleted and recreated +- For `wails dev`, the following default Go flags are used: `-tags dev -gcflags "all=-N -l"` +- For `wails build`, the following default Go flags are used: `-tags desktop,production -ldflags "-w -s"` + - On Windows, `-ldflags "-w -h -H windowsgui"` +- Additional tags passed to the CLI using `-tags` are added to the defaults +- Additional ldflags passed to the CLI using `-ldflags` are added to the defaults +- The `-o` flag is passed through +- The Go compiler specified by `-compiler` will be used for compilation + +#### Manual steps + +- For dev build, the minimum command would be: `go build -tags dev -gcflags "all=-N -l"` +- For production build, the minimum command would be: `go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- Ensure that you compile in the same directory as the `.syso` file + +### Compress application + +#### Wails CLI + +- If the `-upx` flag has been given, the `upx` program will be run to compress the application with the default settings +- If `-upxflags` is also passed, these flags are used instead of the default ones + +#### Manual steps + +- Run `upx [flags]` manually to compress the application. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx new file mode 100644 index 00000000000..7123cbe6b60 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx @@ -0,0 +1,191 @@ +# Migrating from v1 + +## Overview + +Wails v2 is a significant change from v1. This document aims to highlight the changes and the steps in migrating an existing project. + +### Creating the Application + +In v1, the main application is created using `wails.CreateApp`, bindings are added with `app.Bind`, then the application is run using `app.Run()`. + +Example: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +In v2, there is just a single method, `wails.Run()`, that accepts [application options](../reference/options.mdx#application-options). + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### Binding + +In v1, it was possible to bind both arbitrary functions and structs. In v2, this has been simplified to only binding structs. The struct instances that were previously passed to the `Bind()` method in v1, are now specified in the `Bind` field of the [application options](../reference/options.mdx#application-options): + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +In v1, bound methods were available to the frontend at `window.backend`. This has changed to `window.go`.`` + +### Application Lifecycle + +In v1, there were 2 special methods in a bound struct: `WailsInit()` and `WailsShutdown()`. These have been replaced with 3 lifecycle hooks as part of the [application options](../reference/options.mdx#application-options): + +- [OnStartup](../reference/options.mdx#onstartup) +- [OnShutdown](../reference/options.mdx#onshutdown) +- [OnDomReady](../reference/options.mdx#ondomready) + +Note: [OnDomReady](../reference/options.mdx#ondomready) replaces the `wails:ready` system event in v1. + +These methods can be standard functions, but a common practice is to have them part of a struct: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### Runtime + +The runtime in v2 is much richer than v1 with support for menus, window manipulation and better dialogs. The signature of the methods has changed slightly - please refer the the [Runtime Reference](../reference/runtime/intro.mdx). + +In v1, the [runtime](../reference/runtime/intro.mdx) was available via a struct passed to `WailsInit()`. In v2, the runtime has been moved out to its own package. Each method in the runtime takes the `context.Context` that is passed to the [OnStartup](../reference/options.mdx#onstartup) method. + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startup is called at application startup +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} + +``` + +### Assets + +The _biggest_ change in v2 is how assets are handled. + +In v1, assets were passed via 2 application options: + +- `JS` - The application's JavaScript +- `CSS` - The application's CSS + +This meant that the responsibility of generating a single JS and CSS file was on the developer. This essentially required the use of complicated packers such as webpack. + +In v2, Wails makes no assumptions about your frontend assets, just like a webserver. All of your application assets are passed to the application options as an `embed.FS`. + +**This means there is no requirement to bundle your assets, encode images as Base64 or attempt the dark art of bundler configuration to use custom fonts**. + +At startup, Wails will scan the given `embed.FS` for `index.html` and use its location as the root path for all the other application assets - just like a webserver would. + +Example: An application has the following project layout. All final assets are placed in the `frontend/dist` directory: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +Those assets may be used by the application by simply creating an `embed.FS`: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +Of course, bundlers can be used if you wish to. The only requirement is to pass the final application assets directory to Wails using an `embed.FS` in the `Assets` key of the [application options](../reference/options.mdx#application-options). + +### Project Configuration + +In v1, the project configuration was stored in the `project.json` file in the project root. In v2, the project configuration is stored in the `wails.json` file in the project root. + +The format of the file is slightly different. Here is a comparison: + +

+ +| v1 | v2 | Notes | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | Removed | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | Changed | +| frontend / dir | | Removed | +| frontend / install | frontend:install | Changed | +| frontend / build | frontend:build | Changed | +| frontend / bridge | | Removed | +| frontend / serve | | Removed | +| tags | | Removed | +| | wailsjsdir | The directory to generate wailsjs modules | +| | assetdir | The directory of the compiled frontend assets for `dev` mode. This is normally inferred and could be left empty. | +| | reloaddirs | Comma separated list of additional directories to watch for changes and to trigger reloads in `dev` mode. This is only needed for some more advanced asset configurations. | + +

diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..4a3de2a61b5 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# Mouse Buttons + +The Wails runtime intercepts mouse clicks to determine whether a frameless window needs resizing or a window needs to be moved. It has been asked how to detect when a mouse click has occurred, because `window.onclick` doesn't report the mouse buttons correctly. The following code shows how to detect mouse clicks: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +Reference: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx new file mode 100644 index 00000000000..21f7875e389 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# Obfuscated Builds + +Wails includes support for obfuscating your application using [garble](https://github.com/burrowers/garble). + +To produce an obfuscated build, you can use the `-obfuscate` flag with the `wails build` command: + +```bash +wails build -obfuscated +``` + +To customise the obfuscation settings, you can use the `-garbleargs` flag: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +These settings may be persisted in your [project config](../reference/project-config). + +## How it works + +In a standard build, all bound methods are available in the frontend under the `window.go` variable. When these methods are called, the corresponding backend method is called using the fully qualified function name. When using an obfuscated build, methods are bound using an ID instead of a name. The bindings generated in the `wailsjs` directory use these IDs to call the backend functions. + +:::note + +To ensure that your application will work in obfuscated mode, you must use the generated bindings under the `wailsjs` directory in your application. + +::: + +## Example + +Importing the "Greet" method from the bindings like this: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +will ensure that the method will work correctly in obfuscated mode, as the bindings will be regenerated with IDs and the call mechanism updated. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx new file mode 100644 index 00000000000..e94cfbb7e77 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# 오버스크롤(Overscroll) + +[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) 은 페이지의 콘텐츠 경계를 넘어 스크롤할 때 가끔 발생하는 "바운스 효과"입니다. 이것은 모바일 앱에서도 동일합니다. CSS를 사용하여 비활성 할 수 있습니다. + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx new file mode 100644 index 00000000000..ce176943ea5 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx @@ -0,0 +1,47 @@ +# Routing + +Routing is a popular way to switch views in an application. This page offers some guidance around how to do that. + +## Vue + +The recommended approach for routing in Vue is [Hash Mode](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode): + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +The recommended approach for routing in Angular is [HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies#_hashlocationstrategy): + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +The recommended approach for routing in React is [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx new file mode 100644 index 00000000000..4c7cf45ba99 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx @@ -0,0 +1,387 @@ +# Code Signing + +This is a guide on how you can sign your binaries generated with Wails on MacOS and Windows. The guide will target CI environments, more specifically GitHub Actions. + +## Windows + +First off you need a code signing certificate. If you do not already have one, Microsoft's info page lists some providers [here](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate). Please note that an EV certificate is not required unless you need to write kernel-level software such as device drivers. For signing your Wails app, a standard code signing certificate will do just fine. + +It may be a good idea to check with your certificate provider how to sign your binaries on your local machine before targeting automated build systems, just so you know if there are any special requirements. For instance, [here](https://www.ssl.com/how-to/using-your-code-signing-certificate/) is SSL.com's code signing guide for Windows. If you know how to sign locally, it will be easier to troubleshoot any potential issues in a CI environment. For instance, SSL.com code signing certificates require the `/tr` flag for [SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool) while other providers may only need the `/t` flag for providing the timestamping server. Popular GitHub Actions for signing Windows binaries like [this one](https://github.com/Dana-Prajea/code-sign-action) does not support the `/tr` flag on SignTool.exe. Therefore this guide will focus on signing our app manually with PowerShell commands, but you can use actions like the [code-sign-action](https://github.com/Dana-Prajea/code-sign-action) Action if you prefer. + +First off, let's make sure we are able to build our Wails app in our GitHub CI. Here is a small workflow template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend manually here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +Next we need to give the GitHub workflow access to our signing certificate. This is done by encoding your .pfx or .p12 certificate into a base64 string. To do this in PowerShell, you can use the following command assuming your certificate is called 'my-cert.p12': + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +You should now have your .txt file with the base64 encoded certificate. It should start with _-----BEGIN CERTIFICATE-----_ and end with _-----END CERTIFICATE-----_. Now you need to make two action secrets on GitHub. Navigate to _Settings -> Secrets -> Actions_ and create the two following secrets: + +- **WIN_SIGNING_CERT** with the contents of your base64 encoded certificate text. +- **WIN_SIGNING_CERT_PASSWORD** with the contents of your certificate password. + +Now we're ready to implement the signing in our workflow using one of the two methods: + +### Method 1: signing with commands + +This method uses PowerShell commands to sign our app, and leaves you control over the entire signing process. + +After the `"Build Wails app"` step, we can add the following step to our workflow: + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +This script creates a new directory for your certificate file, creates the certificate file from our base64 secret, converts it to a .pfx file, and finally signs the binary. The following variables needs to be replaced in the last line: + +- **signing algorithm**: usually sha256. +- **timestamping server**: URL to the timestamping server to use with your certificate. +- **path to binary**: path to the binary you want to sign. + +Given that our Wails config has `outputfilename` set to "app.exe" and that we have a certificate from SSL.com, this would be our workflow: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### Method 2: automatically signing with Action + +It is possible to use a Windows code signing Action like [this](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) one, but note it requires a SHA1 hash for the certificate and a certificate name. View an example of how to configure it on the Action's [marketplace](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate). + +--- + +## MacOS + +First off you need your code signing certificate from Apple. If you do not have one, a simple Google search will help you acquire one. Once you have your certificate, you need to export it and encode it to base64. [This tutorial](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions) shows you how to do that in an easy manner. Once you have exported your .p12 certificate file, you can encode it to base64 as seen in the tutorial with the following command: + +```bash +base64 Certificates.p12 | pbcopy +``` + +Now you're ready to create some GitHub project secrets, just as with Windows: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64** with the contents of your newly copied base64 certificate. +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD** with the contents of your certificate password. +- **APPLE_PASSWORD** with the contents of an App-Specific password to your Apple-ID account which you can generate [here](https://appleid.apple.com/account/manage). + +Let's make sure we are able to build our Wails app in our GitHub Action workflow. Here is a small template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +For code signing on macOS, [gon](https://github.com/mitchellh/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and will be used in this guide. + +After the `Build Wails app` step, add the following to the workflow: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +Now we need to configure some gon config files in our `build/darwin` directory: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +Where `source` is your Wails binary, `bundle_id` is your bundle ID, `apple_id` contains your Apple ID username and App-Specific password which you created earlier, and `sign.application_identity` is your identity which you can find by running the following command: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +In this file you configure the entitlements you need for you app, e.g. camera permissions if your app uses the camera. Read more about entitlements [here](https://developer.apple.com/documentation/bundleresources/entitlements). + +Make sure you have updated your `Info.plist` file with the same bundle ID as you entered in `gon-sign.json`. Here's an example `Info.plist` file: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +Now we're ready to add the signing step in our workflow after building the Wails app: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +Please note that signing binaries with Apple could take anywhere from minutes to hours. + +## Combined workflow file: + +Here is our GitHub workflow file with Windows + macOS combined: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# End notes + +This guide inspired by the RiftShare project and its workflow, which is highly recommended to check out [here](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml). diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..4651c422ed1 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +This guide will go into: + +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. + +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. + +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. + +- `npm create svelte@latest frontend` + +##### Modify wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. + +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Install/remove dependencies using your favorite package manager. + +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js + +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. + +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. + +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. + +- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! + +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html + +``` + +... + + + +... + +``` + +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. + +- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling + +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions + +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx new file mode 100644 index 00000000000..790e3107f04 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx @@ -0,0 +1,97 @@ +# Templates + +Wails generates projects from pre-created templates. In v1, this was a difficult to maintain set of projects that were subject to going out of date. In v2, to empower the community, a couple of new features have been added for templates: + +- Ability to generate projects from [Remote Templates](../reference/cli.mdx#remote-templates) +- Tooling to help create your own templates + +## Creating Templates + +To create a template, you can use the `wails generate template` command. To generate a default template, run: + +`wails generate template -name mytemplate` + +This creates the directory "mytemplate" with default files: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### Template Overview + +The default template consists of the following files and directories: + +| Filename / Dir | Description | +| --------------- | -------------------------------------------- | +| NEXTSTEPS.md | Instructions on how to complete the template | +| README.md | The README published with the template | +| app.tmpl.go | `app.go` template file | +| frontend/ | The directory containing frontend assets | +| go.mod.tmpl | `go.mod` template file | +| main.tmpl.go | `main.go` template file | +| template.json | The template metadata | +| wails.tmpl.json | `wails.json` template file | + +At this point it is advisable to follow the steps in `NEXTSTEPS.md`. + +## Creating a Template from an Existing Project + +It's possible to create a template from an existing frontend project by passing the path to the project when generating the template. We will now walk through how to create a Vue 3 template: + +- Install the vue cli: `npm install -g @vue/cli` +- Create the default project: `vue create vue3-base` + - Select `Default (Vue 3) ([Vue 3] babel, eslint)` +- After the project has been generated, run: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- The template may now be customised as specified in the `NEXTSTEPS.md` file +- Once the files are ready, it can be tested by running: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- To test the new project, run: `cd my-vue3-project` then `wails build` +- Once the project has compiled, run it: `.\build\bin\my-vue3-project.exe` +- You should have a fully functioning Vue3 application: + +```mdx-code-block +
+ +
+``` + +## Publishing Templates + +Publishing a template is simply pushing the files to GitHub. The following best practice is encouraged: + +- Remove any unwanted files and directories (such as `.git`) from your frontend directory +- Ensure that `template.json` is complete, especially `helpurl` +- Push the files to GitHub +- Create a PR on the [Community Templates](../community/templates.mdx) page +- Announce the template on the [Template Announcement](https://github.com/wailsapp/wails/discussions/825) discussion board diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..2b5b7a25840 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# Troubleshooting + +An assortment of troubleshooting tips. + +## `wails` 명령어를 찾을 수 없나요? + +시스템에서 `wails` 명령어를 찾을 수 없는 경우 Go 설치 지침을 잘 따라했는지 확인해보세요. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment variable. You will also normally need to close and reopen any open command prompts so that changes to the environment made by the installer are reflected at the command prompt. + +## My application is displaying a white/blank screen + +Check that your application includes the assets from the correct directory. In your `main.go` file, you will have something similar to the following code: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +Check that `frontend/dist` contains your application assets. + +### Mac + +If this happens on Mac, try adding the following to your `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Mac application not valid + +If your built application looks like this in finder: + +```mdx-code-block +

+ +

+``` + +it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to the `build/darwin` directory. + +## My application is not displaying the correct icon in Windows Explorer + +If your application is not displaying the correct icon, try deleting the hidden `IconCache.db` file located in the `C:\Users\<your username>\AppData\Local` directory. This will force Windows to rebuild the icon cache. + +Source: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## Cannot call backend method from frontend with variadic arguments + +If you have a backend method defined with variadic parameters, eg: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +calling this method from the frontend like this will fail: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Workaround: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Credit: https://github.com/wailsapp/wails/issues/1186 + +## I'm having getting proxy errors when trying to install Wails + +If you are getting errors like this: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +it's probably because the official Go Proxy is being blocked (Users in China have reported this). The solution is to set up the proxy manually, eg: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +Source: https://github.com/wailsapp/wails/issues/1233 + +## The generated TypeScript doesn't have the correct types + +Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, it is possible to specify what types should be generated using the `ts_type` struct tag. For more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). + +## When I navigate away from `index.html`, I am unable to call methods on the frontend + +If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding the following imports to the `` section of any new page you navigate to: + +```html + + + + +``` + +Source: https://github.com/wailsapp/wails/discussions/1512 + +## I get `too many open files` errors on my Mac when I run `wails dev` + +By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. This limit can be increased by running: `ulimit -n 1024` in the terminal. + +FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). + +## My Mac app gives me weird compilation errors + +A few users have reported seeing compilation errors such as the following: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. + +If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: + +`xcode-select -p` + +If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" + +It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine that may have different versions of Node installed, you may not be able to run your application. + +If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. + +## Build process stuck on "Generating bindings" + +Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx new file mode 100644 index 00000000000..ed258656d63 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +This page is for miscellaneous tips and tricks when using Visual Studio Code with Wails. + +## Vetur Configuration + +Many thanks to [@Lyimmi](https://github.com/Lyimmi) for this tip. Originally posted [here](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349). + +Vetur is a popular plugin for Visual Studio Code that provides syntax highlighting and code completion for Vue projects. When loading a Wails project in VSCode, Vetur will throw an error as it is expecting to find the frontend project in the root directory. To fix this, you can do the following: + +Create a file named `vetur.config.js` in the project's root. + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +Next, configure `frontend/tsconfig.json`: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +This should enable you to now use Vetur as expected. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx new file mode 100644 index 00000000000..22d3aec7078 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# NSIS installer + +```mdx-code-block +

+ +
+

+``` + +Wails는 [NSIS installer](https://nsis.sourceforge.io/)를 사용하여 Windows 인스톨러 생성을 지원합니다. + +## NSIS 설치하기 + +### Windows + +인스톨러는 [NSIS Download](https://nsis.sourceforge.io/Download) 페이지에서 사용이 가능합니다. + +만약 chocolatey 패키지 매니저를 사용한다면, 다음 스크립트를 실행하세요: + +``` +choco install nsis +``` + +만약 NSIS를 수동으로 설치하기 원한다면, NSIS가 설치된 경로에서 `makensis.exe`가 포함된 _Bin_ 폴더를 추가 해야합니다. [여기](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/) Windows에서 어떻게 경로를 추가하는지에 대한 튜토리얼이 있습니다. + +### Linux + +`nsis` 패키지는 사용자의 배포판 패키지 매니저를 통해 이용이 가능합니다. + +### MacOS + +NSIS는 homebrew를 통해 설치가 가능합니다: `brew install nsis`. + +## 인스톨러 생성하기 + +새로운 프로젝트가 생성되었을때, Wails는 NSIS 설정 파일을 `build/windows/installer`에 생성합니다. 설정 데이터는 `installer/info.json`로 부터 읽어오며 이 데이터는 이 프로젝트의 `wails.json` Info 섹션에서 설정되어 사용됩니다: + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +애플리케이션을 위한 인스톨러를 생성하기 위해 `-nsis` 플래그를 `wails build`와 함께 사용하세요: + +``` +wails build -nsis +``` + +이제 인스톨러는 `build/bin` 디렉토리에서 이용이 가능합니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx new file mode 100644 index 00000000000..821808c0b8d --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +This page has miscellaneous guides related to developing Wails applications for Windows. + +## Handling the WebView2 Runtime Dependency + +Wails applications built for Windows have a runtime requirement on the Microsoft [WebView2 Runtime](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). Windows 11 will have this installed by default, but some machines won't. Wails offers an easy approach to dealing with this dependency. + +By using the `-webview2` flag when building, you can decide what your application will do when a suitable runtime is not detected (including if the installed runtime is too old). The four options are: + +1. Download +2. Embed +3. Browser +4. Error + +### Download + +This option will prompt the user that no suitable runtime has been found and then offer to download and run the official bootstrapper from Microsoft's WebView2 site. If the user proceeds, the official bootstrapper will be downloaded and run. + +### Embed + +This option embeds the official bootstrapper within the application. If no suitable runtime has been found, the application will offer to run the bootstrapper. This adds ~150k to the binary size. + +### Browser + +This option will prompt the user that no suitable runtime has been found and then offer to open a browser to the official WebView2 page where the bootstrapper can be downloaded and installed. The application will then exit, leaving the installation up to the user. + +### Error + +If no suitable runtime is found, an error is given to the user and no further action taken. + +## Fixed version runtime + +Another way of dealing with webview2 dependency is shipping it yourself. You can download [fixed version runtime](https://developer.microsoft.com/microsoft-edge/webview2/#download-section) and bundle or download it with your application. + +Also, you should specify path to fixed version of webview2 runtime in the `windows.Options` structure when launching wails. + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +Note: When `WebviewBrowserPath` is specified, `error` strategy will be forced in case of minimal required version mismatch or invalid path to a runtime. + +## Spawning other programs + +When spawning other programs, such as scripts, you will see the window appear on the screen. To hide the window, you can use the following code: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +Solution provided by [sithembiso](https://github.com/sithembiso) on the [discussions board](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172). diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx new file mode 100644 index 00000000000..44fa130cc74 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# How does it work? + +A Wails application is a standard Go application, with a webkit frontend. The Go part of the application consists of the application code and a runtime library that provides a number of useful operations, like controlling the application window. The frontend is a webkit window that will display the frontend assets. Also available to the frontend is a JavaScript version of the runtime library. Finally, it is possible to bind Go methods to the frontend, and these will appear as JavaScript methods that can be called, just as if they were local JavaScript methods. + +```mdx-code-block +
+ +
+``` + +## The Main Application + +### Overview + +The main application consists of a single call to `wails.Run()`. It accepts the application configuration which describes the size of the application window, the window title, what assets to use, etc. A basic application might look like this: + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### Options rundown + +This example has the following options set: + +- `Title` - The text that should appear in the window's title bar +- `Width` & `Height` - The dimensions of the window +- `Assets` - The application's frontend assets +- `OnStartup` - A callback for when the window is created and is about to start loading the frontend assets +- `OnShutdown` - A callback for when the application is about to quit +- `Bind` - A slice of struct instances that we wish to expose to the frontend + +A full list of application options can be found in the [Options Reference](reference/options). + +#### Assets + +The `Assets` option is mandatory as you can't have a Wails application without frontend assets. Those assets can be any files you would expect to find in a web application - html, js, css, svg, png, etc. **There is no requirement to generate asset bundles** - plain files will do. When the application starts, it will attempt to load `index.html` from your assets and the frontend will essentially work as a browser from that point on. It is worth noting that there is no requirement on where in the `embed.FS` the files live. It is likely that the embed path uses a nested directory relative to your main application code, such as `frontend/dist`: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +At startup, Wails will iterate the embedded files looking for the directory containing `index.html`. All other assets will be loaded relative to this directory. + +As production binaries use the files contained in `embed.FS`, there are no external files required to be shipped with the application. + +When running in development mode using the `wails dev` command, the assets are loaded off disk, and any changes result in a "live reload". The location of the assets will be inferred from the `embed.FS`. + +More details can be found in the [Application Development Guide](guides/application-development.mdx). + +#### Application Lifecycle Callbacks + +Just before the frontend is about to load `index.html`, a callback is made to the function provided in [OnStartup](reference/options.mdx#onstartup). A standard Go context is passed to this method. This context is required when calling the runtime so a standard pattern is to save a reference to in this method. Just before the application shuts down, the [OnShutdown](reference/options.mdx#onshutdown) callback is called in the same way, again with the context. There is also an [OnDomReady](reference/options.mdx#ondomready) callback for when the frontend has completed loading all assets in `index.html` and is equivalent of the [`body onload`](https://www.w3schools.com/jsref/event_onload.asp) event in JavaScript. It is also possible to hook into the window close (or application quit) event by setting the option [OnBeforeClose](reference/options.mdx#onbeforeclose). + +#### Method Binding + +The `Bind` option is one of the most important options in a Wails application. It specifies which struct methods to expose to the frontend. Think of structs like "controllers" in a traditional web application. When the application starts, it examines the struct instances listed in the `Bind` field in the options, determines which methods are public (starts with an uppercase letter) and will generate JavaScript versions of those methods that can be called by the frontend code. + +:::info Note + +Wails requires that you pass in an _instance_ of the struct for it to bind it correctly + +::: + +In this example, we create a new `App` instance and then add this instance to the `Bind` option in `wails.Run`: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +You may bind as many structs as you like. Just make sure you create an instance of it and pass it in `Bind`: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +When you run `wails dev` (or `wails generate module`), a frontend module will be generated containing the following: + +- JavaScript bindings for all bound methods +- TypeScript declarations for all bound methods +- TypeScript definitions for all Go structs used as inputs or outputs by the bound methods + +This makes it incredibly simple to call Go code from the frontend, using the same strongly typed datastructures. + +## The Frontend + +### Overview + +The frontend is a collection of files rendered by webkit. It's like a browser and webserver in one. There is virtually[^1] no limit to which frameworks or libraries you can use. The main points of interaction between the frontend and your Go code are: + +- Calling bound Go methods +- Calling runtime methods + +### Calling bound Go methods + +When you run your application with `wails dev`, it will automatically generate JavaScript bindings for your structs in a directory called `wailsjs/go` (You can also do this by running `wails generate module`). The generated files mirror the package names in your application. In the example above, we bind `app`, which has one public method `Greet`. This will lead to the generation of the following files: + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +Here we can see that there is a `main` package that contains the JavaScript bindings for the bound `App` struct, as well as the TypeScript declaration file for those methods. To call `Greet` from our frontend, we simply import the method and call it like a regular JavaScript function: + +```javascript +// ... +import { Greet } from "../wailsjs/go/main/App"; + +function doGreeting(name) { + Greet(name).then((result) => { + // Do something with result + }); +} +``` + +The TypeScript declaration file gives you the correct types for the bound methods: + +```ts +export function Greet(arg1: string): Promise; +``` + +The generated methods return a Promise. A successful call will result in the first return value from the Go call to be passed to the `resolve` handler. An unsuccessful call is when a Go method that has an error type as it's second return value, passes an error instance back to the caller. This is passed back via the `reject` handler. In the example above, `Greet` only returns a `string` so the JavaScript call will never reject - unless invalid data is passed to it. + +All data types are correctly translated between Go and JavaScript. Even structs. If you return a struct from a Go call, it will be returned to your frontend as a JavaScript class. + +:::info Note + +Struct fields _must_ have a valid `json` tag to be included in the generated TypeScript. + +Anonymous nested structs are not supported at this time. + +::: + +It is possible to send structs back to Go. Any JavaScript map/class passed as an argument that is expecting a struct, will be converted to that struct type. To make this process a lot easier, in `dev` mode, a TypeScript module is generated, defining all the struct types used in bound methods. Using this module, it's possible to construct and send native JavaScript objects to the Go code. + +There is also support for Go methods that use structs in their signature. All Go structs specified by a bound method (either as parameters or return types) will have TypeScript versions auto generated as part of the Go code wrapper module. Using these, it's possible to share the same data model between Go and JavaScript. + +Example: We update our `Greet` method to accept a `Person` instead of a string: + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +The `wailsjs/go/main/App.js` file will still have the following code: + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +But the `wailsjs/go/main/App.d.ts` file will be updated with the following code: + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +As we can see, the "main" namespace is imported from a new "models.ts" file. This file contains all the struct definitions used by our bound methods. In this example, this is a `Person` struct. If we look at `models.ts`, we can see how the models are defined: + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +So long as you have TypeScript as part of your frontend build configuration, you can use these models in the following way: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +The combination of generated bindings and TypeScript models makes for a powerful development environment. + +More information on Binding can be found in the [Binding Methods](guides/application-development.mdx#binding-methods) section of the [Application Development Guide](guides/application-development.mdx). + +### Calling runtime methods + +The JavaScript runtime is located at `window.runtime` and contains many methods to do various tasks such as emit an event or perform logging operations: + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +More details about the JS runtime can be found in the [Runtime Reference](reference/runtime/intro). + +[^1]: There is a very small subset of libraries that use features unsupported in WebViews. There are often alternatives and workarounds for such cases. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx new file mode 100644 index 00000000000..9fd585e11f8 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# 소개 + +Wails is a project that enables you to write desktop apps using Go and web technologies. + +Consider it a lightweight and fast Electron alternative for Go. You can easily build applications with the flexibility and power of Go, combined with a rich, modern frontend. + +### Features + +- Native Menus, Dialogs, Theming and Translucency +- Windows, macOS and linux support +- Built in templates for Svelte, React, Preact, Vue, Lit and Vanilla JS +- Easily call Go methods from JavaScript +- Automatic Go struct to TypeScript model generation +- No CGO or external DLLs required on Windows +- Live development mode using the power of [Vite](https://vitejs.dev/) +- Powerful CLI to easily Create, Build and Package applications +- A rich [runtime library](/docs/reference/runtime/intro) +- Applications built with Wails are Apple & Microsoft Store compliant + +This is [varly](https://varly.app) - a desktop application for MacOS & Windows written using Wails. Not only does it look great, it uses native menus and translucency - everything you'd expect from a modern native app. + +```mdx-code-block +

+ + + +

+``` + +### Quick Start Templates + +Wails comes with a number of pre-configured templates that allow you to get your application up and running quickly. There are templates for the following frameworks: Svelte, React, Vue, Preact, Lit and Vanilla. There are both JavaScript and TypeScript versions for each template. + +### Native Elements + +Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. + +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. + +### Go & JavaScript Interoperability + +Wails automatically makes your Go methods available to JavaScript, so you can call them by name from your frontend! It even generates TypeScript models for the structs used by your Go methods, so you can pass the same data structures between Go and JavaScript. + +### Runtime Library + +Wails provides a runtime library, for both Go and JavaScript, that handles a lot of the things modern applications need, like Eventing, Logging, Dialogs, etc. + +### Live Development Experience + +#### Automatic Rebuilds + +When you run your application in "dev" mode, Wails will build your application as a native desktop application, but will read your assets from disk. It will detect any changes to your Go code and automatically rebuild and relaunch your application. + +#### Automatic Reloads + +When changes to your application assets are detected, your running application will "reload", reflecting your changes almost immediately. + +#### Develop your application in a Browser + +If you prefer to debug and develop in a browser then Wails has you covered. The running application also has a webserver that will run your application in any browser that connects to it. It will even refresh when your assets change on disk. + +### Production-ready Native Binaries + +When you're ready to do the final build of your application, the CLI will compile it down to a single executable, with all the assets bundled into it. On Windows and MacOS, it is possible to create a native package for distribution. The assets used in packaging (icon, info.plist, manifest file, etc) are part of your project and may be customised, giving you total control over how your applications are built. + +### Tooling + +The Wails CLI provides a hassle-free way to generate, build and bundle your applications. It will do the heavy lifting of creating icons, compiling your application with optimal settings and delivering a distributable, production ready binary. Choose from a number of starter templates to get up and running quickly! diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx new file mode 100644 index 00000000000..c7aea10a64d --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx @@ -0,0 +1,240 @@ +--- +sidebar_position: 2 +--- + +# CLI + +The Wails CLI has a number of commands that are used for managing your projects. All commands are run in the following way: + +`wails ` + +## init + +`wails init` is used for generating projects. + +| Flag | Description | Default | +|:------------------ |:----------------------------------------------------------------------------------------------------------------------- |:-------------------:| +| -n "project name" | Name of the project. **Mandatory**. | | +| -d "project dir" | Project directory to create | Name of the project | +| -g | Initialise git repository | | +| -l | List available project templates | | +| -q | Suppress output to console | | +| -t "template name" | The project template to use. This can be the name of a default template or a URL to a remote template hosted on github. | vanilla | +| -ide | Generate IDE project files | | +| -f | Force build application | false | + +Example: `wails init -n test -d mytestproject -g -ide vscode -q` + +This will generate a a project called "test" in the "mytestproject" directory, initialise git, generate vscode project files and do so silently. + +More information on using IDEs with Wails can be found [here](../guides/ides.mdx). + +### Remote Templates + +Remote templates (hosted on GitHub) are supported and can be installed by using the template's project URL. + +Example: `wails init -n test -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +A list of community maintained templates can be found [here](../community/templates.mdx) + +:::warning Attention + +**The Wails project does not maintain, is not responsible nor liable for 3rd party templates!** + +If you are unsure about a template, inspect `package.json` and `wails.json` for what scripts are run and what packages are installed. + +::: + +## build + +`wails build` is used for compiling your project to a production-ready binary. + +| Flag | Description | Default | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Cleans the `build/bin` directory | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Force build application | | +| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -m | Skip mod tidy before compile | | +| -nopackage | Do not package application | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o filename | Output filename | | +| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | +| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Build with Go's race detector | | +| -s | Skip building the frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | +| -trimpath | Remove all file system paths from the resulting executable. | | +| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | +| -upx | Compress final binary using "upx" | | +| -upxflags | Flags to pass to upx | | +| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | +| -windowsconsole | Keep the console window for Windows builds | | + +For a detailed description of the `webview2` flag, please refer to the [Windows](../guides/windows.mdx) Guide. + +If you prefer to build using standard Go tooling, please consult the [Manual Builds](../guides/manual-builds.mdx) guide. + +Example: + +`wails build -clean -o myproject.exe` + +:::info + +On Mac, the application will be bundled with `Info.plist`, not `Info.dev.plist`. + +::: + +:::info UPX on Apple Silicon + +There are [issues](https://github.com/upx/upx/issues/446) with using UPX with Apple Silicon. + +::: + +:::info UPX on Windows + +Some Antivirus vendors false positively mark `upx` compressed binaries as virus, see [issue](https://github.com/upx/upx/issues/437). + +::: + +### Platforms + +Supported platforms are: + +| Platform | Description | +|:---------------- |:--------------------------------------------- | +| darwin | MacOS + architecture of build machine | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | MacOS AMD64+ARM64 universal application | +| windows | Windows 10/11 + architecture of build machine | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + architecture of build machine | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## doctor + +`wails doctor` will run diagnostics to ensure that your system is ready for development. + +Example: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## dev + +`wails dev` is used to run your application in a "live development" mode. This means: + +- The application's `go.mod` will be updated to use the same version of Wails as the CLI +- The application is compiled and run automatically +- A watcher is started and will trigger a rebuild of your dev app if it detects changes to your go files +- A webserver is started on `http://localhost:34115` which serves your application (not just frontend) over http. This allows you to use your favourite browser development extensions +- All application assets are loaded from disk. If they are changed, the application will automatically reload (not rebuild). All connected browsers will also reload +- A JS module is generated that provides the following: +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- A second JS module is generated that provides a wrapper + TS declaration for the runtime +- On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. + +| Flag | Description | Default | +|:---------------------------- |:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:--------------------- | +| -appargs "args" | Arguments passed to the application in shell style | | +| -assetdir "./path/to/assets" | Serve assets from the given directory instead of using the provided asset FS | Value in `wails.json` | +| -browser | Opens a browser to `http://localhost:34115` on startup | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debounce | The time to wait for reload after an asset change is detected | 100 (milliseconds) | +| -devserver "host:port" | The address to bind the wails dev server to | "localhost:34115" | +| -extensions | Extensions to trigger rebuilds (comma separated) | go | +| -forcebuild | Force build of application | | +| -frontenddevserverurl "url" | Use 3rd party dev server url to serve assets, EG Vite | "" | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -loglevel "loglevel" | Loglevel to use - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | Turn off colour cli output | false | +| -noreload | Disable automatic reload when assets change | | +| -nosyncgomod | Do not sync go.mod with the Wails version | false | +| -race | Build with Go's race detector | false | +| -reloaddirs | Additional directories to trigger reloads (comma separated) | Value in `wails.json` | +| -s | Skip building the frontend | false | +| -save | Saves the given `assetdir`, `reloaddirs`, `wailsjsdir`, `debounce`, `devserver` and `frontenddevserverurl` flags in `wails.json` to become the defaults for subsequent invocations. | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to compiler (quoted and space separated) | | +| -v | Verbosity level (0 - silent, 1 - standard, 2 - verbose) | 1 | +| -wailsjsdir | The directory to generate the generated Wails JS modules | Value in `wails.json` | + +Example: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +This command will do the following: + +- Build the application and run it (more details [here](../guides/manual-builds.mdx) +- Generate the Wails JS modules in `./frontend/src` +- Watch for updates to files in `./frontend/dist` and reload on any change +- Open a browser and connect to the application + +There is more information on using this feature with existing framework scripts [here](../guides/application-development.mdx#live-reloading). + +## generate + +### template + +Wails uses templates for project generation. The `wails generate template` command helps scaffold a template so that it may be used for generating projects. + +| Flag | Description | +|:---------------- |:------------------------------------------- | +| -name | The template name (Mandatory) | +| -frontend "path" | Path to frontend project to use in template | + +For more details on creating templates, consult the [Templates guide](../guides/templates.mdx). + +### module + +The `wails generate module` command allows you to manually generate the `wailsjs` directory for your application. + +## update + +`wails update` will update the version of the Wails CLI. + +| Flag | Description | +|:------------------ |:------------------------------------- | +| -pre | Update to latest pre-release version | +| -version "version" | Install a specific version of the CLI | + +## version + +`wails version` will simply output the current CLI version. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx new file mode 100644 index 00000000000..ff9a2442281 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# Menus + +It is possible to add an application menu to Wails projects. This is achieved by defining a [Menu](#menu) struct and setting it in the [`Menu`](../reference/options.mdx#menu) application config, or by calling the runtime method [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). + +An example of how to create a menu: + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +It is also possible to dynamically update the menu, by updating the menu struct and calling [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). + +The example above uses helper methods, however it's possible to build the menu structs manually. + +## Menu + +A Menu is a collection of MenuItems: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +For the Application menu, each MenuItem represents a single menu such as "Edit". + +A simple helper method is provided for building menus: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +This makes the layout of the code more like that of a menu without the need to add the menu items manually after creating them. Alternatively, you can just create the menu items and add them to the menu manually. + +## MenuItem + +A MenuItem represents an item within a Menu. + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| Field | Type | Notes | +| ----------- | ------------------------------------ | ------------------------------------------------------------- | +| Label | string | The menu text | +| Accelerator | [\*keys.Accelerator](#accelerator) | Key binding for this menu item | +| Type | [Type](#type) | Type of MenuItem | +| Disabled | bool | Disables the menu item | +| Hidden | bool | Hides this menu item | +| Checked | bool | Adds check to item (Checkbox & Radio types) | +| SubMenu | [\*Menu](#menu) | Sets the submenu | +| Click | [Callback](#callback) | Callback function when menu clicked | +| Role | string | Defines a [role](#role) for this menu item. Mac only for now. | + +### Accelerator + +Accelerators (sometimes called keyboard shortcuts) define a binding between a keystroke and a menu item. Wails defines an Accelerator as a combination or key + [Modifier](#modifier). They are available in the `"github.com/wailsapp/wails/v2/pkg/menu/keys"` package. + +Example: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +Keys are any single character on a keyboard with the exception of `+`, which is defined as `plus`. Some keys cannot be represented as characters so there are a set of named characters that may be used: + +| | | | | +|:-----------:|:-----:|:-----:|:---------:| +| `backspace` | `f1` | `f16` | `f31` | +| `tab` | `f2` | `f17` | `f32` | +| `return` | `f3` | `f18` | `f33` | +| `enter` | `f4` | `f19` | `f34` | +| `escape` | `f5` | `f20` | `f35` | +| `left` | `f6` | `f21` | `numlock` | +| `right` | `f7` | `f22` | | +| `up` | `f8` | `f23` | | +| `down` | `f9` | `f24` | | +| `space` | `f10` | `f25` | | +| `delete` | `f11` | `f36` | | +| `home` | `f12` | `f37` | | +| `end` | `f13` | `f38` | | +| `page up` | `f14` | `f39` | | +| `page down` | `f15` | `f30` | | + +Wails also supports parsing accelerators using the same syntax as Electron. This is useful for storing accelerators in config files. + +Example: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### Modifier + +The following modifiers are keys that may be used in combination with the accelerator key: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +A number of helper methods are available to create Accelerators using modifiers: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +Modifiers can be combined using `keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)`: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### Type + +Each menu item must have a type and there are 5 types available: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +For convenience, helper methods are provided to quickly create a menu item: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +You can also create menu items directly on a menu by using the "Add" helpers: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +A note on radio groups: A radio group is defined as a number of radio menu items that are next to each other in the menu. This means that you do not need to group items together as it is automatic. However, that also means you cannot have 2 radio groups next to each other - there must be a non-radio item between them. + +### Callback + +Each menu item may have a callback that is executed when the item is clicked: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +The function is given a `CallbackData` struct which indicates which menu item triggered the callback. This is useful when using radio groups that may share a callback. + +### Role + +:::info Roles + +Roles are currently supported on Mac only. + +::: + +A menu item may have a role, which is essentially a pre-defined menu item. We currently support the following roles: + +| Role | Description | +| ------------ | ------------------------------------------------------------------------ | +| AppMenuRole | The standard Mac application menu. Can be created using `menu.AppMenu()` | +| EditMenuRole | The standard Mac edit menu. Can be created using `menu.EditMenu()` | diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx new file mode 100644 index 00000000000..46d1c071b60 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx @@ -0,0 +1,853 @@ +--- +sidebar_position: 3 +--- + +# Options + +## Application Options + +The `Options.App` struct contains the application configuration. It is passed to the `wails.Run()` method: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // User messages that can be customised + Messages *windows.Messages + // OnSuspend is called when Windows enters low power mode + OnSuspend func() + // OnResume is called when Windows resumes from low power mode + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### Title + +The text shown in the window's title bar. + +Name: Title
Type: `string` + +### Width + +The initial width of the window. + +Name: Width
Type: `int`
Default: 1024. + +### Height + +The initial height of the window. + +Name: Height
Type: `int`
Default: 768 + +### DisableResize + +By default, the main window is resizable. Setting this to `true` will keep it a fixed size. + +Name: DisableResize
Type: `bool` + +### Fullscreen + +Deprecated: Please use [WindowStartState](#windowstartstate). + +### WindowStartState + +Defines how the window should present itself at startup. + +| Value | Win | Mac | Lin | +| ---------- | --- | --- | --- | +| Fullscreen | ✅ | ✅ | ✅ | +| Maximised | ✅ | ✅ | ✅ | +| Minimised | ✅ | ❌ | ✅ | + +Name: WindowStartState
Type: `options.WindowStartState` + +### Frameless + +When set to `true`, the window will have no borders or title bar. Also see [Frameless Windows](../guides/frameless.mdx). + +Name: Frameless
Type: `bool` + +### MinWidth + +This sets the minimum width for the window. If the value given in `Width` is less than this value, the window will be set to `MinWidth` by default. + +Name: MinWidth
Type: `int` + +### MinHeight + +This sets the minimum height for the window. If the value given in `Height` is less than this value, the window will be set to `MinHeight` by default. + +Name: MinHeight
Type: `int` + +### MaxWidth + +This sets the maximum width for the window. If the value given in `Width` is more than this value, the window will be set to `MaxWidth` by default. + +Name: MaxWidth
Type: `int` + +### MaxHeight + +This sets the maximum height for the window. If the value given in `Height` is more than this value, the window will be set to `MaxHeight` by default. + +Name: MaxHeight
Type: `int` + +### StartHidden + +When set to `true`, the application will be hidden until [WindowShow](../reference/runtime/window.mdx#windowshow) is called. + +Name: StartHidden
Type: `bool` + +### HideWindowOnClose + +By default, closing the window will close the application. Setting this to `true` means closing the window will + +hide the window instead. + +Name: HideWindowOnClose
Type: `bool` + +### BackgroundColour + +This value is the default background colour of the window. Example: options.NewRGBA(255,0,0,128) - Red at 50% transparency + +Name: BackgroundColour
Type: `*options.RGBA`
Default: white + +### AlwaysOnTop + +Indicates that the window should stay above other windows when losing focus. + +Name: AlwaysOnTop
Type: `bool` + +### Assets + +Deprecated: Please use Assets on [AssetServer specific options](#assetserver). + +### AssetsHandler + +Deprecated: Please use AssetsHandler on [AssetServer specific options](#assetserver). + +### AssetServer + +This defines AssetServer specific options. It allows to customize the AssetServer with static assets, serving assets dynamically with an `http.Handler` or hook into the request chain with an `assetserver.Middleware`. + +Not all features of an `http.Request` are currently supported, please see the following feature matrix: + +| Feature | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +Name: AssetServer
Type: `*assetserver.Options` + +#### Assets + +The static frontend assets to be used by the application. + +A GET request is first tried to be served from this `fs.FS`. If the `fs.FS` returns `os.ErrNotExist` for that file, the request handling will fallback to the [Handler](#handler) and tries to serve the GET request from it. + +If set to nil, all GET requests will be forwarded to [Handler](#handler). + +Name: Assets
Type: `fs.FS` + +#### Handler + +The assets handler is a generic `http.Handler` for fallback handling of assets that can't be found. + +The handler will be called for every GET request that can't be served from [Assets](#assets), due to `os.ErrNotExist`. Furthermore all non GET requests will always be served from this Handler. If not defined, the result is the following in cases where the Handler would have been called: + +- GET request: `http.StatusNotFound` +- Other request: `http.StatusMethodNotAllowed` + +NOTE: When used in combination with a Frontend DevServer there might be limitations, eg. Vite serves the index.html on every path, that does not contain a file extension. + +Name: AssetsHandler
Type: `http.Handler` + +#### Middleware + +Middleware is a HTTP Middleware which allows to hook into the AssetServer request chain. It allows to skip the default request handler dynamically, e.g. implement specialized Routing etc. The Middleware is called to build a new `http.Handler` used by the AssetSever and it also receives the default handler used by the AssetServer as an argument. + +If not defined, the default AssetServer request chain is executed. + +Name: Middleware
Type: `assetserver.Middleware` + +### Menu + +The menu to be used by the application. More details about Menus in the [Menu Reference](../reference/runtime/menu.mdx). + +:::note + +On Mac, if no menu is specified, a default menu will be created. + +::: + +Name: Menu
Type: `*menu.Menu` + +### Logger + +The logger to be used by the application. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: Logger
Type: `logger.Logger`
Default: Logs to Stdout + +### LogLevel + +The default log level. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevel
Type: `logger.LogLevel`
Default: `Info` in dev mode, `Error` in production mode + +### LogLevelProduction + +The default log level for production builds. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevelProduction
Type: `logger.LogLevel`
Default: `Error` + +### OnStartup + +This callback is called after the frontend has been created, but before `index.html` has been loaded. It is given the application context. + +Name: OnStartup
Type: `func(ctx context.Context)` + +### OnDomReady + +This callback is called after the frontend has loaded `index.html` and its resources. It is given the application context. + +Name: OnDomReady
Type: `func(ctx context.Context)` + +### OnShutdown + +This callback is called after the frontend has been destroyed, just before the application terminates. It is given the application context. + +Name: OnShutdown
Type: `func(ctx context.Context)` + +### OnBeforeClose + +If this callback is set, it will be called when the application is about to quit, either by clicking the window close button or calling `runtime.Quit`. Returning true will cause the application to continue, false will continue shutdown as normal. This is good for confirming with the user that they wish to exit the program. + +Example: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +Name: OnBeforeClose
Type: `func(ctx context.Context) bool` + +### CSSDragProperty + +Indicates the CSS property to use to identify which elements can be used to drag the window. Default: `--wails-draggable`. + +Name: CSSDragProperty
Type: `string` + +### CSSDragValue + +Indicates what value the `CSSDragProperty` style should have to drag the window. Default: `drag`. + +Name: CSSDragValue
Type: `string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenu enables the browser's default context-menu in production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : + +| CSS Style | Behavior | +| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `--default-contextmenu: auto;` | (**default**) will show the default context menu only if :
contentEditable is true OR text has been selected OR element is input or textarea | +| `--default-contextmenu: show;` | will always show the default context menu | +| `--default-contextmenu: hide;` | will always hide the default context menu | + +This rule is inherited like any normal CSS rule, so nesting works as expected. + +:::note +This filtering functionality is only enabled in production, so in development and in debug build, the full context-menu is always available everywhere. +::: + +:::warning +This filtering functionality is NOT a security measure, the developer should expect that the full context-menu could be leaked anytime which could contain commands like (Download image, Reload, Save webpage), if this is a concern, the developer SHOULD NOT enable the default context-menu. +::: + + +Name: EnableDefaultContextMenu
Type: `bool` + +### EnableFraudulentWebsiteDetection + +EnableFraudulentWebsiteDetection enables scan services for fraudulent content, such as malware or phishing attempts. These services might send information from your app like URLs navigated to and possibly other content to cloud services of Apple and Microsoft. + +Name: EnableFraudulentWebsiteDetection
Type: `bool` + +### ZoomFactor + +Name: ZoomFactor
Type: `float64` + +This defines the zoom factor for the WebView2. This is the option matching the Edge user activated zoom in or out. + +### IsZoomControlEnabled + +Name: IsZoomControlEnabled
Type: `bool` + +This enables the zoom factor to be changed by the user. Please note that the zoom factor can be set in the options while disallowing the user to change it at runtime (f.e. for a kiosk application or similar). + +### Bind + +A slice of struct instances defining methods that need to be bound to the frontend. + +Name: Bind
Type: `[]interface{}` + +### ErrorFormatter + +A function that determines how errors are formatted when returned by a JS-to-Go method call. The returned value will be marshalled as JSON. + +Name: ErrorFormatter
Type: `func (error) any` + +### Windows + +This defines [Windows specific options](#windows). + +Name: Windows
Type: `*windows.Options` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent). + +For Windows 11 versions before build 22621, this will use the [BlurBehind](https://learn.microsoft.com/en-us/windows/win32/dwm/blur-ovw) method for translucency, which can be slow. For Windows 11 versions after build 22621, this will enable the newer translucency types that are much faster. By default, the type of translucency used will be determined by Windows. To configure this, use the [BackdropType](#BackdropType) option. + +Name: WindowIsTranslucent
Type: `bool` + +#### BackdropType + +:::note + +Requires Windows 11 build 22621 or later. + +::: + +Sets the translucency type of the window. This is only applicable if [WindowIsTranslucent](#WindowIsTranslucent) is set to `true`. + +Name: BackdropType
Type `windows.BackdropType` + +The value can be one of the following: + +| Value | Description | +| ------- | ----------------------------------------------------------------------------------------- | +| Auto | Let Windows decide which backdrop to use | +| None | Do not use translucency | +| Acrylic | Use [Acrylic](https://learn.microsoft.com/en-us/windows/apps/design/style/acrylic) effect | +| Mica | Use [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) effect | +| Tabbed | Use Tabbed. This is a backdrop that is similar to Mica. | + +#### DisableWindowIcon + +Setting this to `true` will remove the icon in the top left corner of the title bar. + +Name: DisableWindowIcon
Type: `bool` + +#### DisableFramelessWindowDecorations + +Setting this to `true` will remove the window decorations in [Frameless](#Frameless) mode. This means there will be no 'Aero Shadow' and no 'Rounded Corners' shown for the window. Please note that 'Rounded Corners' are only supported on Windows 11. + +Name: DisableFramelessWindowDecorations
Type: `bool` + +#### WebviewUserDataPath + +This defines the path where the WebView2 stores the user data. If empty `%APPDATA%\[BinaryName.exe]` will be used. + +Name: WebviewUserDataPath
Type: `string` + +#### WebviewBrowserPath + +This defines the path to a directory with WebView2 executable files and libraries. If empty, webview2 installed in the system will be used. + +Important information about distribution of fixed version runtime: + +- [How to get and extract runtime](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [Known issues for fixed version](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [The path of fixed version of the WebView2 Runtime should not contain \Edge\Application\.](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +Name: WebviewBrowserPath
Type: `string` + +#### Theme + +Minimum Windows Version: Windows 10 2004/20H1 + +This defines the theme that the application should use: + +| Value | Description | +| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------- | +| SystemDefault | _Default_. The theme will be based on the system default. If the user changes their theme, the application will update to use the new setting | +| Dark | The application will use a dark theme exclusively | +| Light | The application will use a light theme exclusively | + +Name: Theme
Type: `windows.Theme` + +#### CustomTheme + +:::note + +Minimum Windows Version: Windows 10/11 2009/21H2 Build 22000 + +::: + +Allows you to specify custom colours for TitleBar, TitleText and Border for both light and dark mode, as well as when the window is active or inactive. + +Name: CustomTheme
Type: `windows.CustomTheme` + +##### CustomTheme type + +The CustomTheme struct uses `int32` to specify the colour values. These are in the standard(!) Windows format of: `0x00BBGGAA`. A helper function is provided to do RGB conversions into this format: `windows.RGB(r,g,b uint8)`. + +NOTE: Any value not provided will default to black. + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +Example: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### Messages + +A struct of strings used by the webview2 installer if a valid webview2 runtime is not found. + +Name: Messages
Type: `*windows.Messages` + +Customise this for any language you choose to support. + +#### ResizeDebounceMS + +ResizeDebounceMS is the amount of time to debounce redraws of webview2 when resizing the window. The default value (0) will perform redraws as fast as it can. + +Name: ResizeDebounceMS
Type: `uint16` + +#### OnSuspend + +If set, this function will be called when Windows initiates a switch to low power mode (suspend/hibernate) + +Name: OnSuspend
Type: `func()` + +#### OnResume + +If set, this function will be called when Windows resumes from low power mode (suspend/hibernate) + +Name: OnResume
Type: `func()` + +#### WebviewGpuIsDisabled + +Setting this to `true` will disable GPU hardware acceleration for the webview. + +Name: WebviewGpuIsDisabled
Type: `bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +This defines [Mac specific options](#mac). + +Name: Mac
Type: `*mac.Options` + +#### TitleBar + +The TitleBar struct provides the ability to configure the look and feel of the title bar. + +Name: TitleBar
Type: [`*mac.TitleBar`](#titlebar-struct) + +##### Titlebar struct + +The titlebar of the application can be customised by using the TitleBar options: + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| Name | Description | +| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| TitlebarAppearsTransparent | Makes the titlebar transparent. This has the effect of hiding the titlebar and the content fill the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | Hides the title of the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | Removes [NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/) from the style mask | +| FullSizeContent | Makes the webview fill the entire window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | Adds a default toolbar to the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | Removes the line beneath the toolbar. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +Preconfigured titlebar settings are available: + +| Setting | Example | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +Example: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +Click [here](https://github.com/lukakerr/NSWindowStyles) for some inspiration on customising the titlebar. + +#### Appearance + +Appearance is used to set the style of your app in accordance with Apple's [NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc) names. + +Name: Appearance
Type: [`mac.AppearanceType`](#appearance-type) + +##### Appearance type + +You can specify the application's [appearance](https://developer.apple.com/documentation/appkit/nsappearance?language=objc). + +| Value | Description | +| ----------------------------------------------------- | --------------------------------------------------------------- | +| DefaultAppearance | DefaultAppearance uses the default system value | +| NSAppearanceNameAqua | The standard light system appearance | +| NSAppearanceNameDarkAqua | The standard dark system appearance | +| NSAppearanceNameVibrantLight | The light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastAqua | A high-contrast version of the standard light system appearance | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | A high-contrast version of the standard dark system appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | A high-contrast version of the light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | A high-contrast version of the dark vibrant appearance | + +Example: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent) to make frosty-looking applications. + +Name: WindowIsTranslucent
Type: `bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### About + +This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. + +Name: About
Type: [`*mac.AboutInfo`](#about-struct) + +##### About struct + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +If these settings are provided, an "About" menu item will appear in the app menu (when using the `AppMenu` role). Given this configuration: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +The "About" menu item will appear in the app menu: + +```mdx-code-block +
+ +
+
+``` + +When clicked, that will open an about message box: + +```mdx-code-block +
+ +
+
+``` + +### Linux + +This defines [Linux specific options](#linux). + +Name: Linux
Type: `*linux.Options` + +#### Icon + +Sets up the icon representing the window. This icon is used when the window is minimized (also known as iconified). + +Name: Icon
Type: `[]byte` + +Some window managers or desktop environments may also place it in the window frame, or display it in other contexts. On others, the icon is not used at all, so your mileage may vary. + +NOTE: Gnome on Wayland at least does not display this icon. To have a application icon there, a `.desktop` file has to be used. On KDE it should work. + +The icon should be provided in whatever size it was naturally drawn; that is, don’t scale the image before passing it. Scaling is postponed until the last minute, when the desired final size is known, to allow best quality. + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Some window managers may ignore it, or result in a black window. + +Name: WindowIsTranslucent
Type: `bool` + +#### WebviewGpuPolicy + +This option is used for determining the webview's hardware acceleration policy. + +Name: WebviewGpuPolicy
Type: [`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
Default: `WebviewGpuPolicyAlways` + +##### WebviewGpuPolicy type + +| Value | Description | +| ------------------------ | -------------------------------------------------------------------- | +| WebviewGpuPolicyAlways | Hardware acceleration is always enabled | +| WebviewGpuPolicyOnDemand | Hardware acceleration is enabled/disabled as request by web contents | +| WebviewGpuPolicyNever | Hardware acceleration is always disabled | + +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` property differs form the executable's filename. + +Name: ProgramName
Type: string
+ +### Debug + +This defines [Debug specific options](#Debug) that apply to debug builds. + +Name: Debug
Type: `options.Debug` + +#### OpenInspectorOnStartup + +Setting this to `true` will open the WebInspector on startup of the application. + +Name: OpenInspectorOnStartup
Type: `bool` + +[^1]: This requires WebKit2GTK 2.36+ support and your app needs to be build with the build tag `webkit2_36` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.36 for your app. +[^2]: This requires WebKit2GTK 2.40+ support and your app needs to be build with the build tag `webkit2_40` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.40 for your app. diff --git a/website/versioned_docs/version-v2.5.0/reference/project-config.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx similarity index 99% rename from website/versioned_docs/version-v2.5.0/reference/project-config.mdx rename to website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx index 5c7d578b777..8e763502bcc 100644 --- a/website/versioned_docs/version-v2.5.0/reference/project-config.mdx +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx @@ -6,7 +6,7 @@ sidebar_position: 5 The project config resides in the `wails.json` file in the project directory. The structure of the config is: -```json +```json5 { // Project config version "version": "", diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx new file mode 100644 index 00000000000..b8eef677b92 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# 브라우저 + +이 메소드들은 시스템 브라우저와 관련 있습니다. + +### BrowserOpenURL + +시스템 브라우저에서 URL을 엽니다. + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..805f68ed917 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# Clipboard + +This part of the runtime provides access to the operating system's clipboard.
The current implementation only handles text. + +### ClipboardGetText + +This method reads the currently stored text from the clipboard. + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
Returns: a string (if the clipboard is empty an empty string will be returned) or an error. + +JS: `ClipboardGetText(): Promise`
Returns: a promise with a string result (if the clipboard is empty an empty string will be returned). + +### ClipboardSetText + +This method writes a text to the clipboard. + +Go: `ClipboardSetText(ctx context.Context, text string) error`
Returns: an error if there is any. + +JS: `ClipboardSetText(text: string): Promise`
Returns: a promise with true result if the text was successfully set on the clipboard, false otherwise. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..9b3e6cf145c --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# Dialog + +This part of the runtime provides access to native dialogs, such as File Selectors and Message boxes. + +:::info JavaScript + +Dialog is currently unsupported in the JS runtime. + +::: + +### OpenDirectoryDialog + +Opens a dialog that prompts the user to select a directory. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Returns: Selected directory (blank if the user cancelled) or an error + +### OpenFileDialog + +Opens a dialog that prompts the user to select a file. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Returns: Selected file (blank if the user cancelled) or an error + +### OpenMultipleFilesDialog + +Opens a dialog that prompts the user to select multiple files. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +Returns: Selected files (nil if the user cancelled) or an error + +### SaveFileDialog + +Opens a dialog that prompts the user to select a filename for the purposes of saving. Can be customised using [SaveDialogOptions](#savedialogoptions). + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +Returns: The selected file (blank if the user cancelled) or an error + +### MessageDialog + +Displays a message using a message dialog. Can be customised using [MessageDialogOptions](#messagedialogoptions). + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +Returns: The text of the selected button or an error + +## Options + +### OpenDialogOptions + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| Field | Description | Win | Mac | Lin | +| -------------------------- | ---------------------------------------------- | --- | --- | --- | +| DefaultDirectory | The directory the dialog will show when opened | ✅ | ✅ | ✅ | +| DefaultFilename | The default filename | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | A list of file filters | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Show files hidden by the system | | ✅ | ✅ | +| CanCreateDirectories | Allow user to create directories | | ✅ | | +| ResolvesAliases | If true, returns the file not the alias | | ✅ | | +| TreatPackagesAsDirectories | Allow navigating into packages | | ✅ | | + +### SaveDialogOptions + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| Field | Description | Win | Mac | Lin | +| -------------------------- | ---------------------------------------------- | --- | --- | --- | +| DefaultDirectory | The directory the dialog will show when opened | ✅ | ✅ | ✅ | +| DefaultFilename | The default filename | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | A list of file filters | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Show files hidden by the system | | ✅ | ✅ | +| CanCreateDirectories | Allow user to create directories | | ✅ | | +| TreatPackagesAsDirectories | Allow navigating into packages | | ✅ | | + +### MessageDialogOptions + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| Field | Description | Win | Mac | Lin | +| ------------- | -------------------------------------------------------------------------- | -------------- | --- | --- | +| Type | The type of message dialog, eg question, info... | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| Message | The message to show the user | ✅ | ✅ | ✅ | +| Buttons | A list of button titles | | ✅ | | +| DefaultButton | The button with this text should be treated as default. Bound to `return`. | ✅[*](#windows) | ✅ | | +| CancelButton | The button with this text should be treated as cancel. Bound to `escape` | | ✅ | | + +#### Windows + +Windows has standard dialog types in which the buttons are not customisable. The value returned will be one of: "Ok", "Cancel", "Abort", "Retry", "Ignore", "Yes", "No", "Try Again" or "Continue". + +For Question dialogs, the default button is "Yes" and the cancel button is "No". This can be changed by setting the `DefaultButton` value to `"No"`. + +Example: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linux has standard dialog types in which the buttons are not customisable. The value returned will be one of: "Ok", "Cancel", "Yes", "No" + +#### Mac + +A message dialog on Mac may specify up to 4 buttons. If no `DefaultButton` or `CancelButton` is given, the first button is considered default and is bound to the `return` key. + +For the following code: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, +}) +``` + +the first button is shown as default: + +```mdx-code-block +
+ +
+
+``` + +And if we specify `DefaultButton` to be "two": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +the second button is shown as default. When `return` is pressed, the value "two" is returned. + +```mdx-code-block +
+ +
+
+``` + +If we now specify `CancelButton` to be "three": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +the button with "three" is shown at the bottom of the dialog. When `escape` is pressed, the value "three" is returned: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### DialogType + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### FileFilter + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +Windows allows you to use multiple file filters in dialog boxes. Each FileFilter will show up as a separate entry in the dialog: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linux allows you to use multiple file filters in dialog boxes. Each FileFilter will show up as a separate entry in the dialog: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +Mac dialogs only have the concept of a single set of patterns to filter files. If multiple FileFilters are provided, Wails will use all the Patterns defined. + +Example: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +This will result in the Open File dialog using `*.png,*.jpg,*.mov,*.mp4` as a filter. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx new file mode 100644 index 00000000000..856ba6f0c26 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# Events + +The Wails runtime provides a unified events system, where events can be emitted or received by either Go or JavaScript. Optionally, data may be passed with the events. Listeners will receive the data in the local data types. + +### EventsOn + +This method sets up a listener for the given event name. When an event of type `eventName` is [emitted](#EventsEmit), the callback is triggered. Any additional data sent with the emitted event will be passed to the callback. It returns a function to cancel the listener. + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff + +This method unregisters the listener for the given event name, optionally multiple listeneres can be unregistered via `additionalEventNames`. + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce + +This method sets up a listener for the given event name, but will only trigger once. It returns a function to cancel the listener. + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple + +This method sets up a listener for the given event name, but will only trigger a maximum of `counter` times. It returns a function to cancel the listener. + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit + +This method emits the given event. Optional data may be passed with the event. This will trigger any event listeners. + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx new file mode 100644 index 00000000000..3cacb6ba9a7 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# 소개 + +런타임은 애플리케이션에 유틸리티 메서드를 제공하는 라이브러리입니다. There is both a Go and JavaScript runtime and the aim is to try and keep them at parity where possible. + +다음에 대한 유틸리티 메서드가 있습니다: + +- [Window](window.mdx) +- [Menu](menu.mdx) +- [Dialog](dialog.mdx) +- [Events](events.mdx) +- [Browser](browser.mdx) +- [Log](log.mdx) +- [Clipboard](clipboard.mdx) + +Go 런타임은 `github.com/wailsapp/wails/v2/pkg/runtime` 가져오기를 통해 사용할 수 있습니다. 이 패키지의 모든 메소드는 컨텍스트를 첫 번째 매개변수로 사용합니다. 이 컨텍스트는 [OnStartup](../options.mdx#onstartup) 또는 [OnDomReady](../options.mdx#ondomready) 후크에서 가져와야 합니다. + +:::참고 + +컨텍스트는 [OnStartup](../options.mdx#onstartup) 메서드를 사용할 경우 런타임이 이 메서드에서 다음과 같이 작동할 것이라는 보장은 없습니다. 창이 다른 스레드에서 초기화되고 있습니다. 시작 시 런타임 메서드를 호출하려면 [OnDomReady](../options.mdx#ondomready)를 사용하세요. + +::: + +The JavaScript library is available to the frontend via the `window.runtime` map. There is a runtime package generated when using `dev` mode that provides TypeScript declarations for the runtime. 이것은 프론트엔드 디렉토리의 `wailsjs` 디렉토리에 있어야 합니다. + +### Hide + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +애플리케이션을 숨깁니다. + +:::참고 + +On Mac, this will hide the application in the same way as the `Hide` menu item in standard Mac applications. This is different to hiding the window, but the application still being in the foreground. For Windows and Linux, this is currently the same as `WindowHide`. + +::: + +### Show + +Shows the application. + +:::참고 + +On Mac, this will bring the application back into the foreground. For Windows and Linux, this is currently the same as `WindowShow`. + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### Quit + +Quits the application. + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### Environment + +Returns details of the current environment. + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### EnvironmentInfo + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx new file mode 100644 index 00000000000..2112e1c9e4f --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# Log + +The Wails runtime provides a logging mechanism that may be called from Go or JavaScript. 대부분처럼 loggers 에는 다양한 로그 수준이 있습니다: + +- Trace +- Debug +- Info +- Warning +- Error +- Fatal + +Logger는 현재 이상의 로그 수준에서 모든 로그 메시지를 출력합니다. Example: `Debug` 로그 level은 `Trace` 메시지를 제외한 모든 메시지를 출력합니다. + +### LogPrint + +주어진 메시지를 원본 메시지로 기록합니다. + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf + +주어진 메시지를 원본 메시지로 기록합니다. + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace + +`Trace` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef + +`Trace` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug + +`Debug` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf + +`Debug` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo + +`Info` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof + +`Info` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning + +`Warning` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf + +`Warning` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError + +`Error` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf + +`Error` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal + +`Fatal` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf + +`Fatal` 로그 수준에서 지정된 메시지를 기록합니다. + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel + +로그 레벨을 설정합니다. In JavaScript, the number relates to the following log levels: + +| 값 | 로그 레벨 | +| - | ------- | +| 1 | Trace | +| 2 | Debug | +| 3 | Info | +| 4 | Warning | +| 5 | Error | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## 커스텀 Logger 사용하기 + +사용자 커스텀 logger는 [Logger](../options.mdx#logger)를 사용하여 제공하여 사용할 수 있습니다. 응용 프로그램 옵션. 유일한 요구 사항은 logger가 `logger.Logger` 인터페이스를 구현한다는 것입니다. 이는 `github.com/wailsapp/wails/v2/pkg/logger`에 정의되어 있습니다: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx new file mode 100644 index 00000000000..0eab71c0162 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# 메뉴 + +애플리케이션 메뉴는 이 메소드들과 관련 있습니다. + +:::info JavaScript + +현재 메뉴는 JS runtime을 지원하지 않습니다. + +::: + +### MenuSetApplicationMenu + +애플리케이션 메뉴는 [menu](../menus.mdx)를 통해 설정하세요. + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu + +애플리케이션 메뉴를 업데이트하여 `MenuSetApplicationMenu`에 전달된 메뉴에 대한 모든 변경 사항을 선택합니다. + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx new file mode 100644 index 00000000000..457c92ebff9 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# Screen + +These methods provide information about the currently connected screens. + +### ScreenGetAll + +Returns a list of currently connected screens. + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### Screen + +Go struct: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript interface: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx new file mode 100644 index 00000000000..8e1052289b9 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# Window + +이러한 메서드는 응용 프로그램 창을 제어합니다. + +### WindowSetTitle + +창 제목 표시줄의 텍스트를 설정합니다. + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen + +창을 전체화면으로 만듭니다. + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen + +전체 화면 이전의 이전 창 크기와 위치를 복원합니다. + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen + +창이 전체 화면이면 true를 반환합니다. + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter + +창이 현재 켜져 있는 모니터의 중앙에 창을 맞춥니다. + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS + +창에서 임의의 JS 코드를 실행합니다. + +이 메서드는 브라우저에서 코드를 비동기적으로 실행하고 즉시 반환합니다. 만약 스크립트에서 오류가 발생하면 브라우저 콘솔에서만 사용할 수 있습니다. + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload + +"새로고침"을 수행합니다(현재 페이지 새로고침). + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp + +프론트엔드 애플리케이션을 새로고침합니다. + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme + +윈도우 전용. + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +창의 테마를 시스템 기본값으로 설정합니다(다크/라이트). + +### WindowSetLightTheme + +윈도우 전용. + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +창 테마를 라이트 모드로 설정합니다. + +### WindowSetDarkTheme + +윈도우 전용. + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +윈도우 테마를 다크로 설정합니다. + +### WindowShow + +만약 현재 창의 상태가 숨기기로 되어 있다면, 창을 보여줍니다. + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowHide + +현재 창이 표시되어 있는 경우 창을 숨깁니다. + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal + +창이 최소화, 최대화 또는 전체 화면이 아닌 경우 true를 반환합니다. + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize + +창의 너비와 높이를 설정합니다. + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowGetSize + +창의 너비와 높이를 가져옵니다. + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize + +최소 창 크기를 설정합니다. 창이 현재 지정된 크기보다 작은 경우 창 크기를 조정합니다. + +`0,0`의 크기를 설정하면 이 제약 조건이 비활성화됩니다. + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize + +최대 창 크기를 설정합니다. 창이 현재 지정된 크기보다 큰 경우 창 크기를 조정합니다. + +`0,0`의 크기를 설정하면 이 제약 조건이 비활성화됩니다. + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop + +창을 항상위 또는 맨 위에 놓지 않도록 설정합니다. + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition + +창이 현재 켜져 있는 모니터를 기준으로 창 위치를 설정합니다. + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition + +창이 현재 있는 모니터에 상대적인 창 위치를 가져옵니다. + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximise + +창을 최대화하여 화면을 채웁니다. + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise + +창을 최대화하기 전의 크기와 위치로 복원합니다. + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised + +창이 최대화되면 true를 반환합니다. + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise + +최대화와 최대화 해제 사이를 전환합니다. + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise + +창을 최소화합니다. + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise + +최소화하기 전의 크기와 위치로 창을 복원합니다. + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised + +창이 최소화된 경우 true를 반환합니다. + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour + +창의 배경색을 지정된 RGBA 색상 정의로 설정합니다. 이 색상은 모든 투명 픽셀에 대해 표시됩니다. + +R, G, B 및 A의 유효한 값은 0-255입니다. + +:::info Windows + +Windows에서는 0 또는 255의 알파 값만 지원됩니다. 0이 아닌 모든 값은 255로 간주됩니다. + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## TypeScript Object Definitions + +### 위치 + +```ts +interface Position { + x: number; + y: number; +} +``` + +### 크기 + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..98129e5b753 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# Dogs API + +```mdx-code-block +
+ +
+
+``` + +:::note + +이 튜토리얼은 [@tatadan](https://twitter.com/tatadan)에 의해 제공되었습니다. 그리고 [ Wails 예시 리포지토리](https://github.com/tataDan/wails-v2-examples)에서 예시를 확인할 수 있습니다. + +::: + +이 튜토리얼에서는 웹에서 개 사진을 검색한 다음 표시하는 애플리케이션을 개발할 것입니다. + +### 프로젝트 생성 + +애플리케이션을 생성합니다. 터미널에 다음과 같이 입력합니다: `wails init -n dogs-api -t svelte` + +참고: 원하는 경우 선택적으로 이 명령의 끝에 `-ide vscode` 또는 `-ide goland`를 추가할 수 있습니다. IDE 지원을 추가합니다. + +이제 `cd dogs-api`를 만들고 프로젝트 파일 편집을 시작합니다. + +### 사용하지 않는 코드 제거 + +사용하지 않는 몇가지 요소를 제거합니다. + +- `app.go`를 열고 다음 줄을 제거합니다. + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- `frontend/src/App.svelte`를 열고 모든 줄을 삭제합니다. +- `frontend/src/assets/images/logo-universal.png` 파일을 삭제합니다. + +### 우리만의 애플리케이션 생성하기 + +이제 새로운 Go 코드를 추가합니다. + +함수 정의 앞에 다음 구조체 선언을 `app.go`에 추가합니다. + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +`app.go`에 다음 함수를 추가합니다(함수 정의 뒤쪽에 존재): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +`app.go`의 `import` 섹션을 다음과 같이 수정합니다. + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +`frontend/src/App.svelte`에 다음 줄을 추가합니다. + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### 애플리케이션 테스트하기 + +바인딩을 생성하고 애플리케이션을 테스트하려면 `wails dev`를 실행합니다. + +### 애플리케이션 컴파일하기 + +응용 프로그램을 단일 바이너리로 컴파일하려면 `wails build`를 실행합니다. diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx new file mode 100644 index 00000000000..4bc3bcaafc2 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# Hello World + +이 튜토리얼의 목표는 Wails를 사용하여 가장 기본적인 애플리케이션을 시작하고 실행하는 것입니다. 튜토리얼을 통해 다음과 같은 것들을 할 수있습니다. + +- 새로운 Wails 애플리케이션 생성 +- 애플리케이션 빌드 +- 애플리케이션 실행 + +:::note + +이 튜토리얼에서는 Windows를 대상 플랫폼으로 사용합니다. 출력이 약간 다를 수 있습니다. 운영체제에 따라 다릅니다. + +::: + +## 새로운 Wails 애플리케이션 생성 + +기본 바닐라 JS 템플릿을 사용하여 새 Wails 애플리케이션을 만들려면, 다음 명령을 실행해야 합니다: + +```bash +wails init -n helloworld +``` + +실행 결과로 다음과 유사한 결과가 반환됩니다. + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +이렇게 하면 현재 디렉터리에 `helloworld`라는 새 디렉터리가 생성됩니다. 이 디렉토리에는 여러 파일이 있습니다: + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## 애플리케이션 빌드 + +애플리케이션을 빌드하려면 새 `helloworld` 프로젝트 디렉토리로 변경하고 다음 명령을 실행하십시오. + +```bash +wails build +``` + +다음과 같은 내용이 표시되어야 합니다: + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +이것은 애플리케이션을 컴파일하고 `build/bin` 디렉토리에 결과 실행파일을 저장합니다. + +## 애플리케이션 실행 + +Windows 탐색기에서 `build/bin` 디렉토리를 보면 프로젝트 바이너리가 표시되어야 합니다. + +```mdx-code-block +
+ +
+
+``` + +`helloworld.exe` 파일을 두 번 클릭하면 실행됩니다. + +Mac에서 Wails는 두 번 클릭하여 실행할 수 있는 `helloworld.app` 파일을 생성합니다. + +Linux에서는 `build/bin` 디렉토리에서 `./helloworld`를 사용하여 애플리케이션을 실행할 수 있습니다. + +애플리케이션이 예상대로 동작하는지 확인해야 합니다. + +```mdx-code-block +
+ +
+
+``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx new file mode 100644 index 00000000000..006ff61ed06 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# Links + +Esta página serve como uma lista para links da comunidade relacionados. Envie um PR (clique `Editar esta página` na parte inferior) para enviar links. + +## Awesome Wails + +A [lista definitiva](https://github.com/wailsapp/awesome-wails) dos links relacionados à Wails. + +## Canais de Suporte + +- [Servidor do Discord Wails](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [v2 Beta Discussion Board](https://github.com/wailsapp/wails/discussions/828) + +## Redes Sociais + +- [Twitter](https://twitter.com/wailsapp) +- [Wails Chinese Community QQ Group](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - Group number: 1067173054 + +## Outros tutoriais e Artigos + +- [Building of Bulletin Board](https://blog.customct.com/building-bulletin-board) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..6373a7ff16e --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +

+ +
+

+``` + +O aplicativo [BulletinBoard](https://github.com/raguay/BulletinBoard) é um quadro de mensagens versitais para mensagens estáticas ou diálogos para obter informações do usuário para um script. Ele tem uma TUI para criar novas caixas de diálogo que podem ser usadas por último para obter informações do usuário. É um design que fica em execução no seu sistema e mostra as informações conforme necessário e depois se esconde. Tenho um processo para assistir um arquivo no meu sistema e enviar o conteúdo para o BulletinBoard quando alterado. Funciona bem com meus fluxos de trabalho. Há também um [fluxo de trabalho Alfred](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) para enviar informações ao programa. O fluxo de trabalho também é para trabalhar com [EmailIt](https://github.com/raguay/EmailIt). diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx new file mode 100644 index 00000000000..862c304e973 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/) é um programa Wails 2 que é um remetente de e-mail baseado em markdown com apenas nove notepads, scripts para manipular o texto e templates. Ele também tem um terminal para rodar scripts em EmailIt em arquivos do seu sistema. Os scripts e modelos podem ser usados na própria linha de comando ou com as extensões Alfred, Keyboard Maestro, Dropzone ou PopClip. Ele também suporta scripts e temas baixados no formulário GitHub. A documentação não está completa, mas os programas funcionam. É construído usando Wails2 e Svelte, e o download é um aplicativo macOS universal. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..a5692d35388 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app) é uma ferramenta simples e fácil de usar de criptografia PGP, gerenciando todas as suas chaves de contato e suas chaves. A criptografia deve ser simples. Desenvolvido com Wails.** + +Criptografar mensagens usando PGP é o padrão da indústria. Todos têm uma chave privada e pública. A sua chave privada precisa ser privada para que apenas você possa ler as mensagens. Sua chave pública é distribuída para qualquer pessoa que queira lhe enviar o segredo, mensagens criptografadas. Gerenciar chaves, criptografar mensagens e descriptografar mensagens devem ser uma experiência suave. EncryptEasy é sobre facilitar o processo. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx new file mode 100644 index 00000000000..0b5d7fc36ba --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# Utilitário de Exportação FileHound + +```mdx-code-block +

+ +
+

+``` + +[Utilitário de exportação de FileHound](https://www.filehound.co.uk/) FileHound é uma plataforma de gerenciamento de documentos em nuvem feita para retenção segura de arquivos, automação de processos de negócio e recursos de SmartCaptura. + +O Utilitário de Exportação FileHound permite os admnistradores FileHound a capacidade de executar um documento seguro e extração de tarefas para um backup alternativo. Esta aplicação irá baixar todos os documentos e/ou meta dados salvos no FileHound com base nos filtros que você escolher. Os metadados serão exportados em ambos os formatos JSON e XML. + +Backend construído com: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +Frontend com: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..13ed9dc7c53 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter) é uma ferramenta simples e eficiente de teste de cliente http API. Baseado em Wails, Go and sveltejs. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..d606f87f961 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Atualizador do Minecraft + +```mdx-code-block +

+ +
+

+``` + +[O atualziador do Minecraft](https://github.com/Gurkengewuerz/MinecraftModUpdater) é uma ferramenta utilitária para atualizar e sincronizar mods do Minecraft para sua base de usuários. Ele é construído usando Wails2 e React com [antd](https://ant.design/) como framework frontend. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..9633a7d4a84 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Gerenciador de arquivos Modal + +```mdx-code-block +

+ +
+

+``` + +[Gerenciador de Arquivos Modal](https://github.com/raguay/ModalFileManager) é um gerenciador de arquivos dual pane usando tecnologias da web. Meu design original foi baseado em NW.js e pode ser encontrado [aqui](https://github.com/raguay/ModalFileManager-NWjs). Esta versão usa o mesmo código de frontend baseado em Svelte (mas foi muito modificado desde a partida do NW.), mas o backend implementado com [Wails 2](https://wails.io/). Ao usar esta implementação, não uso mais comandos de linha de comando `rm`, `cp`, etc., mas um git install deve estar no sistema para baixar temas e extensões. Está totalmente codificado usando Go e roda muito mais rápido do que as versões anteriores. + +Este gerenciador de arquivos é projetado em torno do mesmo princípio do Vim: uma ação controlada pelo teclado. O número de estados não é fixo, mas muito programável. Portanto, um número infinito de configurações de teclado pode ser criado e usado. Esta é a principal diferença em relação a outros gerenciadores de arquivos. Existem temas e extensões disponíveis para download do GitHub. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..a0570d03546 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molley Wallet](https://github.com/grvlle/constellation_wallet/) a carteira oficial $DAG da rede Constellation. Ele permitirá que os usuários interajam com a Rede Hypergraph de várias maneiras, não limitado à produção de $DAG transações. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx new file mode 100644 index 00000000000..0069bae7230 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[Constellation ](https://october.utf9k.net) é uma pequena aplicação de Wails que facilita muito a extração de destaques de [Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader) e então os encaminha para [Readwise](https://readwise.io). + +Tem um escopo relativamente pequeno com todas as versões de plataforma pesando abaixo de 10MB, e isso é sem habilitar [a compressão UPX](https://upx.github.io/)! + +Em contraste, as tentativas anteriores do autor com o Electron rapidamente incharam várias centenas de megabytes. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx new file mode 100644 index 00000000000..606d5c4a87f --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# Optimus + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus) é uma aplicação de otimização de imagem para área de trabalho. Ele suporta conversão e compressão entre formatos de imagem WebP, JPEG e PNG. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx new file mode 100644 index 00000000000..4db08021efe --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# Portfall + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - Um portal de encaminhamento de porta para desktop para fácil acesso a todas as suas interfaces diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..fbe86fd6db5 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - Uma GUI de backup simples e multiplataforma [restica](https://github.com/restic/restic) para navegação e restauração de repositórios resóticos. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..961debade5f --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +Compartilhamento de arquivos fácil, seguro e gratuito para todos. Saiba mais em [Riftshare.app](https://riftshare.app) + +## Funcionalidades + +- Compartilhamento fácil e seguro de arquivos entre computadores tanto na rede local quanto através da internet +- Suporta o envio de arquivos ou diretórios de forma segura através do [magic wormhole protocol](https://magic-wormhole.readthedocs.io/en/latest/) +- Compatível com todos os outros aplicativos que usam magic wormhole (magic-wormhole ou wormhole-william CLI, wormhole-gui, etc.) +- Compactação automática de vários arquivos selecionados para enviar de uma vez +- Animações completas, barra de progresso e cancelamento do suporte para envio e recebimento +- Seleção de arquivo Native OS +- Abrir arquivos em um clique uma vez recebido +- Atualização Automática - não se preocupe em ter a versão mais recente! diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..335e360e37a --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) é um programa que mostra a saída dos scripts ou o servidor [Node-Red](https://nodered.org). Ele executa scripts definidos no programa EmailIt e mostra a saída. Scripts de xBar ou TextBar podem ser usados, mas atualmente em scripts da TextBar funcionam bem. Também exibe a saída de scripts no seu sistema. ScriptBar não os coloca na barra de menus, mas os tenha todos em uma janela convidada para fácil visualização. Você pode ter várias abas para ter muitas coisas diferentes mostradas. Você também pode manter os links para os sites mais visitados. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx new file mode 100644 index 00000000000..78db0e7b460 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/) é um aplicativo de compartilhamento de arquivos p2p projetado para utilizar tecnologias blockchain para habilitar transferências de arquivos 100% anônimas. O Surgeé criptografado, descentralizado e de código aberto. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx new file mode 100644 index 00000000000..8b2f2c6c4fb --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally) é o flasher oficial para os teclados [Ergodox](https://ergodox-ez.com/). Parece excelente e é um exemplo fantástico do que você pode conseguir com o Wails: a capacidade de combinar o poder do Go e as ricas ferramentas gráficas do mundo de desenvolvimento web. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx new file mode 100644 index 00000000000..c56880f0e19 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# Launcher Minecraft para WarMine + +```mdx-code-block +

+ + +
+

+``` + +[Minecraft Launcher para WarMine](https://warmine.ru/) é uma aplicação Wails que permite que você entre nos servidores de jogos modificados facilmente e gerencie suas contas de jogo. + +O Launcher baixa os arquivos do jogo, verifica sua integridade e lança o jogo com uma grande variedade de opções de personalização para os argumentos de lançamento pelo backend. + +Frontend está escrito em Svelte, todo o launcher se enquadra em 9MB e suporta Windows 7-11. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx new file mode 100644 index 00000000000..d2e5eaf3ee6 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# Wombat + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat) é um cliente gRPC multiplataforma. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx new file mode 100644 index 00000000000..01a56449817 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails) é um aplicativo para baixar faixas do youtube, criar listas de reprodução offline e compartilhar com seus amigos, seus amigos poderão reproduzir suas playlists ou baixá-las para escuta offline, tem um player embutido. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx new file mode 100644 index 00000000000..5fab64fd42a --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# Templates + +Esta página serve como uma lista para modelos suportados pela comunidade. Por favor, envie um PR (clique `Editar esta página` na parte inferior) para incluir seus modelos. Para construir seu próprio modelo, consulte o guia de [Modelos](../guides/templates.mdx). + +Para usar estes templates, execute `wails init -n "Seu Nome do Projeto" -t [o link abaixo[@version]]` + +Se não houver nenhum sufixo de versão, o modelo de código principal da branch é usado por padrão. Se houver um sufixo de versão, o template de código correspondente à tag desta versão é usado. + +Exemplo: `wails init -n "Seu Nome do Projeto" -t https://github.com/misitebao/wails-template-vue` + +:::warning Atenção + +**O projeto Wails não é mantido, não é responsável nem responsável por modelos de terceiros!** + +Se você não tiver certeza sobre um template, inspecione `package.json` e `wails.json` para quais scripts são executados e quais pacotes estão instalados. + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - Wails template based on Vue ecology (Integrated TypeScript, Dark theme, Internationalization, Single page routing, TailwindCSS) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Vue 3 TypeScript with Vite (and instructions to add features) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vue 3 TypeScript with Vite, Vuex, Vue Router, Sass, and ESLint + Prettier +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - Um template usando JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - Um modelo usando TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, API de composição com <configuração de script>) +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Modelo de ervas baseado em Naive UI (biblioteca de componentes Vue 3) + +## Angular + +- [A-modelo de wails-angular](https://github.com/mateothegreat/wails-template-angular) - ação Angular 15 + compactada & pronta para ser rolada para produção. +- [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - Angular with TypeScript, Sass, Hot-Reload, Code-Splitting and i18n + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - A template using reactjs +- [wails-react-template](https://github.com/flin7/wails-react-template) - A minimal template for React that supports live development +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - A template using Next.js and TypeScript +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - A template for React + TypeScript + Vite + TailwindCSS +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - Um modelo com Vite, React, TypeScript, TailwindCSS, e shadcn/ui + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - A template using Svelte +- [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - A template using Svelte and Vite +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - A template using Svelte and Vite with TailwindCSS v3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - Um modelo atualizado usando Svelte v4.2.0 e Vite com TailwindCSS v3.3.3 +- [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - A template using SvelteKit + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - Um modelo usando Sólido + Ts + Vite +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - Um modelo usando Sólid + Js + Vite + +## Elm + +- [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + +## Pure JavaScript (Vanilla) + +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx new file mode 100644 index 00000000000..d06faa9eab2 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# Compilando seu projeto + +No diretório do projeto, execute `wails builds`. Isso compilará seu projeto e salvará o binário pronto para produção no diretório `build/bin`. + +Se você executar o binário, você verá a aplicação padrão: + +```mdx-code-block +
+ +
+
+``` + +Para mais detalhes sobre as opções de compilação, consulte a [Referência do CLI](../reference/cli.mdx#build). diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx new file mode 100644 index 00000000000..be641d3b98d --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# Desenvolvendo sua aplicação + +Você pode executar sua aplicação no modo de desenvolvimento executando `wails dev` no diretório do seu projeto. Isso fará o seguinte: + +- Construa seu aplicativo e o execute +- Vincule seu código Go para o frontend para que ele possa ser chamado a partir de JavaScript +- Usando o poder do [Vite](https://vitejs.dev/), observará modificações em seus arquivos Go e reconstruir/re-executar na alteração +- Configure um [servidor web](http://localhost:34115) que irá servir seu aplicativo em um navegador. Isso permite usar suas extensões de navegador favoritas. Você pode até mesmo chamar seu código Go do console + +Para começar, execute `wails dev` no diretório do projeto. Mais informações sobre isso podem ser encontradas [aqui](../reference/cli.mdx#dev). + +Em breve: Tutorial diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..f98b71ee493 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# Criando um projeto + +## Geração de Projeto + +Agora que o CLI está instalado, você pode gerar um novo projeto usando o comando `wails init`. + +Escolha seu framework favorito: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Generate a Svelte project using JavaScript with:

+ + wails init -n myproject -t svelte + +If you would rather use TypeScript:
+ + wails init -n myproject -t svelte-ts + +
+ + Generate a React project using JavaScript with:

+ + wails init -n myproject -t react + +If you would rather use TypeScript:
+ + wails init -n myproject -t react-ts + +
+ + Generate a Vue project using JavaScript with:

+ + wails init -n myproject -t vue + +If you would rather use TypeScript:
+ + wails init -n myproject -t vue-ts + +
+ + Generate a Preact project using JavaScript with:

+ + wails init -n myproject -t preact + +If you would rather use TypeScript:
+ + wails init -n myproject -t preact-ts + +
+ + Generate a Lit project using JavaScript with:

+ + wails init -n myproject -t lit + +If you would rather use TypeScript:
+ + wails init -n myproject -t lit-ts + +
+ + Generate a Vanilla project using JavaScript with:

+ + wails init -n myproject -t vanilla + +If you would rather use TypeScript:
+ + wails init -n myproject -t vanilla-ts + +
+
+``` + +
+ +Há também [templates da comunidade](../community/templates.mdx) disponíveis que oferecem diferentes recursos e frameworks. + +Para ver as outras opções disponíveis, você pode executar `wails init -help`. Mais detalhes podem ser encontrados na [Referência da CLI](../reference/cli.mdx#init). + +## Layout do Projeto + +Os projetos Wails possuem o seguinte layout: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### Estrutura do projeto em execução + +- `/main.go` - A aplicação principal +- `/frontend/` - Arquivos do Frontend do projeto +- `/build/` - Diretório de compilação do projeto +- `/build/appicon.png` - O ícone da aplicação +- `/build/darwin/` - Arquivos do projeto específicos do Mac +- `/build/windows/` - Arquivos de projeto específicos do Windows +- `/wails.json` - A configuração do projeto +- `/go.mod` - O arquivo de módulos do Go +- `/go.sum` - O arquivo de checagem dos módulos do Go + +O diretório `frontend` não tem nada específico para o Wails e pode ser qualquer projeto frontend de sua escolha. + +O diretório `build` é usado durante o processo de compilação. Esses arquivos podem ser atualizados para personalizar suas compilações. Se arquivos forem removidos do diretório de compilação, as versões padrão serão restauradas. + +O nome do módulo padrão em `go.mod` é "changeme". Você deveria mudar isso para algo mais apropriado. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx new file mode 100644 index 00000000000..33400cdfdbf --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx @@ -0,0 +1,90 @@ +--- +sidebar_position: 1 +--- + +# Instalação + +## Plataformas Suportadas + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## Dependências + +O Wails tem várias dependências comuns que são necessárias antes da instalação: + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +Baixe o Go direto da [página de download do Go](https://go.dev/dl/). + +Certifique-se de seguir as [instruções oficiais de instalação do Go](https://go.dev/doc/install). Você também precisará garantir que sua `variável de ambiente PATH` também inclua o caminho para o seu diretório `~/go/bin`. Reinicie seu terminal e faça as seguintes verificações: + +- Verificar se o Go está instalado corretamente: `go version` +- Cheque se `~/go/bin` está nas suas variáveis de ambiente `echo $PATH | grep go/bin` + +### NPM + +Baixe o NPM da [Página de downloads do Node](https://nodejs.org/en/download/). É melhor usar o último lançamento, pois é isso que costumamos testar. + +Execute `npm --version` para verificar. + +## Dependências específicas da plataforma + +Você também precisará instalar as dependências específicas da plataforma: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wails requires that the xcode command line tools are installed. This can be + done by running xcode-select --install. + + + Wails requires that the WebView2 runtime is installed. Algumas instalações do Windows já terão isto instalado. Você pode conferir usando o comando do wails doctor. + + + Linux requires the standard gcc build tools plus libgtk3 and libwebkit. Ao invés de listar uma tonelada de comandos para diferentes distros, os Wails podem tentar determinar quais são os comandos de instalação para sua distribuição específica. Execute o wails doctor após a instalação para ser mostrado como instalar as dependencias. Se o seu gerenciador de destro/pacote não for suportado, por favor, consulte o guia Adicionar Distro do Linux. + + +``` + +## Dependências opcionais + +- [UPX](https://upx.github.io/) para comprimir suas aplicações. +- [NSIS](https://wails.io/docs/guides/windows-installer/) for generating Windows installers. + +## Instalando Wails + +Execute `instale github.com/wailsapp/wails/v2/cmd/wails@latest` para instalar a CLI do Wails. + +Note: If you get an error similar to this: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +please check you have Go 1.18+ installed: +```shell +go version +``` + +## Verificação do sistema + +Ao executar `wails doctor ` verificará quais dependências estão instaladas corretamente. Caso contrário, aconselhará sobre o que está em falta e ajudará a corrigir quaisquer problemas. + +## O comando `wail` parece estar faltando? + +Se o sistema está relatando que o comando `wail` está faltando, certifique-se de ter seguido o guia de instalação corretamente. Normalmente, isso significa que o diretório `go/bin` no diretório do seu usuário não está na variável do ambiente `PATH`. Você normalmente também precisará fechar e reabrir qualquer prompt de comando aberto para que as alterações no ambiente feitas pelo instalador sejam refletidas no prompt de comando. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx new file mode 100644 index 00000000000..cae0d7f95a5 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +Embora as Wails não tenham um modelo angular, é possível usar Angular com Wails. + +## Modo Desenvolvedor + +Para que o modo de desenvolvimento funcione com Angular, você precisa adicionar o seguinte ao seu `wails.json`: + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx new file mode 100644 index 00000000000..f81eae7fa8d --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx @@ -0,0 +1,215 @@ +# Desenvolvimento de Aplicações + +Não existem regras rígidas e rápidas para o desenvolvimento das aplicações com o "Wails”, mas existem algumas orientações básicas. + +## Configurar Aplicação + +O padrão usado pelos templates padrão é que `main.go` é usado para configurar e executar a aplicação, enquanto `app.go` é usado para definir a lógica da aplicação. + +O arquivo `app.go` definirá uma struct que tem 2 métodos que funcionam como hooks na aplicação principal: + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- O método `startup` é chamado assim que as Wails alocam os recursos de que precisa e é um bom lugar para a criação de recursos configurando ouvintes de eventos e qualquer outra coisa que o aplicativo precise na inicialização. É dado a ele um `context.Context` que é geralmente salvo em um campo struct. Este contexto é necessário para chamar o [runtime](../reference/runtime/intro.mdx). Se este método retornar um erro, o aplicativo será encerrado. No modo de desenvolvimento, o erro será exibido no console. + +- O método de `shutdown` será chamado pelo Wails logo no final do processo de encerramento. Este é um bom lugar para limpar memória e executar qualquer tarefa encerrada. + +O arquivo `main.go` geralmente consiste de uma única chamada para `wails.Run()`, que aceita a configuração da aplicação. O padrão usado pelos templates é o antes da chamada para `wails.Run()`, uma instância do struct definido no aplicativo `app. o` é criado e salvo em uma variável chamada `app`. Essa configuração é onde adicionamos os nossos callbacks: + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +Mais informações sobre os hooks do ciclo de vida da aplicação podem ser encontradas [aqui](../howdoesitwork.mdx#application-lifecycle-callbacks). + +## Métodos de vínculo + +É provável que você queira chamar métodos Go do frontend. Isso normalmente é feito adicionando métodos públicos ao a struct já definida no `app.go`: + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +Na configuração principal do aplicativo, a tecla `Bind` é onde podemos dizer aos Wails o que queremos vincular: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +Isso vinculará todos os métodos públicos em nosso `App` de construção (isso nunca vinculará os métodos de inicialização e desligação). + +### Lidando com contexto quando vincular várias structs + +Se você quiser vincular métodos para várias structs , mas deseja que cada structs mantenha uma referência ao contexto para que você possa usar as funções de tempo de execução, um bom padrão é passar o contexto do método `OnStartup` para suas instâncias de construção : + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +Mais informações sobre isso podem ser encontradas [aqui](../howdoesitwork.mdx#method-binding). + +## Menu da aplicação + +O Wails suporta adicionar um menu à sua aplicação. Isso é feito passando uma [struct](../reference/menus.mdx#menu) de menu para a configuração da aplicação. É comum o uso de um método que retorna um Menu, e ainda mais comum que seja um método na struct `App` usada para hooks de ciclo de vida. + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## Assets + +A grande coisa do jeito que o Wails v2 lida com assets é que não o faz! A única coisa que você precisa fazer no Wails é um `embed.FS`. A forma como se chega a isso depende inteiramente de você. Você pode usar arquivos html/css/js como o modelo vanilla. Você pode ter um sistema de compilação complicado, isso não importa. + +Quando `wails build` é executado, ele vai verificar o arquivo do projeto `wails.json` na raiz do projeto. Há duas chaves no arquivo do projeto que são lidas: + +- "frontend:install" +- "frontend:build" + +O primeiro, se dado, será executado no diretório `frontend` para instalar os módulos do Node. O segundo, se dado, será executado no diretório `frontend` para realizar o build do projeto frontend. + +Se essas duas chaves não são dadas, então as Wails não faz absolutamente nada com a frontend. É apenas esperando o `embed.FS`. + +### AssetsHandler + +Um aplicativo Wails v2 pode opcionalmente definir um `http.Handler` nas opções `options.App`, que permite o gancho do AssetServer criar arquivos em tempo real ou processe solicitações POST/PUT. Solicitações GET sempre são tratadas primeiramente pelos `assets` FS. Se o FS não encontrar o arquivo solicitado, a solicitação será encaminhada para `http.Handler` para servir. Qualquer requisição que não seja o GET será diretamente processada pelo `AssetsHandler` se especificado. Também é usar apenas o `AssetsHandler` por especificação é `nil` como a opção `Assets`. + +## Servidor de Desenvolvedor nativo + +Executar `wails dev` iniciará o servidor de desenvolvimento que também iniciará um observador de arquivos no seu diretório de projeto. Por padrão, se qualquer arquivo for alterado, verifica-se se era um arquivo da aplicação (padrão: `.go`, configurável com `-e` flag). Se foi, então ele irá reconstruir sua aplicação e reiniciá-la. Se o arquivo alterado estiver nos assets, emitirá uma recarga após um curto período de tempo. + +O servidor de desenvolvimento utiliza uma técnica chamada "debwaring" que significa que não recarrega imediatamente, como pode haver vários arquivos alterados em um curto período de tempo. Quando um gatilho ocorre, ele espera por uma quantidade de tempo definida antes de emitir um recarregamento. Se outro gatilho acontecer, ele será redefinido para esperar novamente. Por padrão, esse valor é de `100ms`. Se este valor não funcionar para o seu projeto, ele pode ser configurado usando a flag `-debounce`. Se usado, este valor será salvo na configuração do seu projeto e se tornará o padrão. + +## Servidor de Desenvolvedor Externo + +Alguns frameworks vêm com seu próprio servidor ao vivo, no entanto, eles não serão capazes de tirar proveito das ligações Go/Wails. Neste cenário, é melhor executar um script de observador que reconstrui o projeto no diretório build, que Wails estará assistindo. Por exemplo, veja o modelo padrão do svelte que usa [rollup](https://rollupjs.org/guide/en/). + +### Criar Aplicativo React + +O processo para um projeto Create-React-App é um pouco mais complicado. Para ajudar o frontend a recarregar ao vivo a seguinte configuração precisa ser adicionado ao seu `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +O comando `frontend: dev:watcher` iniciará o servidor de desenvolvimento Create-React-App (hospedado na porta `3000` normalmente). O comando `frontend: dev:serverUrl` e instrui a Wails a servir assets do servidor de desenvolvimento ao carregar o frontend em vez de a partir da pasta de compilação. Além do acima acima o `index.html` precisa ser atualizado com o seguinte: + +```html + + + + + +``` + +Isso é necessário pois o comando do observador que reconstrui o frontend impede que o Wails injete os scripts necessários. Isso contorna esse problema garantindo os scripts são sempre injetados. Com esta configuração, `ondas de desenvolvimento` pode ser executado, o que irá construir adequadamente o frontend e o backend com o carregamento de quente ativado. Além disso, ao acessar o aplicativo a partir de um navegador, as ferramentas de desenvolvedor do React agora podem ser usadas em uma versão não minificada do aplicativo para simplificar depuração. Finalmente, para compilações mais rápidas, `wail dev -s` pode ser executado para ignorar o edifício padrão do frontend pelas Wails, pois este é um passo desnecessário. + +## Go Module + +Os modelos padrão do Wails geram um arquivo `go.mod` que contém o nome do módulo "changeme". Você deve alterar isto para algo mais apropriado após a geração do projeto. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..627bbeb5e36 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# Recursos Dinâmicos + +Se você deseja carregar ou gerar assets para seu frontend de forma dinâmica, você pode conseguir isso usando a opção [AssetsHandler](../reference/options#assetshandler). O AssetsHandler é um `http.Handler` genérico que irá ser chamado para qualquer solicitação não GET no servidor de ativos e para solicitações GET que não podem ser atendidas pelo ativos agrupados porque o arquivo não foi encontrado. + +Ao instalar um AssetsHandler personalizado, você pode servir seus próprios assets usando um servidor de arquivos personalizado. + +## Exemplo + +Em nosso projeto de exemplo, vamos criar um gerenciador de arquivos simples que irá carregar arquivos fora do disco: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +Quando executarmos o aplicativo no modo de desenvolvimento usando `wail dev`, veremos a seguinte saída: + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +Como você pode ver, o manipulador de assets é chamado quando o servidor de assets padrão não consegue servir o arquivo `favicon.ico`. + +Se você clicar com o botão direito no aplicativo principal e selecionar "inspeção" para abrir as devtools, você pode testar esta função digitando o seguinte no console: + +``` +let response = await fetch('does-not-exist.txt'); +``` + +Isto irá gerar um erro no devtools. Podemos ver que o erro é o que esperamos, retornado por nosso manipulador de ativos personalizados: + +```mdx-code-block +

+ +

+``` + +No entanto, se requisitarmos `go.mod`, veremos a seguinte saída: + +```mdx-code-block +

+ +

+``` + +Esta técnica pode ser usada para carregar imagens diretamente para a página. Se atualizarmos nosso modelo padrão do vanilla e substituirmos a imagem do logotipo: + +```html + +``` + +com: + +```html + +``` + +Veremos então o seguinte: + +```mdx-code-block +

+ +

+``` + +:::warning + +Expor seu sistema de arquivos desta forma é um risco à segurança. É recomendável que você gerencie corretamente o acesso ao seu sistema de arquivos. + +::: diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..167955b12d7 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,244 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx new file mode 100644 index 00000000000..430f53a6afc --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx @@ -0,0 +1,87 @@ +# Aplicações sem frames + +O Wails suporta aplicativos que não possuem frames. Isso pode ser conseguido usando o campo [sem frameless](../reference/options.mdx#frameless) no [Application Options](../reference/options.mdx#application-options). + +Wails oferece uma solução simples para arrastar a janela: qualquer elemento HTML que tenha o estilo CSS `--wails-draggable:drag` irá atuar como uma "alça de arrastar". Esta propriedade se aplica a todos os elementos filhos. Se você precisar indicar que um elemento aninhado não deve arrastar, então use o atributo '--wails-draggable:no-drag' nesse elemento. + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +Para alguns projetos, usar uma variável CSS pode não ser possível devido a um estilo dinâmico. Neste caso, você pode usar o aplicativo `CSSDragProperty` e `CSSDragValue` opções para definir uma propriedade e valor que serão usados para indicar regiões arrastáveis: + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info Tela Cheia + +Se você permitir que seu aplicativo vá para tela cheia, esta funcionalidade de arrastar será desativada. + +::: diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx new file mode 100644 index 00000000000..aade2b7f4cc --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx @@ -0,0 +1,72 @@ +# Frontend + +## Injeção de script + +Quando Wails serve a página `index.html`, por padrão injetará duas entradas de escript na ``tag para carregar `/wails/ipc.js` e `/wails/runtime.js`. Esses arquivos instalam o bindings e o tempo de execução, respectivamente. + +O código abaixo mostra onde estas são injetadas por padrão: + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### Sobrescrevendo a injeção de script padrão + +Para oferecer mais flexibilidade aos desenvolvedores, há uma meta tag que pode ser usada para personalizar este comportamento: + +```html + +``` + +As configurações são as seguintes: + +| Valor | Descrição | +| ------------------- | ---------------------------------------------- | +| noautoinjectruntime | Desativa a auto-injeção no `/wails/runtime.js` | +| noautoinjectipc | Desativa a autoinjeção no `/wails/ipc.js` | +| noautoinject | Desativa todas as autoinjeções de scripts | + +Várias opções podem ser usadas, desde que sejam separadas por vírgulas. + +Este código é perfeitamente válido e opera o mesmo que a versão da auto-injeção: + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx new file mode 100644 index 00000000000..fdb83a6adf3 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx @@ -0,0 +1,127 @@ +# IDEs + +O Wails pretende proporcionar uma grande experiência de desenvolvimento. Para atingir isso, agora suportamos a geração de configuração específica de IDE para fornecer uma configuração de projeto mais suave. + +Atualmente, nós suportamos o [Visual Studio Code](https://code.visualstudio.com/) mas buscamos oferecer suporte a outros IDEs, como o Goland. + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +Ao gerar um projeto usando a `-ide vscode` flag, arquivos IDE serão criados juntamente com os outros arquivos do projeto. Estes arquivos são colocados no diretório `.vscode` e fornecem a configuração correta para depurar seu aplicativo. + +Os 2 arquivos gerados são `tasks.json` e `launch.json`. Abaixo estão os arquivos gerados para o projeto vanilla padrão: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### Configurando os passos de instalação e compilação + +O arquivo `tasks.json` é simples para o projeto padrão, já que não há nenhuma etapa de instalação em que `npm` ou `npm run build` seja necessário. Para projetos que tenham uma etapa de build de front-end, como o modelo do Svelte, precisaríamos editar `tasks.json` e adicionar os passos de instalação e compilação: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +:::info Melhoria Futura + +No futuro, esperamos gerar um `tasks.json` que inclui os passos de instalação e compilação automaticamente. + +::: diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..0241bdb5a65 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Suporte a Distros Linux + +## Visão geral + +O Wails oferece suporte Linux, mas fornecer instruções de instalação para todas as distribuições disponíveis é uma tarefa impossível. Em vez disso, a Wails tenta determinar se os pacotes que você precisa para desenvolver aplicações estão disponíveis através do gerenciador de pacote do seu sistema. Atualmente, suportamos os seguintes gestores de pacotes: + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## Adicionando nomes de pacotes + +Pode haver circunstâncias em que sua distro use um dos gerentes de pacote suportados, mas o nome do pacote é diferente. Por exemplo, você pode usar um derivado de Ubuntu, mas o nome do pacote para gtk pode ser diferente. Wails tenta encontrar o pacote correto iterando por meio de uma lista de nomes de pacotes. A lista de pacotes é armazenada no arquivo específico do gerenciador de pacotes no diretório `v2/internal/system/packagemanager`. No nosso exemplo, isso seria `v2/internal/system/packagemanager/apt.go`. + +Neste arquivo, a lista de pacotes são definidos pelo método `Packages()`: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +Vamos supor que na nossa distração em linux, a `libgtk-3` é empacotada com o nome `lib-gtk3-dev`. Podemos adicionar suporte para isso adicionando a seguinte linha: + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## Adicionando novos gerenciadores de pacotes + +Para adicionar um novo gerenciador de pacotes, execute os seguintes passos: + +- Crie um novo arquivo em `v2/internal/system/packagemanager` chamado `.go`, onde `` é o nome do gerenciador de pacote. +- Defina uma struct que esteja em conformidade com a interface de gerenciador de pacotes definida em `pm.go`: + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()` deve retornar o nome do gerenciador de pacotes +- `Packages()` deve retornar um `packagemap`, que fornece nomes de arquivo de candidatos para dependências +- `PackageInstalled()` deve retornar `true` se o pacote for instalado +- `PackageAvailable()` deve retornar `true` se o pacote fornecido não estiver instalado, mas disponível para instalação +- `InstallCommand()` deve retornar o comando exato para instalar o nome do pacote dado + +Dê uma olhada no código de outros gerenciadores de pacotes para ter uma ideia de como isto funciona. + +:::info Lembre-se + +Se você adicionar suporte para um novo gerenciador de pacotes, não se esqueça de também atualizar esta página! + +::: diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx new file mode 100644 index 00000000000..a975968f316 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +Esta página possui vários guias relacionados ao desenvolvimento de aplicativos Wails para Linux. + +## A tag de vídeo não ativa o evento "ended" + +Ao usar uma tag de vídeo, o evento "ended" não é acionado quando o vídeo terminar de ser reproduzido. Este é um erro no WebkitGTK, no entanto, você pode usar a seguinte solução alternativa para corrigi-lo: + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +Fonte: [Lyimmi](https://github.com/Lyimmi) no [fórum de discussões](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx new file mode 100644 index 00000000000..4ba06231463 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx @@ -0,0 +1,55 @@ +# Desenvolvimento Local + +## Visão geral + +Os dispositivos estão em constante desenvolvimento e novos lançamentos são regularmente "marcados". Isso geralmente acontece quando todo o código no `master` foi testado e confirmado funcionando. Se você precisar de uma correção de bug ou recurso que ainda não foi lançado, é possível usar a versão mais recente "bleeding edge" seguindo as seguintes etapas: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +NOTA: O diretório para o qual você clonou o projeto será agora chamado de "clonedir". + +A CLI do Wails estará na versão mais recente. + +### Atualizando seu projeto + +Para atualizar projetos para usar a versão mais recente da biblioteca Wails, atualize o `do projeto. od` e certifique-se que a seguinte linha está no final do arquivo: + +`replace github.com/wailsapp/wails/v2 => ` + +Exemplo: + +Windows: `substitua github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +Em 'nix: `substitui github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +Para reverter para uma versão estável, execute: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## Testando uma Branch + +Se você deseja testar um branch, siga as instruções acima, mas certifique-se de alternar o branch que você deseja testar antes de instalar: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +Certifique-se de [atualizar seu projeto](#updating-your-project) conforme descrito acima. + +## Testando uma PR + +Se você deseja testar um branch, siga as instruções acima, mas certifique-se de alternar o branch que você deseja testar antes de instalar. Por favor, substitua `[IDofThePR]` pelo ID do PR mostrado no github.com: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +Certifique-se de [atualizar seu projeto](#updating-your-project) conforme descrito acima. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx new file mode 100644 index 00000000000..4b7b589c1a4 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Guia para Mac App Store + +Esta página dá uma breve visão geral de como enviar seu App Wails para a Mac App Store. + +## Pré-requisitos + +- Você precisará ter uma conta para o Apple Develop. Encontre mais informações no site [do Programa de Desenvolvedor Apple](https://developer.apple.com/support/compare-memberships/) +- Você precisará ter seus Certificados, Identificadores e App criados no portal de desenvolvimento. Mais sobre isto abaixo +- Ferramentas de linha de comando Xcode precisarão ser instaladas na sua máquina local + +#### Criar certificados e identificadores + +1. Vá para sua [Conta de desenvolvedor Apple](https://developer.apple.com/account/) +2. Sob `Certificados, Identificadores & Perfis`, clique em `Identificadores` e Registrar um Novo App ID. Use o formato (com.exemplo.app) +3. Sob a mesma página, clique `Certificados` e gere novos Certificados para a Distribuição da Loja de Aplicativos Mac. Baixe-os e importe os certificados para o Keychain em sua máquina local. + +#### Criar Envio de App + +1. Ir para [App Store Connect Site](https://appstoreconnect.apple.com/apps) +2. Registre um novo aplicativo e vincule o ID do pacote que você criou no passo anterior +3. Preencher seu aplicativo com as capturas de tela corretas, descrições, etc. conforme exigido pela Apple +4. Criar uma nova versão do seu aplicativo + +#### Criar perfil de provisionamento +1. Vá para a página [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) +2. Adicionar um novo perfil de provisionamento para Mac App Store de distribuição +3. Defina o tipo de perfil como Mac e selecione o ID do aplicativo criado acima +4. Selecione o certificado de Distribuição de Aplicativos Mac +5. Nomeie o Perfil de Provisão incorporado e baixe o perfil criado. + +## Guia para Mac App Store + +#### Ativar a App Sandbox da Apple + +Os aplicativos enviados para Mac App Store devem ser executados na [App Sandbox](https://developer.apple.com/app-sandboxing/) da Apple. Você deve criar um arquivo `entitlements.plist` para que isso funcione. A recomendação é criar este arquivo sob este caminho `{PROJECT_DIR}/build/darwin/entitlements.plist`. + +**Exemplo de arquivo de direitos** + +Este é um exemplo de titularidade de arquivo do aplicativo [RiftShare](https://github.com/achhabra2/riftshare). Para referência, por favor coloque os direitos que seu aplicativo exigir. Consulte [este site](https://developer.apple.com/documentation/bundleresources/entitlements) para obter mais informações. Você precisará substituir a ID da Equipe e o Nome da Aplicação pelos que você se registrou acima. + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +**Adicionar Perfil de Provisão Embutido** O Perfil de Provisionamento criado acima precisa ser adicionado à raiz da aplicação. Precisa ser nomeado como embedded.provisionprofile. + +#### Construa e assine o Pacote de Aplicativos + +O seguinte é um exemplo de script para construir e assinar seu aplicativo para o envio da Mac App Store. Ele presume que você está executando o script do diretório do seu projeto raiz. + +Observe que os certificados para a assinatura do aplicativo e a assinatura do instalador são diferentes. Certifique-se de que ambos são importados para o Keychain. Encontre as sequências de caracteres no Keychain e insira-as abaixo. Preencha os nomes do seu certificado e o nome do app abaixo. Executar o seguinte script irá gerar um arquivo `assinado app.pkg` no diretório raiz do seu aplicativo. + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### Enviar pacote de aplicativos + +Você precisará enviar o arquivo do pacote gerado e associá-lo à sua Aplicação antes de poder enviá-lo para revisão. + +1. Baixe o [aplicativo Transporter](https://apps.apple.com/us/app/transporter/id1450874784) na Mac App Store +2. Abra e inicie sessão com a sua Apple ID +3. Clique no sinal + e selecione o arquivo `APP_NAME.pkg` que você gerou na etapa anterior. Carregar isto +4. Volte para [Loja de Apps Conectar](https://appstoreconnect.apple.com/apps) e navegue de volta para a submissão de seu aplicativo. Selecione a versão que você está pronto para disponibilizar na App Store. Em `Build` selecione o pacote que você enviou via Transporter. + +É isso! Agora você pode usar o site para enviar seu Aplicativo para análise. Após alguns dias úteis, se tudo correr bem, você verá seu App ao vivo na Mac App Store. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx new file mode 100644 index 00000000000..f3142438d76 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# Compilações manuais + +O CLI Wails faz um trabalho pesado pelo projeto, mas às vezes é desejável construir o seu projeto manualmente. Este documento discutirá as diferentes operações da CLI e a forma como isso pode ser conseguido de diferentes maneiras. + +## Processo de compilação + +Quando `wails build` ou `wails dev` são usados, o Wails CLI executa um processo de compilação comum: + + - Instalar dependências frontend + - Construir projeto frontend + - Gerar mídias de construção + - Compilar aplicação + - [optional] Comprimir o aplicativo + +### Instalar dependências frontend + +#### Passos CLI + +- Se a flag `-s` for dada, esta etapa é ignorada +- Verifica `wails.json` para ver se há um comando de instalação na chave `frontend:install` +- Se não houver, ela pula esta etapa +- Se houver, ele verifica se `package.json` existe no diretório do frontend. Se não existir, pula esta etapa +- Uma soma MD5 é gerada a partir do conteúdo do arquivo `package.json` +- Verifica a existência de `package.json.md5` e, se existir, irá comparar o conteúdo dele (uma soma MD5) com o gerado para ver se o conteúdo mudou. Se eles forem iguais, este passo é ignorado +- Se `package.json.md5` não existir, será criado usando a soma MD5 gerada +- Se uma compilação é necessária agora, ou a `node_modules` não existe, ou a flag `-f` é dada, o comando de instalação é executado no diretório frontend + +#### Passos manuais + +Esta etapa pode ser feita na linha de comando ou um script com o `npm install`. + +### Construir projeto frontend + +#### Wails CLI + +- Se a flag `-s` for dada, esta etapa é ignorada +- Verifica `wails.json` para ver se há um comando de compilação na chave `frontend:build` +- Se não houver, ela pula esta etapa +- Se houver, ele é executado no diretório frontend + +#### Passos manuais + +Esta etapa poderia ser feita a partir da linha de comando ou um script com `npm run build` ou qualquer que seja o script de compilação do frontend. + +### Gerar mídias + +#### Wails CLI + +- Se a flag `-nopackage` estiver definida, este estágio será ignorado +- Se o arquivo `build/appicon.png` não existir, um arquivo padrão será criado +- Para Windows, consulte [Conjuntos para Windows](#windows) +- Se `build/windows/icon.ico` não existir, ele será criado a partir da imagem `build/appicon.png`. + +##### Windows + +- Se `build/windows/icon.ico` não existir, irá criá-lo a partir de `build/appicon. ng` usando tamanhos de ícones de 256, 128, 64, 48, 32 e 16. Isso é feito usando o [winicon](https://github.com/leaanthony/winicon). +- Se o arquivo `build/windows/.manifest` não existir, será criado a partir de uma versão padrão. +- Compila a aplicação como uma compilação de produção (acima) +- Usa [winres](https://github.com/tc-hib/winres) para empacotar o ícone e manifestar em um arquivo `.syso` pronto para vincular. + +#### Passos manuais + +- Crie `icon.ico` usando o [winicon](https://github.com/leaanthony/winicon) Ferramenta de CLI (ou qualquer outra ferramenta). +- Criar / Atualizar o arquivo de `.manifest` para sua aplicação +- Use o [winres CLI](https://github.com/tc-hib/go-winres) para gerar um arquivo `.syso`. + +### Compilar aplicação + +#### Wails CLI + +- Se o sinalizador `-clean` for fornecido, o diretório `build` é excluído e recriado +- Para `wails dev`, as seguintes bandeiras Go padrão são usadas: `-tags dev -gcflags "all=-N -l"` +- Para `wails build`, as seguintes flags padrão do Go são usadas: `-tags desktop,production -ldflags "-w -s"` + - No Windows, `-ldflags "-w -h -H windowsgui"` +- Tags adicionais passadas para o CLI usando `-tags` são adicionadas aos padrões +- Ldflags adicionais passadas para a CLI usando `-ldflags` são adicionados aos padrões +- A bandeira `-o` é passada +- O compilador Go especificado por `-compiler` será usado para compilação + +#### Passos manuais + +- Para compilação de desenvolvedores, o comando mínimo seria: `go build -tags dev -gcflags "all=-N -l"` +- Para a compilação de produção, o comando mínimo seria: `go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- Certifique-se de compilar no mesmo diretório que o arquivo `.syso` + +### Comprimir aplicativo + +#### Wails CLI + +- Se a flag `-upx` tiver sido dada, o programa `upx` será executado para comprimir o aplicativo com as configurações padrão +- Se `-upxflags` também for colado, essas flags são usadas em vez das padrão + +#### Passos manuais + +- Execute `upx [flags]` manualmente para comprimir o aplicativo. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx new file mode 100644 index 00000000000..a874b03f4f5 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx @@ -0,0 +1,191 @@ +# Migrando da v1 + +## Visão geral + +Wails v2 é uma mudança significativa em relação à v1. Este documento visa destacar as mudanças e as etapas na migração de um projeto existente. + +### Criando uma aplicação Cli + +Na v1, o aplicativo principal é criado usando `wails.CreateApp`, as ligações são adicionadas com `app.Bind` e, em seguida, o o aplicativo é executado usando `app.Run()`. + +Exemplo: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +Na v2, há apenas um único método, `wails.Run()`, que aceita as opções de aplicação [](../reference/options.mdx#application-options). + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### Mapeamento + +Na v1, foi possível vincular funções e estruturas arbitrárias. Em v2, foi simplificado apenas para estruturas vinculativas. As instâncias de construção que foram passadas anteriormente para o método `Bind()` na v1, estão agora especificados no campo `Vincular` do as opções de aplicação [](../reference/options.mdx#application-options): + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +Na v1, métodos vinculados estavam disponíveis para o frontend em `window.backend`. Isto mudou para `window.go` + +### Ciclo de vida da Aplicação + +Na v1, havia 2 métodos especiais em uma estrutura vinculada: `WailsInit()` e `WailsShutdown()`. Foi substituído por três ganchos de ciclo de vida como parte das [opções do aplicativo](../reference/options.mdx#application-options): + +- [OnStartup](../reference/options.mdx#onstartup) +- [OnShutdown](../reference/options.mdx#onshutdown) +- [OnDomReady](../reference/options.mdx#ondomready) + +Nota: [OnDomReady](../reference/options.mdx#ondomready) substitui o evento do sistema `wails:ready` na v1. + +Estes métodos podem ser funções padrão, mas uma prática comum é tê-los incluído numa estrutura: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### Tempo de execução + +O tempo de execução na v2 é muito mais rico que a v1 com suporte para menus, manipulação de janelas e melhores diálogos. A assinatura dos métodos mudou ligeiramente - consulte a [Referência de tempo de execução](../reference/runtime/intro.mdx). + +Na v1, o [runtime](../reference/runtime/intro.mdx) estava disponível através de uma struct passada para `WailsInit()`. Em v2, o tempo de execução foi movido para o seu próprio pacote. Cada método no tempo de execução leva o `context.Context` que é passado para o método [OnStartup](../reference/options.mdx#onstartup). + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startup is called at application startup +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} + +``` + +### Assets + +A _maior mudança_ na v2 é como os ativos são geridos. + +Na v1, os ativos foram passados via 2 opções de aplicativo: + +- `JS` - O JavaScript do aplicativo +- `CSS` - O CSS da aplicação + +Isso significava que a responsabilidade de gerar um único arquivo JS e CSS era do desenvolvedor. Isto exigia essencialmente a utilização de embalagens complicadas como o webpack. + +Na v2, Wails não faz suposições sobre seus ativos no frontend, como um servidor web. Todos os seus ativos de aplicação são passados para as opções de aplicação como um `embed.FS`. + +**Isso significa que não há necessidade de agrupar seus ativos, codificar imagens como Base64 ou experimente a arte obscura da configuração do bundler para usar fontes personalizadas**. + +Na inicialização, Wails verificará o `embed.FS` fornecido em busca de `index.html` e usará sua localização como caminho raiz para todos os outros ativos do aplicativo - assim como faria um servidor web. + +Exemplo: Uma aplicação tem o seguinte layout do projeto. Todos os arquivos finais são colocados no diretório `frontend/dist`: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +Esses ativos podem ser usados pelo aplicativo simplesmente criando um `embed.FS`: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +Claro, empacotadores podem ser usados se você quiser. O único requisito é passar o diretório final de ativos do aplicativo para Wails usando um `embed.FS` no `Assets` chave das [opções do aplicativo](../reference/options.mdx#application-options). + +### Configuração do Projeto + +Na v1, a configuração do projeto foi armazenada no arquivo `project.json` na raiz do projeto. Na v2, a configuração do projeto é armazenada no arquivo `wails.json` na raiz do projeto. + +O formato do arquivo é ligeiramente diferente. Aqui está uma comparação: + +

+ +| v1 | v2 | Notes | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | Removed | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | Changed | +| frontend / dir | | Removed | +| frontend / install | frontend:install | Changed | +| frontend / build | frontend:build | Changed | +| frontend / bridge | | Removed | +| frontend / serve | | Removed | +| tags | | Removed | +| | wailsjsdir | The directory to generate wailsjs modules | +| | assetdir | The directory of the compiled frontend assets for `dev` mode. Normalmente, isto é inferido e pode ser deixado vazio. | +| | reloaddirs | Lista separada por vírgulas de diretórios adicionais para observar alterações e acionar recarregamentos no modo `dev`. Isso só é necessário para algumas configurações de ativos mais avançadas. | + +

diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..054ac263bd1 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# Botões do Mouse + +As Wails runtime interceptam cliques do mouse para determinar se uma janela precisa ser redimensionada ou uma janela precisa ser movida. Foi perguntado como detectar quando um clique do mouse ocorreu, porque `window.onclick` não relata os botões do mouse corretamente. O código a seguir mostra como detectar cliques do mouse: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +Reference: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx new file mode 100644 index 00000000000..d921a6a5fc2 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# Builds ofuscadas + +Wails inclui suporte para ofuscar a sua aplicação usando [garble](https://github.com/burrowers/garble). + +Para produzir uma compilação ofuscada, você pode usar o sinalizador `-obfuscate` com o comando `wails build`: + +```bash +wails build -obfuscated +``` + +Para personalizar a configuração de ofuscação, pode-se utilizar a flag: `-garbleargs`: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +Essas configurações podem estar persistentes na configuração do seu [projeto](../reference/project-config). + +## Como funciona + +Em uma compilação padrão, todos os métodos vinculados estão disponíveis no frontend sob a variável `window.go`. Quando esses métodos são chamados, o método backend correspondente é chamado usando o nome da função totalmente qualificada. Quando usando uma compilação ofuscada, os métodos são vinculados usando um ID em vez de um nome. Os bindings gerados no diretório `wailsjs` usam esses IDs para chamar as funções de backend. + +:::note + +Para garantir que o seu aplicativo irá funcionar no modo ofuscado, você deve usar os bindings geradas sob o diretório `wailsjs` no seu aplicativo. + +::: + +## Exemplo + +Importing the "Greet" method from the bindings like this: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +irá garantir que o método funcionará corretamente no modo ofuscado, como os bindings serão regenerados com IDs e o mecanismo de chamada atualizado. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx new file mode 100644 index 00000000000..dcd13f147c3 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# Rolar demais + +[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) é o "efeito de salto" que você às vezes obtém quando você rola além dos limites de conteúdo de uma página. This is common in mobile apps. Isso pode ser desativado usando CSS: + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx new file mode 100644 index 00000000000..550e73a57f4 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx @@ -0,0 +1,47 @@ +# Roteamento + +O roteamento é uma maneira popular de alternar as telas de um aplicativo. Esta página oferece algumas orientações sobre como fazer isso. + +## Vue + +A abordagem recomendada para roteamento em Vue é o [Hash Mode](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode): + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +A abordagem recomendada para roteamento em Angular é [HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies#_hashlocationstrategy): + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +A abordagem recomendada para roteamento em React é [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx new file mode 100644 index 00000000000..c3e0c269658 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx @@ -0,0 +1,387 @@ +# Assinatura de Código + +Este é um guia sobre como você pode assinar seus binários gerados com Wails no MacOS e Windows. O guia visará ambientes de CI, mais especificamente as GitHub Actions. + +## Windows + +Primeiro você precisa de um certificado de assinatura de código. Se você ainda não tiver uma, a página de informações da Microsoft lista alguns provedores [aqui](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate). Observe que um certificado EV não é necessário a menos que você precise escrever um software no nível kernel como drivers de dispositivos. Para assinar o seu aplicativo Wails, um certificado de assinatura de código padrão vai dar certo. + +Pode ser uma boa ideia verificar com seu provedor de certificados como assinar seus binários na sua máquina local antes de direcionar sistemas de compilação automatizada. só para que você saiba se há exigências especiais. Por exemplo, [aqui](https://www.ssl.com/how-to/using-your-code-signing-certificate/) é o guia de assinatura de código de SSL.com para Windows. Se você sabe como assinar localmente, será mais fácil solucionar quaisquer possíveis problemas em um ambiente de CI. Por exemplo, os certificados de assinatura de código SSL.com exigem a flag `/tr` para [SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool) enquanto outros provedores só precisam da flag `/t` para fornecer o servidor de marcação de tempo. GitHub Actions populares para assinar binários Windows como [este](https://github.com/Dana-Prajea/code-sign-action) não suporta a flag `/tr` no SignTool.exe. Portanto, este guia focará em assinar nosso aplicativo manualmente com comandos do PowerShell, mas você pode usar ações como a ação de [de co-sign-action](https://github.com/Dana-Prajea/code-sign-action) se preferir. + +Em primeiro lugar, vamos nos certificar de que somos capazes de construir nosso aplicativo Wails no nosso GitHub CI. Aqui está um pequeno modelo de fluxo de trabalho: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend manually here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +Em seguida, precisamos dar ao fluxo de trabalho do GitHub acesso ao nosso certificado de assinatura. Isso é feito codificando seu certificado .pfx ou .p12 em uma string base64. Para fazer isso em PowerShell, use o seguinte comando assumindo que seu certificado se chama 'my-cert.p12': + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +Você deve ter seu arquivo .txt com o certificado codificado em base64. Deve começar com _-----BEGIN CERTIFICATE-----_ e terminar com _-----END CERTIFICATE-----_. Agora você precisa criar duas ações secretas no GitHub. Navegue até _Configurações -> Segredos -> Ações_ e crie os dois seguintes segredos: + +- **WIN_SIGNING_CERT** com o conteúdo de seu texto codificado no certificado base64. +- **WIN_SIGNING_CERT_PASSWORD** com o conteúdo da sua senha do certificado. + +Agora estamos prontos para implementar a assinatura em nosso fluxo de trabalho usando um dos dois métodos: + +### Método 1: Assinar com comandos + +Esse método usa comandos PowerShell para assinar nosso aplicativo, e deixa o controle de todo o processo de assinatura. + +Após o passo `Build Wails app`, podemos adicionar o seguinte passo ao nosso fluxo de trabalho: + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +Este script cria um novo diretório para o seu arquivo de certificado, cria o arquivo de certificado do nosso segredo base64, converte-o em um arquivo .pfx, e finalmente assina o binário. As seguintes variáveis precisam ser substituídas na última linha: + +- **signing algorithm**: geralmente sha256. +- **timestamping server**: URL para o servidor de timestamping a ser usado com seu certificado. +- **path to binary**: caminho para o binário que você deseja assinar. + +Dado que nossa configuração Wails tem `outputfilename` definido para "app.exe" e que temos um certificado de SSL.com, este é o nosso fluxo de trabalho: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### Método 2: assinar automaticamente com Action + +É possível usar um código Windows assinando uma ação como [este](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) um, mas nota que requer um hash SHA1 para o certificado e um nome de certificado. Veja um exemplo de como configurá-lo na [loja da ações](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate). + +--- + +## MacOS + +Primeiro você precisa do seu certificado de assinatura de código da Apple. Se você não tem uma, uma pesquisa simples do Google irá ajudá-lo a adquirir uma. Depois de ter o certificado, você precisa exportá-lo e codificá-lo para base64. [Este tutorial](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions) te mostra como fazer isso de uma maneira fácil. Depois de ter exportado seu arquivo de certificado .p12, você pode codificá-lo para base64, como visto no tutorial com o seguinte comando: + +```bash +base64 Certificates.p12 | pbcopy +``` + +Agora você está pronto para criar alguns segredos de projeto do GitHub, assim como no Windows: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64** com o conteúdo de seu certificado base64 recém-copiado. +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD** com o conteúdo da senha do seu certificado. +- **APPLE_PASSWORD** com o conteúdo de uma senha específica para sua conta Apple-ID que pode gerar [aqui](https://appleid.apple.com/account/manage). + +Vamos nos certificar de que somos capazes de construir nosso aplicativo Wails em nosso fluxo de trabalho do GitHub. Aqui está um pequeno modelo: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +Para assinar o código no macOS, [gon](https://github.com/mitchellh/gon) é uma ferramenta muito útil para a assinatura do código e a comunicação com os servidores Apple, também escrito em Go, e será usado neste guia. + +Após o passo `Build Wails app`, adicione o seguinte ao fluxo de trabalho: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +Agora precisamos configurar alguns arquivos de configuração do gon no nosso diretório `build/darwin`: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +Onde o `source` é o binário do seu Wails, `bundle_id` é o seu ID do pacote, `apple_id` contém o seu nome de usuário do ID Apple e a senha do aplicativo que você criou antes, e `sign.application_identity` é sua identidade que você pode encontrar executando o seguinte comando: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +Neste arquivo, você configura os direitos que você precisa para seu aplicativo, por exemplo, permissões de câmera se seu app usa a câmera. Leia mais sobre entitlements [aqui](https://developer.apple.com/documentation/bundleresources/entitlements). + +Verifique se você atualizou seu arquivo `Info.plist` com o mesmo pacote de ID que você digitou em `gon-sign.json`. Aqui está um arquivo `Info.plis` de exemplo: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +Agora estamos prontos para adicionar a etapa de assinatura em nosso fluxo de trabalho após a construção do aplicativo Wails: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +Por favor, note que a assinatura de binários com Apple pode levar a de minutos a horas. + +## Arquivo de fluxo combinado: + +Aqui está nosso arquivo de fluxo de trabalho do GitHub com Windows + macOS combinados: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# Notas finais + +Este guia inspirado no projeto RiftShare e seu fluxo de trabalho, o que é altamente recomendado para verificar [aqui](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml). diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..8281281f325 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +Este guia será para: + +1. Passos de instalação do Miminal - Os passos necessários para obter uma instalação mínima de Wails funcionando para o SvelteKit. +2. Script de Instalação - Bash script para cumprir os passos de Instalação Mínima com a marca opcional de Wails. +3. Notas importantes - Problemas que podem ser encontrados ao usar o SvelteKit + Wails e correções. + +## 1. Passos mínimos de instalação + +##### Instalar Wails para Svelte. + +- `wails init -n myapp -t svelte` + +##### Exclua o front-end do Svelte. + +- Navegue até a pasta myapp recém-criada. +- Excluir a pasta chamada "frontend" + +##### Enquanto estiver na raiz do projeto Wails. Use o seu gerenciador de pacotes favorito e instale o SvelteKit como o novo frontend. Siga as instruções. + +- `npm create svelte@latest frontend` + +##### Modificar o wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Observe que é aqui que suas funções Go e runtime aparecerão. +- Mude o frontend do seu gerenciador de pacotes se não estiver usando npm. + +##### Modificar o main.go. + +- O primeiro comentário `//go:embed all:frontend/dist` precisa ser alterado para `//go:embed all:frontend/build` + +##### Instalar/remover dependências usando seu gerenciador de pacote favorito. + +- Entre na sua pasta "frontend". +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Alterar adaptador em svelte.config.js + +- A primeira linha de arquivo altera `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Coloque o SvelteKit no modo SPA com pré-renderização. + +- Crie um arquivo sob myapp/frontend/src/routes/ chamado +layout.ts/+layout.js. +- Adicione duas linhas ao recém-criado arquivo `export const prerender = true` e `export const ssr = false` + +##### Testar instalação. + +- Navegue de volta à raiz do projeto Wails (um diretório para cima). +- execute `wails dev` +- Se a aplicação não executar, por favor verifique os passos anteriores. + +## 2. Script de Instalação + +##### Este Script do Bash faz as etapas listadas acima. Certifique-se de ler o script e de entender o que o script está fazendo no seu computador. + +- Criar um arquivo sveltekit-wails.sh +- Copie o código abaixo para o novo arquivo e o salve. +- Torná-lo executável com `chmod +x sveltekit-wails.sh` +- A marca é um parâmetro opcional abaixo que adiciona de volta na marca de fregueses. Deixe o terceiro parâmetro em branco para não inserir a marca das Wails. +- Exemplo de uso: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Notas importantes + +##### Os arquivos do servidor causarão falhas de construção. + +- \+layout.server.ts, +page.server.ts, +server.ts ou qualquer arquivo com "server" em nome falhará na construção enquanto todas as rotas forem pré-renderizadas. + +##### O tempo de execução do Wails descarrega com navegações de página inteira! + +- Tudo que causa navegações de página completa: `window.location.href = '//'` ou recarga do menu de contexto ao usar wails dev. Isso significa que você pode acabar perdendo a capacidade de chamar qualquer falha no aplicativo em tempo de execução. Há duas formas de trabalhar em torno desta questão. +- Use `import { goto } from '$app/navigation'` e então chame `goto('//')` em sua + page.svelte. Isso impedirá uma navegação de página completa. +- Se a navegação por página inteira não puder ser impedido que o tempo de execução do Wails seja adicionado a todas as páginas, adicionando o abaixo ao `` de myapp/frontend/src/app. Mt + +``` + +... + + + +... + +``` + +Veja https://wails.io/docs/guides/frontend para mais informações. + +##### Os dados de e-mail podem ser carregados e atualizados a partir de +page.ts/+page.js para +page.svelte. + +- \+page.ts/+page.js funciona bem com o load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() em +page.svelte irá chamar load() de +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Tratamento de erros + +- Erros esperados usando o erro Throw funcionam em +page.ts/+page.js com uma página +error.svelte. https://kit.svelte.dev/docs/errors#expected-errors +- Erros inesperados farão com que o aplicativo se torne inutilizável. Somente a opção de recuperação (conhecida até agora) de erros inesperados é recarregar o aplicativo. Para fazer isso, crie um arquivo myapp/frontend/src/hooks.client.ts e adicione o código abaixo ao arquivo. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Usando formas e funções de manipulação + +- A maneira mais simples é chamar uma função do formulário é o padrão, vincular:valor suas variáveis e evitar submissão `` +- A maneira mais avançada é use:enhance (aprimoramento progressivo) que permitirá acesso conveniente a formData, formElemento, emissor. A nota importante é sempre cancel() o formulário que impede o comportamento do lado do servidor. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Exemplo: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx new file mode 100644 index 00000000000..2ae61796b73 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx @@ -0,0 +1,97 @@ +# Templates + +Wails gera projetos a partir de modelos pré-criados. Na v1, este era um conjunto de projetos que estavam sujeitos a sair de moda. No v2, para capacitar a comunidade, alguns novos recursos foram adicionados para os templates: + +- Capacidade de gerar projetos a partir de [Modelos Remotos](../reference/cli.mdx#remote-templates) +- Ferramentas para ajudar a criar seus próprios modelos + +## Criando Templates + +Para criar um template, você pode usar o comando `wails generate template`. Para gerar um modelo padrão, execute: + +`wails generate template -name mytemplate` + +Isso cria o diretório "mytemplate" com os arquivos padrão: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### Visão Geral do Modelo + +O modelo padrão consiste nos seguintes arquivos e diretórios: + +| Nome do arquivo / diretório | Descrição | +| --------------------------- | -------------------------------------------- | +| NEXTSTEPS.md | Instruções sobre como completar o modelo | +| README.md | O README publicado com o modelo | +| app.tmpl.go | Arquivo de modelo `app.go` | +| frontend/ | O diretório que contém os assets do frontend | +| go.mod.tmpl | Arquivo de modelo `go.mod` | +| main.tmpl.go | Arquivo de modelo `main.go` | +| template.json | Os metadados do modelo | +| wails.tmpl.json | Arquivo de modelo `wails.json` | + +Neste ponto é aconselhável seguir os passos em `NEXTSTEPS.md`. + +## Criando um Template de um Projeto Existente + +É possível criar um modelo a partir de um projeto de frontend existente, passando o caminho para o projeto ao gerar o template. Vamos agora andar sobre como criar um modelo do Vue 3: + +- Instale o vue cli: `npm install -g @vue/cli` +- Crie o projeto padrão: `vue create vue3-base` + - Selecione `Padrão (Vue 3) ([Vue 3] babel, eslint)` +- Depois que o projeto for gerado, execute: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- O template agora pode ser personalizado conforme especificado no arquivo `NEXTSTEPS.md` +- Uma vez que os arquivos estão prontos, eles podem ser testados executando: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- Para testar o novo projeto, execute: `cd meu-vue3-projeto` e `wails constroem` +- Uma vez que o projeto tenha compilado, execute-o: `.\build\bin\my-vue3-project.exe` +- Você deve ter um aplicativo Vue3 que funcione plenamente: + +```mdx-code-block +
+ +
+``` + +## Publicando Templates + +A publicação de um template está simplesmente enviando os arquivos para o GitHub. São encorajadas as seguintes melhores práticas: + +- Remova todos os arquivos e diretórios indesejados (como `.git`) do seu diretório no frontend +- Certifique-se de que `template.json` esteja completo, especialmente `helpurl` +- Faça push dos arquivos para o GitHub +- Crie um PR na página [Templates de Comunidade](../community/templates.mdx) +- Anuncie o modelo no fórum de discussão [de Anúncio de Modelo](https://github.com/wailsapp/wails/discussions/825) diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..b4980716836 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# Resolução de Problemas + +Uma variedade de dicas de solução de problemas. + +## O comando `wail` parece estar faltando? + +Se o sistema está relatando que o comando `wails` está faltando, verifique se você seguiu o guia de instalação do Go corretamente. Normalmente, isso significa que o diretório `go/bin` no diretório inicial do seu usuário não está na variável `PATH` ambiente. Você normalmente também precisará fechar e reabrir qualquer prompt de comando aberto para que as alterações no ambiente feitas pelo instalador sejam refletidas no prompt de comando. + +## Meu aplicativo está exibindo uma tela branca/em branco + +Verifique se sua aplicação inclui os conteúdos do diretório correto. No seu arquivo `main.go`, você terá algo semelhante ao seguinte código: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +Verifique que `frontend/dist` contém os ativos da aplicação. + +### Mac + +Se isso acontecer no Mac, tente adicionar o seguinte ao seu `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Aplicativo Mac inválido + +Se a sua aplicação construída se parece com isso no buscador: + +```mdx-code-block +

+ +

+``` + +é provável que o `info.plist` do seu aplicativo seja inválido. Atualize o arquivo em `build/.app/Contents/info.plist` e verifique se os dados são válidos, verifique se o nome binário está correto. Para persistir nas alterações, copie o arquivo de volta para o diretório `build/darwin`. + +## Meu aplicativo não está exibindo o ícone correto no Windows Explorer + +Se seu aplicativo não estiver exibindo o ícone correto, tente excluir o arquivo `IconCache.db` oculto localizado na pasta Diretório `C:\Users\<seu nome de usuário>\AppData\Local`. Isto irá forçar o Windows a reconstruir o cache de ícones. + +Reference: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## Não é possível chamar o método backend no frontend com argumentos variados + +Se você tem um método de backend definido com parâmetros variadicos, por exemplo: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +chamar esse método a partir do frontend como isso irá falhar: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Gambiarra: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Credit: https://github.com/wailsapp/wails/issues/1186 + +## Estou recebendo erros de proxy ao tentar instalar o Wails + +Se você estiver recebendo erros como este: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +é provavelmente porque o Proxy oficial Go está sendo bloqueado (usuários na China relataram isto). A solução é configurar o proxy manualmente, por exemplo: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +Reference: https://github.com/wailsapp/wails/issues/1233 + +## O TypeScript gerado não tem os tipos corretos + +Às vezes, o TypeScript gerado não tem os tipos corretos. Para mitigar isso, é possível especificar quais tipos devem ser gerados usando a tag de struct `ts_type`. Para mais detalhes do, leia [isto](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). + +## Quando navego longe do `index.html`, não consigo chamar métodos no frontend + +Se você navegar do `index.html` para um novo arquivo html, o contexto será perdido. Isso pode ser corrigido adicionando as seguintes importações para a seção `` de qualquer nova página que você navegar: + +```html + + + + +``` + +Reference: https://github.com/wailsapp/wails/discussions/1512 + +## Eu recebo `muitos arquivos abertos` erros no meu Mac quando eu rodo `wails` + +Por padrão, o macOS só permitirá que você abra um máximo de 256 arquivos. Isso pode afetar o comando `wails dev`. Este limite pode ser aumentado em execução: `ulimit -n 1024` no terminal. + +FSNotify é [procurando mudar para os fsevents](https://github.com/fsnotify/fsnotify/issues/11) da Apple para Mac. Se isso não estiver concluído em breve, criaremos nossa própria implementação, monitorada [aqui](https://github.com/wailsapp/wails/issues/1733). + +## Meu aplicativo para Mac me dá erros estranhos de compilação + +Alguns usuários relataram ver erros de compilação como os seguintes: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +Isto é _normalmente_ devido a uma incompatibilidade com a versão do sistema operacional que você está executando e a versão das Ferramentas de Comando XCode instalada. Se você vir um erro como este, tente atualizar suas Ferramentas de Linha de Comando XCode para a versão mais recente. + +Se reinstalar as Ferramentas de Comando Xcode ainda falhar, você pode verificar o caminho onde o kit de ferramentas está usando: + +`xcode-select -p` + +Se `/Applications/Xcode.app/Contents/Developer` for exibido, rode `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Fontes: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +Se você estiver recebendo erros como este: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Não foi possível iniciar o serviço: A versão do host "x.x.x não coincide com a versão binária "x.x.x" + +É preferível adicionar `frontend/node_modules` e `frontend/package-lock.json` ao seu `.gitignore`. Caso contrário, ao abrir o repositório em outra máquina que pode ter diferentes versões do Node instaladas, talvez você não seja capaz de executar seu aplicativo. + +Se isso acontecer, simplesmente exclua `frontend/node_modules` e `frontend/pacote-lock. soa` e corra os seus wails `constroem` ou `wails dev` comando. + +## Processo de compilação travado em "Gerando vinculações" + +Processo de geração de Bindings executa sua aplicação em um modo especial. Se o aplicativo, intencionalmente ou não intencionalmente, contém um laço infinito (ou seja, não sair após `wails.Run()` terminado), isto pode levar a construção do processo travado na geração do palco de amarras. Por favor, certifique-se de que seu código sai corretamente. + +## Aplicação Mac pisca branco na inicialização + +Isto é devido ao plano de fundo padrão do webview ser branco. Se você quiser usar a cor de fundo da janela, você pode tornar o plano de fundo do webview transparente usando a seguinte configuração: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx new file mode 100644 index 00000000000..03962a2184c --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +Esta página é para dicas e truques diversos ao usar o Visual Studio Code com Wails. + +## Configuração do Vetur + +Muito obrigado [@Lyimmi](https://github.com/Lyimmi) por esta dica. Originalmente publicada [aqui](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349). + +Vetur é um popular plugin para o Visual Studio Code que fornece realce de sintaxe e conclusão de código para projetos Vue. Ao carregar um projeto Wails no VSCode, Vetur lançará um erro já que está esperando para encontrar o projeto frontend no diretório raiz. Para consertar isso, você pode fazer o seguinte: + +Crie um arquivo chamado `vetur.config.js` na raiz do projeto. + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +Agora, configure `frontend/tsconfig.json`: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +Isso deve permitir que você use Vetur como esperado. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx new file mode 100644 index 00000000000..40a4256d3f9 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# Instalador NSIS + +```mdx-code-block +

+ +
+

+``` + +O Wails suporta a geração de instaladores do Windows usando o [instalador do NSIS](https://nsis.sourceforge.io/). + +## Installing NSIS + +### Windows + +O instalador está disponível na página de [Download do NSIS](https://nsis.sourceforge.io/Download). + +Se você usar o gerenciador de pacotes de chocolatey, execute o seguinte script: + +``` +choco install nsis +``` + +Se você instalar o NSIS manualmente, você precisará adicionar a pasta _Bin_, que contém `makensis.exe`, e sua instalação NSIS ao seu PATH. [Aqui](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/) está um bom tutorial sobre como adicionar ao PATH no Windows. + +### Linux + +O pacote `nsis` deve estar disponível através do gerenciador de pacotes da sua distribuição. + +### MacOS + +O NSIS está disponível para instalação através de homebrew: `brew install nsis`. + +## Gerando o instalador + +Quando um novo projeto é criado, o Wails gera os arquivos de configuração do NSIS em `build/windows/installer`. Os dados de configuração são lidos do `installer/info.json` e estão configurados para usar a seção de informação do projeto `wails.json`: + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +Para gerar um instalador para o seu aplicativo, use a flag `-nsis` com o comando `wails build`: + +``` +wails build -nsis +``` + +O instalador agora estará disponível no diretório `build/bin`. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx new file mode 100644 index 00000000000..dd76db15adb --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +Esta página tem guias diversos relacionados ao desenvolvimento de aplicativos Wails para Windows. + +## Manipulando a dependência de runtime WebView2, + +Aplicativos de Wails feitos para Windows possuem uma exigência de tempo de execução na Microsoft [Runtime WebView2](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). Windows 11 terá isto instalado por padrão, mas algumas máquinas não. O Wails oferece uma abordagem fácil para lidar com esta dependência. + +Usando a flag `-webview2` ao realizar o build, você pode decidir o que sua aplicação fará quando não for detectado um tempo de execução adequado (incluindo se o tempo de execução instalado for muito antigo). As quatro opções são: + +1. Baixar +2. Incorporar +3. Navegador +4. Erro + +### Baixar + +Esta opção irá solicitar ao usuário que nenhum tempo de execução adequado foi encontrado e, em seguida, oferecer para baixar e executar o bootstrapper oficial no site WebView2 da Microsoft. Se o usuário continuar, o bootstrapper oficial será baixado e executado. + +### Incorporar + +Esta opção incorpora o bootstrapper oficial dentro do aplicativo. Se nenhum tempo de execução adequado for encontrado, o aplicativo oferecerá para executar o bootstrapper. Isto adiciona ~150k ao tamanho do binário. + +### Navegador + +Esta opção pedirá ao usuário que nenhum tempo de execução adequado foi encontrado e, em seguida, oferecerá para abrir um navegador na página oficial WebView2, onde o bootstrapper pode ser baixado e instalado. O aplicativo irá então sair, deixando a instalação para o usuário. + +### Erro + +Se não for encontrado um tempo de execução adequado, um erro é dado ao usuário e nenhuma ação foi feita. + +## Execução da versão fixa + +Outra forma de lidar com a dependência webview2 é enviando-a você mesmo. Você pode baixar [versão fixa em tempo de execução](https://developer.microsoft.com/microsoft-edge/webview2/#download-section) e empacotar ou baixá-lo com seu aplicativo. + +Além disso, você deve especificar o caminho para a versão fixa do tempo de execução webview2 nas `windows.Options` structure ao iniciar wails. + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +Observação: Quando `WebviewBrowserPath` é especificado a estratégia `error` seráforçada em caso de exigência de versão mínima ou erros na busca do runtime. + +## Gerando outros programas + +Ao gerar outros programas, tais como scripts, você verá a janela aparecer na tela. Para ocultar a janela, você pode usar o seguinte código: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +Solução fornecida pelo [sithembiso](https://github.com/sithembiso) no [quadro de discussões](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172). diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx new file mode 100644 index 00000000000..44fa130cc74 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# How does it work? + +A Wails application is a standard Go application, with a webkit frontend. The Go part of the application consists of the application code and a runtime library that provides a number of useful operations, like controlling the application window. The frontend is a webkit window that will display the frontend assets. Also available to the frontend is a JavaScript version of the runtime library. Finally, it is possible to bind Go methods to the frontend, and these will appear as JavaScript methods that can be called, just as if they were local JavaScript methods. + +```mdx-code-block +
+ +
+``` + +## The Main Application + +### Overview + +The main application consists of a single call to `wails.Run()`. It accepts the application configuration which describes the size of the application window, the window title, what assets to use, etc. A basic application might look like this: + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### Options rundown + +This example has the following options set: + +- `Title` - The text that should appear in the window's title bar +- `Width` & `Height` - The dimensions of the window +- `Assets` - The application's frontend assets +- `OnStartup` - A callback for when the window is created and is about to start loading the frontend assets +- `OnShutdown` - A callback for when the application is about to quit +- `Bind` - A slice of struct instances that we wish to expose to the frontend + +A full list of application options can be found in the [Options Reference](reference/options). + +#### Assets + +The `Assets` option is mandatory as you can't have a Wails application without frontend assets. Those assets can be any files you would expect to find in a web application - html, js, css, svg, png, etc. **There is no requirement to generate asset bundles** - plain files will do. When the application starts, it will attempt to load `index.html` from your assets and the frontend will essentially work as a browser from that point on. It is worth noting that there is no requirement on where in the `embed.FS` the files live. It is likely that the embed path uses a nested directory relative to your main application code, such as `frontend/dist`: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +At startup, Wails will iterate the embedded files looking for the directory containing `index.html`. All other assets will be loaded relative to this directory. + +As production binaries use the files contained in `embed.FS`, there are no external files required to be shipped with the application. + +When running in development mode using the `wails dev` command, the assets are loaded off disk, and any changes result in a "live reload". The location of the assets will be inferred from the `embed.FS`. + +More details can be found in the [Application Development Guide](guides/application-development.mdx). + +#### Application Lifecycle Callbacks + +Just before the frontend is about to load `index.html`, a callback is made to the function provided in [OnStartup](reference/options.mdx#onstartup). A standard Go context is passed to this method. This context is required when calling the runtime so a standard pattern is to save a reference to in this method. Just before the application shuts down, the [OnShutdown](reference/options.mdx#onshutdown) callback is called in the same way, again with the context. There is also an [OnDomReady](reference/options.mdx#ondomready) callback for when the frontend has completed loading all assets in `index.html` and is equivalent of the [`body onload`](https://www.w3schools.com/jsref/event_onload.asp) event in JavaScript. It is also possible to hook into the window close (or application quit) event by setting the option [OnBeforeClose](reference/options.mdx#onbeforeclose). + +#### Method Binding + +The `Bind` option is one of the most important options in a Wails application. It specifies which struct methods to expose to the frontend. Think of structs like "controllers" in a traditional web application. When the application starts, it examines the struct instances listed in the `Bind` field in the options, determines which methods are public (starts with an uppercase letter) and will generate JavaScript versions of those methods that can be called by the frontend code. + +:::info Note + +Wails requires that you pass in an _instance_ of the struct for it to bind it correctly + +::: + +In this example, we create a new `App` instance and then add this instance to the `Bind` option in `wails.Run`: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +You may bind as many structs as you like. Just make sure you create an instance of it and pass it in `Bind`: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +When you run `wails dev` (or `wails generate module`), a frontend module will be generated containing the following: + +- JavaScript bindings for all bound methods +- TypeScript declarations for all bound methods +- TypeScript definitions for all Go structs used as inputs or outputs by the bound methods + +This makes it incredibly simple to call Go code from the frontend, using the same strongly typed datastructures. + +## The Frontend + +### Overview + +The frontend is a collection of files rendered by webkit. It's like a browser and webserver in one. There is virtually[^1] no limit to which frameworks or libraries you can use. The main points of interaction between the frontend and your Go code are: + +- Calling bound Go methods +- Calling runtime methods + +### Calling bound Go methods + +When you run your application with `wails dev`, it will automatically generate JavaScript bindings for your structs in a directory called `wailsjs/go` (You can also do this by running `wails generate module`). The generated files mirror the package names in your application. In the example above, we bind `app`, which has one public method `Greet`. This will lead to the generation of the following files: + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +Here we can see that there is a `main` package that contains the JavaScript bindings for the bound `App` struct, as well as the TypeScript declaration file for those methods. To call `Greet` from our frontend, we simply import the method and call it like a regular JavaScript function: + +```javascript +// ... +import { Greet } from "../wailsjs/go/main/App"; + +function doGreeting(name) { + Greet(name).then((result) => { + // Do something with result + }); +} +``` + +The TypeScript declaration file gives you the correct types for the bound methods: + +```ts +export function Greet(arg1: string): Promise; +``` + +The generated methods return a Promise. A successful call will result in the first return value from the Go call to be passed to the `resolve` handler. An unsuccessful call is when a Go method that has an error type as it's second return value, passes an error instance back to the caller. This is passed back via the `reject` handler. In the example above, `Greet` only returns a `string` so the JavaScript call will never reject - unless invalid data is passed to it. + +All data types are correctly translated between Go and JavaScript. Even structs. If you return a struct from a Go call, it will be returned to your frontend as a JavaScript class. + +:::info Note + +Struct fields _must_ have a valid `json` tag to be included in the generated TypeScript. + +Anonymous nested structs are not supported at this time. + +::: + +It is possible to send structs back to Go. Any JavaScript map/class passed as an argument that is expecting a struct, will be converted to that struct type. To make this process a lot easier, in `dev` mode, a TypeScript module is generated, defining all the struct types used in bound methods. Using this module, it's possible to construct and send native JavaScript objects to the Go code. + +There is also support for Go methods that use structs in their signature. All Go structs specified by a bound method (either as parameters or return types) will have TypeScript versions auto generated as part of the Go code wrapper module. Using these, it's possible to share the same data model between Go and JavaScript. + +Example: We update our `Greet` method to accept a `Person` instead of a string: + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +The `wailsjs/go/main/App.js` file will still have the following code: + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +But the `wailsjs/go/main/App.d.ts` file will be updated with the following code: + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +As we can see, the "main" namespace is imported from a new "models.ts" file. This file contains all the struct definitions used by our bound methods. In this example, this is a `Person` struct. If we look at `models.ts`, we can see how the models are defined: + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +So long as you have TypeScript as part of your frontend build configuration, you can use these models in the following way: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +The combination of generated bindings and TypeScript models makes for a powerful development environment. + +More information on Binding can be found in the [Binding Methods](guides/application-development.mdx#binding-methods) section of the [Application Development Guide](guides/application-development.mdx). + +### Calling runtime methods + +The JavaScript runtime is located at `window.runtime` and contains many methods to do various tasks such as emit an event or perform logging operations: + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +More details about the JS runtime can be found in the [Runtime Reference](reference/runtime/intro). + +[^1]: There is a very small subset of libraries that use features unsupported in WebViews. There are often alternatives and workarounds for such cases. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx new file mode 100644 index 00000000000..f25eaba30c2 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# Introdução + +Wails é um projeto que permite a você escrever aplicativos de desktop usando Go e web. + +Considere uma alternativa leve e rápida do Electron para o Go. Você pode facilmente construir aplicativos com a flexibilidade e o poder do Go, combinados com um frontend rico e moderno. + +### Funcionalidades + +- Menus Nativos, Dialogs, Temas e Transparência +- Suporte a Windows, macOS e Linux +- Modelos embutidos para Svelte, React, Preact, Vue, Lit e Vanilla JS +- Easily call Go methods from JavaScript +- Automatic Go struct to TypeScript model generation +- Não são necessárias DLLs externas ou CGO no Windows +- Modo de desenvolvimento ao vivo usando o poder do [Vite](https://vitejs.dev/) +- Poderoso CLI para criar, Construir e Empacotar facilmente +- Uma rica [biblioteca de tempo de execução](/docs/reference/runtime/intro) +- Aplicativos construídos com Wails são compatíveis com Apple & Microsoft Store + +Este é [vago](https://varly.app) - um aplicativo para desktop para MacOS & Windows escrito usando Wails. Não só parece ótimo, como também usa menus nativos e translucidez - tudo o que você esperaria de um aplicativo nativo moderno. + +```mdx-code-block +

+ + + +

+``` + +### Criação Rápida de Template + +Há uma série de modelos pré-configurados que permitem que o aplicativo entre em funcionamento rapidamente. Há modelos para os seguintes frameworks: Svelte, React, Vue, Preact, Lit e Vanilla. There are both JavaScript and TypeScript versions for each template. + +### Elementos Nativos + +Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. + +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. No Windows, esta é a nova biblioteca da Microsoft Webview2, construída no Chromium. + +### Go & JavaScript Interoperability + +Wails automatically makes your Go methods available to JavaScript, so you can call them by name from your frontend! It even generates TypeScript models for the structs used by your Go methods, so you can pass the same data structures between Go and JavaScript. + +### Biblioteca de tempo de execução + +Wails provides a runtime library, for both Go and JavaScript, that handles a lot of the things modern applications need, like Eventing, Logging, Dialogs, etc. + +### Experiência de Desenvolvimento Ao Vivo + +#### Reconstruções automáticas + +Quando você executar seu aplicativo no modo "dev", o Wails construirá seu aplicativo como uma aplicação de desktop nativa, e lerá seus ativos a partir do disco. Ele detectará quaisquer alterações no seu Código Go e reconstruirá automaticamente e reiniciará seu aplicativo. + +#### Recarregamentos automáticos + +Quando alterações nos assets de seu aplicativo forem detectadas, seu aplicativo em execução irá "recarregar", reflectindo suas alterações quase imediatamente. + +#### Desenvolva sua aplicação com um navegador + +If you prefer to debug and develop in a browser then Wails has you covered. The running application also has a webserver that will run your application in any browser that connects to it. It will even refresh when your assets change on disk. + +### Production-ready Native Binaries + +When you're ready to do the final build of your application, the CLI will compile it down to a single executable, with all the assets bundled into it. On Windows and MacOS, it is possible to create a native package for distribution. The assets used in packaging (icon, info.plist, manifest file, etc) are part of your project and may be customised, giving you total control over how your applications are built. + +### Tooling + +The Wails CLI provides a hassle-free way to generate, build and bundle your applications. It will do the heavy lifting of creating icons, compiling your application with optimal settings and delivering a distributable, production ready binary. Choose from a number of starter templates to get up and running quickly! diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx new file mode 100644 index 00000000000..c2a5d706bac --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx @@ -0,0 +1,240 @@ +--- +sidebar_position: 2 +--- + +# Linha de comandos + +A CLI do Wails tem vários comandos que são usados para gerenciar seus projetos. Todos os comandos são executados da seguinte maneira: + +`wails ` + +## init + +`wails init` é usado para gerar projetos. + +| Flag | Descrição | Padrão | +|:------------------------- |:---------------------------------------------------------------------------------------------------------------------------------- |:---------------:| +| -n "nome do projeto" | Nome do projeto. **Obrigatório**. | | +| -d "diretório do projeto" | Diretório de projeto a criar | Nome do projeto | +| -g | Inicializar o repositório git | | +| -l | Listar modelos de projetos disponíveis | | +| -q | Suprimir saída no console | | +| -t "nome do template" | O modelo do projeto a ser usado. Este pode ser o nome de um template padrão ou um URL para um template remoto hospedado no github. | vanilla | +| -ide | Gerar arquivos de projeto IDE | | +| -f | Forçar compilação de aplicação | false | + +Exemplo: `wails init -n test -d mytestproject -g -ide vscode -q` + +Isso irá gerar um projeto chamado "test" no diretório "mytestproject", inicializar o git, gerar arquivos de projeto vscode e fazê-lo silenciosamente. + +Mais informações sobre isso podem ser encontradas [aqui](../guides/ides.mdx). + +### Modelos Remotos + +Modelos remotos (hospedados no GitHub) são suportados e podem ser instalados usando a URL do projeto do template. + +Exemplo: `wails init -n teste -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +Uma lista de modelos mantidos pela comunidade pode ser encontrada [aqui](../community/templates.mdx) + +:::warning Attention + +**The Wails project does not maintain, is not responsible nor liable for 3rd party templates!** + +If you are unsure about a template, inspect `package.json` and `wails.json` for what scripts are run and what packages are installed. + +::: + +## build + +`wails build` é usado para compilar seu projeto para um binário pronto para produção. + +| Flag | Descrição | Padrão | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:----------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Limpa o diretório `compilação/bin` | | +| -compiler "compiler" | Use um compilador de ida diferente para realizar build, por exemplo, go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Permite o uso das ferramentas devtools na janela do aplicativo | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Forçar compilação de aplicação | | +| -garbleargs | Argumentos para passar para o garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Ldflags adicionais para passar para o compilador | | +| -m | Skip mod tidy before compile | | +| -nopackage | Não empacotar aplicação | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o nome de arquivo | Nome do Arquivo de Saída | | +| -obfuscated | Ofuscar a aplicação usando [garble](https://github.com/burrowers/garble) | | +| -platform | Compila para as plataformas [(delimitadas por vírgula)](../reference/cli.mdx#platforms) por exemplo. `windows/arm64`. Note, se você não der arquitetura, `runtime.GOARCH` é usado. | platform = `variável de ambiente GOOS` se determinado `runtime.GOOS`.
arch = `GOARCH` variável de envrionment se for dado `runtime.GOARCH`. | +| -race | Realiza build com o Go race detector | | +| -s | Pular build do frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Compilar tags para passar para o compilador Go. Deve ser citado. Separados por espaço ou vírgula (mas não ambos) | | +| -trimpath | Remove todos os caminhos do sistema de arquivo do executável resultante. | | +| -u | Atualiza o `go.mod` do seu projeto para usar a mesma versão de Wails que o CLI | | +| -upx | Comprimir binário final usando "upx" | | +| -upxflags | Flags para passar para o upx | | +| -v int | Nível de verbosidade(0 - silencioso, 1 - padrão, 2 - verbose) | 1 | +| -webview2 | Estratégia de instalação WebView2: download,embed,browser,error | baixar | +| -windowsconsole | Manter a janela de console para builds do Windows | | + +Para uma descrição detalhada do sinalizador `webview2`, consulte o guia [Windows](../guides/windows.mdx). + +Se você preferir construir usando a ferramenta Go padrão, consulte o guia de [Builds Manuais](../guides/manual-builds.mdx). + +Exemplo: + +`wails build -clean -o myproject.exe` + +:::info + +On Mac, the application will be bundled with `Info.plist`, not `Info.dev.plist`. + +::: + +:::info UPX no Apple Silicon + +Há alguns [problemas](https://github.com/upx/upx/issues/446) com o uso de UPX com o Apple Silicon. + +::: + +:::info UPX no Windows + +Alguns fornecedores de antivírus marcam como falsos positivos os binários `upx` comprimidos como vírus, veja [problema](https://github.com/upx/upx/issues/437). + +::: + +### Plataformas + +Plataformas Suportadas: + +| Plataforma | Descrição | +|:---------------- |:--------------------------------------------- | +| darwin | MacOS + architecture of build machine | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | MacOS AMD64+ARM64 universal application | +| windows | Windows 10/11 + architecture of build machine | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + architecture of build machine | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## doctor + +`wails doctor` executará o diagnóstico para garantir que seu sistema esteja pronto para desenvolvimento. + +Exemplo: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## dev + +`wails dev` é usado para executar sua aplicação em um modo de "desenvolvimento ao vivo". Isso significa que: + +- O aplicativo `go.mod` será atualizado para usar a mesma versão do Wails como a CLI +- A aplicação é compilada e executada automaticamente +- Um observador é iniciado e acionará uma reconstrução do seu aplicativo de desenvolvimento se ele detectar alterações em seus arquivos go +- Um servidor web foi iniciado em `http://localhost:34115` que serve sua aplicação (não apenas frontend) sobre http. Isso permite que você use suas extensões de desenvolvimento de navegador favoritas +- Todos os conteúdos do aplicativo são carregados do disco. Se forem alterados, o aplicativo irá recarregar automaticamente (não reconstruir). Todos os navegadores conectados também recarregarão +- A JS module is generated that provides the following: +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- A second JS module is generated that provides a wrapper + TS declaration for the runtime +- On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. + +| Flag | Descrição | Padrão | +|:--------------------------------- |:---------------------------------------------------------------------------------------------------------------------------------------------------------------- |:--------------------- | +| -appargs "args" | Argumentos passados para o aplicativo no estilo shell | | +| -assetdir "./caminho/para/midias" | Serve os arquivos a partir do diretório fornecido em vez de usar os arquivos FS fornecidos | Valor em `wails.json` | +| -browser | Abre um navegador para `http://localhost:34115` na inicialização | | +| -compiler "compiler" | Use um compilador de ida diferente para realizar build, por exemplo, go1.15beta1 | go | +| -debounce | O tempo de esperar por recarregar depois que uma alteração de ativo for detectada | 100 (milliseconds) | +| -devserver "host:port" | O endereço para vincular o servidor de desenvolvimento de wails a | "localhost:34115" | +| -extensions | Extensões para acionar reconstruções (separadas por vírgula) | go | +| -forcebuild | Force build of application | | +| -frontenddevserverurl "url" | Usar URL do servidor de desenvolvimento de terceiros para servir midias, Vite EG | "" | +| -ldflags "flags" | Ldflags adicionais para passar para o compilador | | +| -loglevel "loglevel" | Nível de log a ser usado - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | Desativar saída da colorida da cli | false | +| -noreload | Desativar a recarga automática quando os arquivos forem alterados | | +| -nosyncgomod | Do not sync go.mod with the Wails version | false | +| -race | Realiza build com o Go race detector | false | +| -reloaddirs | Diretórios adicionais para acionar recarregamentos (separados por vírgula) | Valor em `wails.json` | +| -s | Pular build do frontend | false | +| -save | Salva o `assetdir`, `reloaddirs`,`wailsjsdir`,`debounce`,`devserver` e `frontenddevserverurl` passado, flag em `wails.json` para realizar chamadas subsequentes. | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Compilar tags para passar para o compilador (citado e espaço separado) | | +| -v | Nível de verbosidade(0 - silencioso, 1 - padrão, 2 - verbose) | 1 | +| -wailsjsdir | O diretório para gerar os módulos gerados do Wails JS | Valor em `wails.json` | + +Exemplo: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +Este comando fará o seguinte: + +- Compilação do aplicativo e execução (mais detalhes [aqui](../guides/manual-builds.mdx) +- Gerar os módulos JS de Wails em `./frontend/src` +- Assistir por atualizações em arquivos em `./frontend/dist` e recarregar em qualquer alteração +- Abrir um navegador e conectar-se ao aplicativo + +Há mais informações sobre como usar este recurso com scripts de framework existentes [aqui](../guides/application-development.mdx#live-reloading). + +## generate + +### template + +A Wails usa modelos para a geração de projeto. O comando `wails generate template` ajuda a fazer um template para que ele possa ser usado para gerar projetos. + +| Flag | Descrição | +|:---------------- |:----------------------------------------------------- | +| -name | O nome do template (Obrigatório) | +| -frontend "path" | Caminho para o projeto frontend para usar no template | + +Para mais detalhes sobre a criação de modelos, consulte o [guia de Modelos](../guides/templates.mdx). + +### module + +O comando `wails generate template` permite que você gere manualmente o diretório `wailsjs` para o seu aplicativo. + +## update + +`wails update` irá atualizar a versão da CLI do Wails. + +| Flag | Descrição | +|:------------------ |:------------------------------------- | +| -pre | Atualizar para a versão mais recente | +| -version "version" | Instalar uma versão específica do CLI | + +## version + +`wails version` simplesmente irá retornar a versão CLI atual. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx new file mode 100644 index 00000000000..a60919a04b3 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# Menus + +É possível adicionar um menu de aplicação aos projetos do Wails. Isso é conseguido definindo uma struct [Menu](#menu) e configurando-a no configuração do aplicativo [`Menu`](../reference/options.mdx#menu) ou chamando o método [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). + +Um exemplo de como criar um menu: + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +Também é possível atualizar dinamicamente o menu, atualizando o menu struct e chamando [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). + +O exemplo acima usa métodos de ajuda, no entanto, é possível construir as construções do menu manualmente. + +## Menu + +Um Menu é uma coleção de MenuItems: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +Para o menu de Aplicação, cada MenuItem representa um único menu como "Editar". + +Um método simples de ajuda é fornecido para menus de construção: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +Isto torna o layout do código mais parecido com o de um menu sem a necessidade de adicionar os itens de menu manualmente depois de criá-los. Como alternativa, você pode apenas criar os itens de menu e adicioná-los ao menu manualmente. + +## MenuItem + +Um MenuItem representa um item dentro de um Menu. + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| Campo | Tipo | Notas | +| ----------- | ------------------------------------ | -------------------------------------------------------------------------- | +| Label | string | O texto do menu | +| Accelerator | [\*keys.Accelerator](#accelerator) | Vinculação de teclas para este item de menu | +| Tipo | [Tipo](#type) | Tipo de MenuItem | +| Disabled | bool | Desativa o item de menu | +| Hidden | bool | Oculta este item de menu | +| Checked | bool | Adiciona uma seleção para o item ( & Tipos de Rádio) | +| SubMenu | [\*Menu](#menu) | Define o submenu | +| Click | [Callback](#callback) | Função Callback quando clicado no menu | +| Role | string | Define um papel [](#role) para este item de menu. Mac apenas por enquanto. | + +### Accelerator + +Os aceleradores (às vezes chamados atalhos de teclado) definem uma ligação entre um toque de tecla e um item de menu. Lamentos define um Acelerador como uma combinação ou tecla + [Modificador](#modifier). Eles estão disponíveis no pacote `"github.com/wailsapp/wails/v2/pkg/menu/keys"`. + +Exemplo: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +Teclas são qualquer caractere único em um teclado com exceção de `+`, que é definido como `plus`. Algumas chaves não podem ser representadas como caracteres, portanto há um conjunto de caracteres nomeados que podem ser usados: + +| | | | | +|:-----------:|:-----:|:-----:|:---------:| +| `backspace` | `f1` | `f16` | `f31` | +| `tab` | `f2` | `f17` | `f32` | +| `return` | `f3` | `f18` | `f33` | +| `enter` | `f4` | `f19` | `f34` | +| `escape` | `f5` | `f20` | `f35` | +| `left` | `f6` | `f21` | `numlock` | +| `right` | `f7` | `f22` | | +| `up` | `f8` | `f23` | | +| `down` | `f9` | `f24` | | +| `space` | `f10` | `f25` | | +| `delete` | `f11` | `f36` | | +| `home` | `f12` | `f37` | | +| `end` | `f13` | `f38` | | +| `page up` | `f14` | `f39` | | +| `page down` | `f15` | `f30` | | + +Wails também suportam a análise de aceleradores usando a mesma sintaxe que o Electron. Isso é útil para armazenar aceleradores em arquivos de configuração. + +Exemplo: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### Modificador + +Os seguintes modificadores são chaves que podem ser usadas em combinação com a tecla de aceleração: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +Vários métodos de ajuda estão disponíveis para criar aceleradores usando modificadores: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +Modificadores podem ser combinados usando `keys.Combo(string de chaves, modificador 1 modificador, modificador modificador, rest ...Modificador)`: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### Tipo + +Cada item de menu deve ter um tipo e existem 5 tipos disponíveis: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +Para conveniência, métodos auxiliares são fornecidos para criar rapidamente um item de menu: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +Você também pode criar itens de menu diretamente em um menu, usando os ajudantes "Adicionar": + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +Uma nota nos grupos de rádio: Um grupo de rádio é definido como um número de itens do menu de rádio que estão próximos um ao outro no menu. Isso significa que não é necessário agrupar os itens porque é automático. No entanto, isso também significa que você não pode ter 2 grupos lado a lado - deve haver um item que não seja de rádio entre eles. + +### Callback + +Cada item de menu pode ter um callback que é executado quando o item é clicado: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +A função recebe uma instrução `CallbackData` que indica qual item de menu acionou a callback. Isso é útil quando usar grupos de rádio que podem compartilhar um callback. + +### Role + +:::info Regras + +As regras que são atualmente suportados apenas no Mac. + +::: + +Um item de menu pode ter uma função, que é essencialmente um item de menu pré-definido. Atualmente, apoiamos as seguintes funções: + +| Role | Descrição | +| ------------ | --------------------------------------------------------------------------- | +| AppMenuRole | Menu padrão do aplicativo para Mac. Pode ser criado usando `menu.AppMenu()` | +| EditMenuRole | O menu de edição padrão para Mac. Pode ser criado usando `menu.EditMenu()` | diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx new file mode 100644 index 00000000000..46d1c071b60 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx @@ -0,0 +1,853 @@ +--- +sidebar_position: 3 +--- + +# Options + +## Application Options + +The `Options.App` struct contains the application configuration. It is passed to the `wails.Run()` method: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // User messages that can be customised + Messages *windows.Messages + // OnSuspend is called when Windows enters low power mode + OnSuspend func() + // OnResume is called when Windows resumes from low power mode + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### Title + +The text shown in the window's title bar. + +Name: Title
Type: `string` + +### Width + +The initial width of the window. + +Name: Width
Type: `int`
Default: 1024. + +### Height + +The initial height of the window. + +Name: Height
Type: `int`
Default: 768 + +### DisableResize + +By default, the main window is resizable. Setting this to `true` will keep it a fixed size. + +Name: DisableResize
Type: `bool` + +### Fullscreen + +Deprecated: Please use [WindowStartState](#windowstartstate). + +### WindowStartState + +Defines how the window should present itself at startup. + +| Value | Win | Mac | Lin | +| ---------- | --- | --- | --- | +| Fullscreen | ✅ | ✅ | ✅ | +| Maximised | ✅ | ✅ | ✅ | +| Minimised | ✅ | ❌ | ✅ | + +Name: WindowStartState
Type: `options.WindowStartState` + +### Frameless + +When set to `true`, the window will have no borders or title bar. Also see [Frameless Windows](../guides/frameless.mdx). + +Name: Frameless
Type: `bool` + +### MinWidth + +This sets the minimum width for the window. If the value given in `Width` is less than this value, the window will be set to `MinWidth` by default. + +Name: MinWidth
Type: `int` + +### MinHeight + +This sets the minimum height for the window. If the value given in `Height` is less than this value, the window will be set to `MinHeight` by default. + +Name: MinHeight
Type: `int` + +### MaxWidth + +This sets the maximum width for the window. If the value given in `Width` is more than this value, the window will be set to `MaxWidth` by default. + +Name: MaxWidth
Type: `int` + +### MaxHeight + +This sets the maximum height for the window. If the value given in `Height` is more than this value, the window will be set to `MaxHeight` by default. + +Name: MaxHeight
Type: `int` + +### StartHidden + +When set to `true`, the application will be hidden until [WindowShow](../reference/runtime/window.mdx#windowshow) is called. + +Name: StartHidden
Type: `bool` + +### HideWindowOnClose + +By default, closing the window will close the application. Setting this to `true` means closing the window will + +hide the window instead. + +Name: HideWindowOnClose
Type: `bool` + +### BackgroundColour + +This value is the default background colour of the window. Example: options.NewRGBA(255,0,0,128) - Red at 50% transparency + +Name: BackgroundColour
Type: `*options.RGBA`
Default: white + +### AlwaysOnTop + +Indicates that the window should stay above other windows when losing focus. + +Name: AlwaysOnTop
Type: `bool` + +### Assets + +Deprecated: Please use Assets on [AssetServer specific options](#assetserver). + +### AssetsHandler + +Deprecated: Please use AssetsHandler on [AssetServer specific options](#assetserver). + +### AssetServer + +This defines AssetServer specific options. It allows to customize the AssetServer with static assets, serving assets dynamically with an `http.Handler` or hook into the request chain with an `assetserver.Middleware`. + +Not all features of an `http.Request` are currently supported, please see the following feature matrix: + +| Feature | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +Name: AssetServer
Type: `*assetserver.Options` + +#### Assets + +The static frontend assets to be used by the application. + +A GET request is first tried to be served from this `fs.FS`. If the `fs.FS` returns `os.ErrNotExist` for that file, the request handling will fallback to the [Handler](#handler) and tries to serve the GET request from it. + +If set to nil, all GET requests will be forwarded to [Handler](#handler). + +Name: Assets
Type: `fs.FS` + +#### Handler + +The assets handler is a generic `http.Handler` for fallback handling of assets that can't be found. + +The handler will be called for every GET request that can't be served from [Assets](#assets), due to `os.ErrNotExist`. Furthermore all non GET requests will always be served from this Handler. If not defined, the result is the following in cases where the Handler would have been called: + +- GET request: `http.StatusNotFound` +- Other request: `http.StatusMethodNotAllowed` + +NOTE: When used in combination with a Frontend DevServer there might be limitations, eg. Vite serves the index.html on every path, that does not contain a file extension. + +Name: AssetsHandler
Type: `http.Handler` + +#### Middleware + +Middleware is a HTTP Middleware which allows to hook into the AssetServer request chain. It allows to skip the default request handler dynamically, e.g. implement specialized Routing etc. The Middleware is called to build a new `http.Handler` used by the AssetSever and it also receives the default handler used by the AssetServer as an argument. + +If not defined, the default AssetServer request chain is executed. + +Name: Middleware
Type: `assetserver.Middleware` + +### Menu + +The menu to be used by the application. More details about Menus in the [Menu Reference](../reference/runtime/menu.mdx). + +:::note + +On Mac, if no menu is specified, a default menu will be created. + +::: + +Name: Menu
Type: `*menu.Menu` + +### Logger + +The logger to be used by the application. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: Logger
Type: `logger.Logger`
Default: Logs to Stdout + +### LogLevel + +The default log level. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevel
Type: `logger.LogLevel`
Default: `Info` in dev mode, `Error` in production mode + +### LogLevelProduction + +The default log level for production builds. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevelProduction
Type: `logger.LogLevel`
Default: `Error` + +### OnStartup + +This callback is called after the frontend has been created, but before `index.html` has been loaded. It is given the application context. + +Name: OnStartup
Type: `func(ctx context.Context)` + +### OnDomReady + +This callback is called after the frontend has loaded `index.html` and its resources. It is given the application context. + +Name: OnDomReady
Type: `func(ctx context.Context)` + +### OnShutdown + +This callback is called after the frontend has been destroyed, just before the application terminates. It is given the application context. + +Name: OnShutdown
Type: `func(ctx context.Context)` + +### OnBeforeClose + +If this callback is set, it will be called when the application is about to quit, either by clicking the window close button or calling `runtime.Quit`. Returning true will cause the application to continue, false will continue shutdown as normal. This is good for confirming with the user that they wish to exit the program. + +Example: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +Name: OnBeforeClose
Type: `func(ctx context.Context) bool` + +### CSSDragProperty + +Indicates the CSS property to use to identify which elements can be used to drag the window. Default: `--wails-draggable`. + +Name: CSSDragProperty
Type: `string` + +### CSSDragValue + +Indicates what value the `CSSDragProperty` style should have to drag the window. Default: `drag`. + +Name: CSSDragValue
Type: `string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenu enables the browser's default context-menu in production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : + +| CSS Style | Behavior | +| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `--default-contextmenu: auto;` | (**default**) will show the default context menu only if :
contentEditable is true OR text has been selected OR element is input or textarea | +| `--default-contextmenu: show;` | will always show the default context menu | +| `--default-contextmenu: hide;` | will always hide the default context menu | + +This rule is inherited like any normal CSS rule, so nesting works as expected. + +:::note +This filtering functionality is only enabled in production, so in development and in debug build, the full context-menu is always available everywhere. +::: + +:::warning +This filtering functionality is NOT a security measure, the developer should expect that the full context-menu could be leaked anytime which could contain commands like (Download image, Reload, Save webpage), if this is a concern, the developer SHOULD NOT enable the default context-menu. +::: + + +Name: EnableDefaultContextMenu
Type: `bool` + +### EnableFraudulentWebsiteDetection + +EnableFraudulentWebsiteDetection enables scan services for fraudulent content, such as malware or phishing attempts. These services might send information from your app like URLs navigated to and possibly other content to cloud services of Apple and Microsoft. + +Name: EnableFraudulentWebsiteDetection
Type: `bool` + +### ZoomFactor + +Name: ZoomFactor
Type: `float64` + +This defines the zoom factor for the WebView2. This is the option matching the Edge user activated zoom in or out. + +### IsZoomControlEnabled + +Name: IsZoomControlEnabled
Type: `bool` + +This enables the zoom factor to be changed by the user. Please note that the zoom factor can be set in the options while disallowing the user to change it at runtime (f.e. for a kiosk application or similar). + +### Bind + +A slice of struct instances defining methods that need to be bound to the frontend. + +Name: Bind
Type: `[]interface{}` + +### ErrorFormatter + +A function that determines how errors are formatted when returned by a JS-to-Go method call. The returned value will be marshalled as JSON. + +Name: ErrorFormatter
Type: `func (error) any` + +### Windows + +This defines [Windows specific options](#windows). + +Name: Windows
Type: `*windows.Options` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent). + +For Windows 11 versions before build 22621, this will use the [BlurBehind](https://learn.microsoft.com/en-us/windows/win32/dwm/blur-ovw) method for translucency, which can be slow. For Windows 11 versions after build 22621, this will enable the newer translucency types that are much faster. By default, the type of translucency used will be determined by Windows. To configure this, use the [BackdropType](#BackdropType) option. + +Name: WindowIsTranslucent
Type: `bool` + +#### BackdropType + +:::note + +Requires Windows 11 build 22621 or later. + +::: + +Sets the translucency type of the window. This is only applicable if [WindowIsTranslucent](#WindowIsTranslucent) is set to `true`. + +Name: BackdropType
Type `windows.BackdropType` + +The value can be one of the following: + +| Value | Description | +| ------- | ----------------------------------------------------------------------------------------- | +| Auto | Let Windows decide which backdrop to use | +| None | Do not use translucency | +| Acrylic | Use [Acrylic](https://learn.microsoft.com/en-us/windows/apps/design/style/acrylic) effect | +| Mica | Use [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) effect | +| Tabbed | Use Tabbed. This is a backdrop that is similar to Mica. | + +#### DisableWindowIcon + +Setting this to `true` will remove the icon in the top left corner of the title bar. + +Name: DisableWindowIcon
Type: `bool` + +#### DisableFramelessWindowDecorations + +Setting this to `true` will remove the window decorations in [Frameless](#Frameless) mode. This means there will be no 'Aero Shadow' and no 'Rounded Corners' shown for the window. Please note that 'Rounded Corners' are only supported on Windows 11. + +Name: DisableFramelessWindowDecorations
Type: `bool` + +#### WebviewUserDataPath + +This defines the path where the WebView2 stores the user data. If empty `%APPDATA%\[BinaryName.exe]` will be used. + +Name: WebviewUserDataPath
Type: `string` + +#### WebviewBrowserPath + +This defines the path to a directory with WebView2 executable files and libraries. If empty, webview2 installed in the system will be used. + +Important information about distribution of fixed version runtime: + +- [How to get and extract runtime](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [Known issues for fixed version](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [The path of fixed version of the WebView2 Runtime should not contain \Edge\Application\.](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +Name: WebviewBrowserPath
Type: `string` + +#### Theme + +Minimum Windows Version: Windows 10 2004/20H1 + +This defines the theme that the application should use: + +| Value | Description | +| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------- | +| SystemDefault | _Default_. The theme will be based on the system default. If the user changes their theme, the application will update to use the new setting | +| Dark | The application will use a dark theme exclusively | +| Light | The application will use a light theme exclusively | + +Name: Theme
Type: `windows.Theme` + +#### CustomTheme + +:::note + +Minimum Windows Version: Windows 10/11 2009/21H2 Build 22000 + +::: + +Allows you to specify custom colours for TitleBar, TitleText and Border for both light and dark mode, as well as when the window is active or inactive. + +Name: CustomTheme
Type: `windows.CustomTheme` + +##### CustomTheme type + +The CustomTheme struct uses `int32` to specify the colour values. These are in the standard(!) Windows format of: `0x00BBGGAA`. A helper function is provided to do RGB conversions into this format: `windows.RGB(r,g,b uint8)`. + +NOTE: Any value not provided will default to black. + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +Example: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### Messages + +A struct of strings used by the webview2 installer if a valid webview2 runtime is not found. + +Name: Messages
Type: `*windows.Messages` + +Customise this for any language you choose to support. + +#### ResizeDebounceMS + +ResizeDebounceMS is the amount of time to debounce redraws of webview2 when resizing the window. The default value (0) will perform redraws as fast as it can. + +Name: ResizeDebounceMS
Type: `uint16` + +#### OnSuspend + +If set, this function will be called when Windows initiates a switch to low power mode (suspend/hibernate) + +Name: OnSuspend
Type: `func()` + +#### OnResume + +If set, this function will be called when Windows resumes from low power mode (suspend/hibernate) + +Name: OnResume
Type: `func()` + +#### WebviewGpuIsDisabled + +Setting this to `true` will disable GPU hardware acceleration for the webview. + +Name: WebviewGpuIsDisabled
Type: `bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +This defines [Mac specific options](#mac). + +Name: Mac
Type: `*mac.Options` + +#### TitleBar + +The TitleBar struct provides the ability to configure the look and feel of the title bar. + +Name: TitleBar
Type: [`*mac.TitleBar`](#titlebar-struct) + +##### Titlebar struct + +The titlebar of the application can be customised by using the TitleBar options: + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| Name | Description | +| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| TitlebarAppearsTransparent | Makes the titlebar transparent. This has the effect of hiding the titlebar and the content fill the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | Hides the title of the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | Removes [NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/) from the style mask | +| FullSizeContent | Makes the webview fill the entire window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | Adds a default toolbar to the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | Removes the line beneath the toolbar. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +Preconfigured titlebar settings are available: + +| Setting | Example | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +Example: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +Click [here](https://github.com/lukakerr/NSWindowStyles) for some inspiration on customising the titlebar. + +#### Appearance + +Appearance is used to set the style of your app in accordance with Apple's [NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc) names. + +Name: Appearance
Type: [`mac.AppearanceType`](#appearance-type) + +##### Appearance type + +You can specify the application's [appearance](https://developer.apple.com/documentation/appkit/nsappearance?language=objc). + +| Value | Description | +| ----------------------------------------------------- | --------------------------------------------------------------- | +| DefaultAppearance | DefaultAppearance uses the default system value | +| NSAppearanceNameAqua | The standard light system appearance | +| NSAppearanceNameDarkAqua | The standard dark system appearance | +| NSAppearanceNameVibrantLight | The light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastAqua | A high-contrast version of the standard light system appearance | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | A high-contrast version of the standard dark system appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | A high-contrast version of the light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | A high-contrast version of the dark vibrant appearance | + +Example: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent) to make frosty-looking applications. + +Name: WindowIsTranslucent
Type: `bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### About + +This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. + +Name: About
Type: [`*mac.AboutInfo`](#about-struct) + +##### About struct + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +If these settings are provided, an "About" menu item will appear in the app menu (when using the `AppMenu` role). Given this configuration: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +The "About" menu item will appear in the app menu: + +```mdx-code-block +
+ +
+
+``` + +When clicked, that will open an about message box: + +```mdx-code-block +
+ +
+
+``` + +### Linux + +This defines [Linux specific options](#linux). + +Name: Linux
Type: `*linux.Options` + +#### Icon + +Sets up the icon representing the window. This icon is used when the window is minimized (also known as iconified). + +Name: Icon
Type: `[]byte` + +Some window managers or desktop environments may also place it in the window frame, or display it in other contexts. On others, the icon is not used at all, so your mileage may vary. + +NOTE: Gnome on Wayland at least does not display this icon. To have a application icon there, a `.desktop` file has to be used. On KDE it should work. + +The icon should be provided in whatever size it was naturally drawn; that is, don’t scale the image before passing it. Scaling is postponed until the last minute, when the desired final size is known, to allow best quality. + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Some window managers may ignore it, or result in a black window. + +Name: WindowIsTranslucent
Type: `bool` + +#### WebviewGpuPolicy + +This option is used for determining the webview's hardware acceleration policy. + +Name: WebviewGpuPolicy
Type: [`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
Default: `WebviewGpuPolicyAlways` + +##### WebviewGpuPolicy type + +| Value | Description | +| ------------------------ | -------------------------------------------------------------------- | +| WebviewGpuPolicyAlways | Hardware acceleration is always enabled | +| WebviewGpuPolicyOnDemand | Hardware acceleration is enabled/disabled as request by web contents | +| WebviewGpuPolicyNever | Hardware acceleration is always disabled | + +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` property differs form the executable's filename. + +Name: ProgramName
Type: string
+ +### Debug + +This defines [Debug specific options](#Debug) that apply to debug builds. + +Name: Debug
Type: `options.Debug` + +#### OpenInspectorOnStartup + +Setting this to `true` will open the WebInspector on startup of the application. + +Name: OpenInspectorOnStartup
Type: `bool` + +[^1]: This requires WebKit2GTK 2.36+ support and your app needs to be build with the build tag `webkit2_36` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.36 for your app. +[^2]: This requires WebKit2GTK 2.40+ support and your app needs to be build with the build tag `webkit2_40` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.40 for your app. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx new file mode 100644 index 00000000000..8e763502bcc --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx @@ -0,0 +1,92 @@ +--- +sidebar_position: 5 +--- + +# Project Config + +The project config resides in the `wails.json` file in the project directory. The structure of the config is: + +```json5 +{ + // Project config version + "version": "", + // The project name + "name": "", + // Relative path to the directory containing the compiled assets, this is normally inferred and could be left empty + "assetdir": "", + // Additional directories to trigger reloads (comma separated), this is only used for some advanced asset configurations + "reloaddirs": "", + // The directory where the build files reside. Defaults to 'build' + "build:dir": "", + // Relative path to the frontend directory. Defaults to 'frontend' + "frontend:dir": "", + // The command to install node dependencies, run in the frontend directory - often `npm install` + "frontend:install": "", + // The command to build the assets, run in the frontend directory - often `npm run build` + "frontend:build": "", + // This command has been replaced by frontend:dev:build. If frontend:dev:build is not specified will falls back to this command. \nIf this command is also not specified will falls back to frontend:build + "frontend:dev": "", + // This command is the dev equivalent of frontend:build. If not specified falls back to frontend:dev + "frontend:dev:build": "", + // This command is the dev equivalent of frontend:install. If not specified falls back to frontend:install + "frontend:dev:install": "", + // This command is run in a separate process on `wails dev`. Useful for 3rd party watchers or starting 3d party dev servers + "frontend:dev:watcher": "", + // URL to a 3rd party dev server to be used to serve assets, EG Vite. \nIf this is set to 'auto' then the devServerUrl will be inferred from the Vite output + "frontend:dev:serverUrl": "", + // Relative path to the directory that the auto-generated JS modules will be created + "wailsjsdir": "", + // The name of the binary + "outputfilename": "", + // The default time the dev server waits to reload when it detects a change in assets + "debounceMS": 100, + // Address to bind the wails dev sever to. Default: localhost:34115 + "devServer": "", + // Arguments passed to the application in shell style when in dev mode + "appargs": "", + // Defines if build hooks should be run though they are defined for an OS other than the host OS. + "runNonNativeBuildHooks": false, + "preBuildHooks": { + // The command that will be executed before a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed before a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed before every build: ${platform} is replaced with the "GOOS/GOARCH". + "*/*": "" + }, + "postBuildHooks": { + // The command that will be executed after a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed after a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed after every build: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. + "*/*": "" + }, + // Data used to populate manifests and version info. + "info": { + // The company name. Default: [The project name] + "companyName": "", + // The product name. Default: [The project name] + "productName": "", + // The version of the product. Default: '1.0.0' + "productVersion": "", + // The copyright of the product. Default: 'Copyright.........' + "copyright": "", + // A short comment of the app. Default: 'Built using Wails (https://wails.app)' + "comments": "" + }, + // 'multiple': One installer per architecture. 'single': Single universal installer for all architectures being built. Default: 'multiple' + "nsisType": "", + // Whether the app should be obfuscated. Default: false + "obfuscated": "", + // The arguments to pass to the garble command when using the obfuscated flag + "garbleargs": "" +} +``` + +This file is read by the Wails CLI when running `wails build` or `wails dev`. + +The `assetdir`, `reloaddirs`, `wailsjsdir`, `debounceMS`, `devserver` and `frontenddevserverurl` flags in `wails build/dev` will update the project config +and thus become defaults for subsequent runs. + +The JSON Schema for this file is located [here](https://wails.io/schemas/config.v2.json). diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx new file mode 100644 index 00000000000..2765c844d53 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# Browser + +Esses métodos estão relacionados com o navegador do sistema. + +### BrowserOpenURL + +Abre a URL fornecida no navegador do sistema. + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..805f68ed917 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# Clipboard + +This part of the runtime provides access to the operating system's clipboard.
The current implementation only handles text. + +### ClipboardGetText + +This method reads the currently stored text from the clipboard. + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
Returns: a string (if the clipboard is empty an empty string will be returned) or an error. + +JS: `ClipboardGetText(): Promise`
Returns: a promise with a string result (if the clipboard is empty an empty string will be returned). + +### ClipboardSetText + +This method writes a text to the clipboard. + +Go: `ClipboardSetText(ctx context.Context, text string) error`
Returns: an error if there is any. + +JS: `ClipboardSetText(text: string): Promise`
Returns: a promise with true result if the text was successfully set on the clipboard, false otherwise. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..6a12ce1350a --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# Diálogo + +Esta parte do tempo de execução fornece acesso a dialogos nativos, como os Seletores de Arquivo e caixas de mensagem. + +:::info JavaScript + +O dialog não é suportado pelo runtime do JS. + +::: + +### OpenDirectoryDialog + +Abre um diálogo que solicita o usuário para selecionar um diretório. Pode ser personalizado usando o [OpenDialogOptions](#opendialogoptions). + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Retorna: Diretório selecionado (nulo se o usuário cancelar) ou um erro + +### OpenFileDialog + +Abre um diálogo que solicita o usuário para selecionar um arquivo. Pode ser personalizado usando o [OpenDialogOptions](#opendialogoptions). + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Retorna: Arquivo selecionado (nulo se o usuário cancelar) ou um erro + +### OpenMultipleFilesDialog + +Abre um diálogo que pede o usuário para selecionar vários arquivos. Pode ser personalizado usando o [OpenDialogOptions](#opendialogoptions). + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +Retorna: Arquivos selecionados (nulo se o usuário cancelar) ou um erro + +### SaveFileDialog + +Abre um diálogo que solicita o usuário selecionar um nome de arquivo para salvar. Pode ser personalizado usando [SaveDialogOptions](#savedialogoptions). + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +Retorna: O arquivo selecionado (nulo se o usuário cancelar) ou um erro + +### MessageDialog + +Exibe uma mensagem usando a caixa de diálogo da mensagem. Pode ser personalizado usando o [MessageDialogOptions](#messagedialogoptions). + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +Retorna: O texto do botão selecionado ou um erro + +## Opções + +### OpenDialogOptions + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| Atributo | Descrição | Win | Mac | Lin | +| -------------------------- | ------------------------------------------------------------- | --- | --- | --- | +| DefaultDirectory | O diretório que a caixa de diálogo será exibida quando aberta | ✅ | ✅ | ✅ | +| DefaultFilename | O nome do arquivo padrão | ✅ | ✅ | ✅ | +| Title | Título para a caixa de diálogo | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | Uma lista de filtros de arquivos | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Exibir arquivos ocultos pelo sistema | | ✅ | ✅ | +| CanCreateDirectories | Permitir que o usuário crie diretórios | | ✅ | | +| ResolvesAliases | Se verdadeiro, retorna o arquivo não o alias | | ✅ | | +| TreatPackagesAsDirectories | Permitir a navegação em pacotes | | ✅ | | + +### SaveDialogOptions + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| Atributo | Descrição | Win | Mac | Lin | +| -------------------------- | ------------------------------------------------------------- | --- | --- | --- | +| DefaultDirectory | O diretório que a caixa de diálogo será exibida quando aberta | ✅ | ✅ | ✅ | +| DefaultFilename | O nome do arquivo padrão | ✅ | ✅ | ✅ | +| Title | Título para a caixa de diálogo | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | Uma lista de filtros de arquivos | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Exibir arquivos ocultos pelo sistema | | ✅ | ✅ | +| CanCreateDirectories | Permitir que o usuário crie diretórios | | ✅ | | +| TreatPackagesAsDirectories | Permitir a navegação em pacotes | | ✅ | | + +### MessageDialogOptions + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| Atributo | Descrição | Win | Mac | Lin | +| ------------- | ------------------------------------------------------------------------ | -------------- | --- | --- | +| Tipo | O tipo de diálogo de mensagem, por exemplo, pergunta, informações... | ✅ | ✅ | ✅ | +| Title | Título para a caixa de diálogo | ✅ | ✅ | ✅ | +| Message | A mensagem para mostrar o usuário | ✅ | ✅ | ✅ | +| Buttons | Uma lista de títulos de botões | | ✅ | | +| DefaultButton | O botão com este texto deve ser tratado como padrão. Vincule a `return`. | ✅[*](#windows) | ✅ | | +| CancelButton | O botão com este texto deve ser tratado como padrão. Vincule a `return` | | ✅ | | + +#### Windows + +Windows tem tipos de diálogo padrão em que os botões não são personalizáveis. O valor retornado será um dos: "Ok", "Cancel", "Abort", "Retry", "Ignore", "Yes", "No", "Try Again" or "Continue". + +Para diálogos de Perguntas, o botão padrão é "Sim" e o botão cancelar é "Não". Isso pode ser alterado definindo o valor do `DefaultButton` para `"No"`. + +Exemplo: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linux tem tipos de diálogo padrão em que os botões não são personalizáveis. O valor retornado será um de: "Ok", "Cancel", "Yes", "No" + +#### Mac + +Uma caixa de diálogo de mensagem no Mac pode especificar até 4 botões. Se nenhum `DefaultButton` ou `CancelButton` for dado, o primeiro botão é considerado padrão e está ligado à chave `return`. + +Para o código a seguir: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, +}) +``` + +o primeiro botão é mostrado como padrão: + +```mdx-code-block +
+ +
+
+``` + +E se especificarmos o `DefaultButton` como "two": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +o segundo botão será mostrado como padrão. Quando `return` é pressionado, o valor "dois" é retornado. + +```mdx-code-block +
+ +
+
+``` + +Se especificarmos agora o `CancelButton` como "três: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +o botão com "três" é mostrado na parte inferior do diálogo. Quando `escape` é pressionado, o valor "três" é retornado: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### DialogType + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### FileFilter + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +O Windows permite que você use vários filtros de arquivos em caixas de diálogo. Cada FileFilter aparecerá como uma entrada separada na caixa de diálogo: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linux permite que você use vários filtros de arquivo em caixas de diálogo. Cada FileFilter aparecerá como uma entrada separada na caixa de diálogo: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +As caixas de diálogo Mac possuem apenas o conceito de um único conjunto de padrões para filtrar arquivos. Se vários Filtros forem fornecidos, o Wails usará todos os padrões definidos. + +Exemplo: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +Isso resultará na caixa de diálogo Abrir Arquivo usando `*.png,*.jpg,*.mov,*.mp4` como filtro. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx new file mode 100644 index 00000000000..ec0eb61e4e7 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# Events + +The Wails runtime provides a unified events system, where events can be emitted or received by either Go or JavaScript. Opcionalmente, dados podem ser passados com os eventos. Os ouvintes receberão os dados nos tipos de dados locais. + +### EventsOn + +Este método configura um ouvinte para o nome do evento. Quando um evento do tipo `eventName` é [emitted](#EventsEmit), a callback é acionada. Quaisquer dados adicionais enviados com o evento emitido serão passados para o retorno de chamada. Retorna uma função para cancelar o ouvinte. + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff + +Este método desregistra o listener para o nome do evento, opcionalmente várias listagens podem ser desregistradas via `adicionalEventNames`. + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce + +Esse método configura um listener para o nome do evento, mas só será acionado uma vez. Retorna uma função para cancelar o ouvinte. + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple + +Este método configura um listener para o nome do evento, mas acionará apenas um máximo de `counter` vezes. Retorna uma função para cancelar o ouvinte. + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit + +Este método emite o evento fornecido. Os dados opcionais podem ser passados com o evento. Isto irá acionar quaisquer ouvintes de eventos. + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx new file mode 100644 index 00000000000..b6f4e8a787c --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# Introduction + +O runtime é uma biblioteca que fornece métodos utilitários para sua aplicação. There is both a Go and JavaScript runtime and the aim is to try and keep them at parity where possible. + +Ele tem métodos de utilitário para: + +- [Janelas](window.mdx) +- [Menus](menu.mdx) +- [Caixa de diálogo](dialog.mdx) +- [Eventos](events.mdx) +- [Browser](browser.mdx) +- [Registro](log.mdx) +- [Clipboard](clipboard.mdx) + +O runtime do Go está disponível através da importação de `github.com/wailsapp/wails/v2/pkg/runtime`. Todos os métodos neste pacote assumem um contexto como o primeiro parâmetro. Este contexto deve ser obtido a partir dos ganchos [OnStartup](../options.mdx#onstartup) ou [OnDomReady](../options.mdx#ondomready). + +:::info Nota + +Embora o contexto seja fornecido para o método [Inicialização](../options.mdx#onstartup), não há garantia de que o runtime funcionará neste método, pois a janela está inicializando em um tópico diferente. Se você deseja chamar métodos de tempo de execução na inicialização, use [OnDomReady](../options.mdx#ondomready). + +::: + +The JavaScript library is available to the frontend via the `window.runtime` map. There is a runtime package generated when using `dev` mode that provides TypeScript declarations for the runtime. Isso deve ser localizado no diretório `wailsjs` em seu diretório frontend. + +### Hide + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +Oculta a aplicação. + +:::info Nota + +No Mac, isto irá ocultar o aplicativo da mesma forma que o `Ocultar` item de menu nos aplicativos padrão Mac. Isso é diferente de esconder a janela, mas a aplicação ainda está em primeiro plano. Para Windows e Linux, atualmente isso é o mesmo que `WindowHide`. + +::: + +### Show + +Mostra a aplicação. + +:::info Nota + +No Mac, a aplicação voltará a ser apresentada em primeiro plano. Para Windows e Linux, atualmente isso é o mesmo que `WindowShow`. + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### Quit + +Encerra a aplicação. + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### Environment + +Retorna detalhes do ambiente atual. + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### EnvironmentInfo + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx new file mode 100644 index 00000000000..0513b9431fd --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# Registro + +The Wails runtime provides a logging mechanism that may be called from Go or JavaScript. Como a maioria dos registros de, há um número de níveis de log: + +- Trace +- Debug +- Info +- Warning +- Error +- Fatal + +O logger irá gerar qualquer mensagem de log no nível atual, ou superior, de log. Exemplo: O `Debug` log level irá retornar todas as mensagens exceto `Trace` mensagens. + +### LogPrint + +Registra a mensagem dada como uma mensagem "bruta". + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf + +Registra a mensagem dada como uma mensagem "bruta". + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace + +Registra a mensagem dada no `nível de log`. + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef + +Registra a mensagem dada no `nível de log`. + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug + +Registra a mensagem dada no nível de log `Debug`. + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf + +Registra a mensagem dada no nível de log `Debug`. + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo + +Registra a mensagem dada no nível de log de `Info`. + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof + +Registra a mensagem dada no nível de log de `Info`. + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning + +Registra a mensagem dada no nível de log de `Warning`. + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf + +Registra a mensagem dada no nível de log de `Warning`. + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError + +Registra a mensagem dada no nível de log de `Error`. + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf + +Registra a mensagem dada no nível de log de `Error`. + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal + +Registra a mensagem dada no nível de log `Fatal`. + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf + +Registra a mensagem dada no nível de log `Fatal`. + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel + +Define o nível de log. In JavaScript, the number relates to the following log levels: + +| Valor | Nível de Log | +| ----- | ------------ | +| 1 | Trace | +| 2 | Debug | +| 3 | Info | +| 4 | Warning | +| 5 | Erro | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## Usando um Logger personalizado + +Um logger personalizado pode ser usado fornecendo usando a opção de aplicativo [Logger](../options.mdx#logger). O único requisito é que o logger implemente a interface de `logger.Logger` definida em `github.com/wailsapp/wails/v2/pkg/logger`: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx new file mode 100644 index 00000000000..ae07d327c93 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# Menus + +Estes métodos estão relacionados ao menu da aplicação. + +:::info JavaScript + +O menu não é suportado atualmente no runtime do JS. + +::: + +### MenuSetApplicationMenu + +Define o menu do aplicativo para o [menu](../menus.mdx) fornecido. + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu + +Atualiza o menu do aplicativo, pegando qualquer alteração no menu passado para `MenuSetApplicationMenu`. + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx new file mode 100644 index 00000000000..457c92ebff9 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# Screen + +These methods provide information about the currently connected screens. + +### ScreenGetAll + +Returns a list of currently connected screens. + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### Screen + +Go struct: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript interface: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx new file mode 100644 index 00000000000..04cc08ea679 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# Janela + +Esses métodos dão controle da janela do aplicativo. + +### WindowSetTitle + +Define o texto na barra de título da janela. + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen + +Faz a janela estar em tela cheia. + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen + +Restaura as dimensões e a posição anteriores da janela antes da tela cheia. + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen + +Retorna verdadeiro se a janela estiver em tela cheia. + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter + +Centraliza a janela no monitor que a janela está ativada. + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS + +Executa código JS arbitrário na janela. + +Este método executa o código no navegador de forma assíncrona e retorna imediatamente. Se o script causar quaisquer erros, eles só estarão disponíveis no console do navegador. + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload + +Executa um "recarregar" (Recarrega a página atual). + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp + +Recarrega o front-end do aplicativo. + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme + +Somente para Windows. + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +Define o tema da janela para padrão do sistema (escuro/claro). + +### WindowSetLightTheme + +Somente para Windows. + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +Define o tema da janela como claro. + +### WindowSetDarkTheme + +Somente para Windows. + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +Define o tema da janela como escuro. + +### WindowShow + +Mostra a janela, se ela estiver oculta no momento. + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowHide + +Oculta a janela, se ela estiver visível no momento. + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal + +Retorna verdadeiro se a janela não for minimizada, maximizada ou tela inteira. + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize + +Define a largura e a altura da janela. + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowGetSize + +Obtém a largura e a altura da janela. + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize + +Define o tamanho mínimo da janela. Será redimensionada a janela se a janela for atualmente menor do que as dimensões fornecidas. + +Definir um tamanho de `0,0` irá desativar esta restrição. + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize + +Define o tamanho máximo da janela. Será redimensionada a janela se a janela for atualmente maior do que as dimensões fornecidas. + +Definir um tamanho de `0,0` irá desativar esta restrição. + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop + +Define a janela sempreOnTop ou não no topo. + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition + +Define a posição da janela em relação ao monitor da janela ativada. + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition + +Obtém a posição da janela em relação ao monitor que a janela está ativada. + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximise + +Maximiza a janela para preencher a tela. + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise + +Restaurar a janela para as dimensões e a posição antes de maximizar. + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised + +Retorna verdadeiro se a janela for maximizada. + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise + +Alterna entre Maximizado e Não Maximizado. + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise + +Minimiza a janela. + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise + +Restaura a janela para as dimensões e a posição antes de minimizar. + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised + +Retorna verdadeiro se a janela estiver minimizada. + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour + +Define a cor de fundo da janela para a definição de cores RGBA fornecida. Esta cor será exibida para todos os pixels transparentes. + +Valores válidos para R, G, B e A são 0-255. + +:::info Windows + +No Windows, apenas valores alfa de 0 ou 255 são suportados. Qualquer valor que não for 0 será considerado 255. + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## TypeScript Object Definitions + +### Position + +```ts +interface Position { + x: number; + y: number; +} +``` + +### Size + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..1af16f7745a --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# Dogs API + +```mdx-code-block +
+ +
+
+``` + +:::note + +This tutorial has been kindly provided by [@tatadan](https://twitter.com/tatadan) and forms part of their [Wails Examples Repository](https://github.com/tataDan/wails-v2-examples). + +::: + +In this tutorial we are going to develop an application that retrieves photos of dogs from the web and then displays them. + +### Create the project + +Let's create the application. From a terminal enter: `wails init -n dogs-api -t svelte` + +Note: We could optionally add `-ide vscode` or `-ide goland` to the end of this command if you wanted to add IDE support. + +Now let's `cd dogs-api` and start editing the project files. + +### Remove unused code + +We will start by removing some elements that we know we will not use: + +- Open `app.go` and remove the following lines: + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- Open `frontend/src/App.svelte` and delete all lines. +- Delete the `frontend/src/assets/images/logo-universal.png` file + +### Creating our application + +Now let's add our new Go code. + +Add the following struct declarations to `app.go` before the function definitions: + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +Add the following functions to `app.go` (perhaps after the existing function definitions): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +Modify the `import` section of `app.go` to look like this: + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +Add the following lines to `frontend/src/App.svelte`: + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### Testing the application + +To generate the bindings and test the application, run `wails dev`. + +### Compiling the application + +To compile the application to a single, production grade binary, run `wails build`. diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx new file mode 100644 index 00000000000..ea350890049 --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# Hello World + +The aim of this tutorial is to get you up and running with the most basic application using Wails. You will be able to: + +- Create a new Wails application +- Build the application +- Run the application + +:::note + +This tutorial uses Windows as the target platform. Output will vary slightly depending on your operating system. + +::: + +## Create a new Wails application + +To create a new Wails application using the default vanilla JS template, you need to run the following command: + +```bash +wails init -n helloworld +``` + +You should see something similar to the following: + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +This will create a new directory called `helloworld` in the current directory. In this directory, you will find a number of files: + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## Build the application + +To build the application, change to the new `helloworld` project directory and run the following command: + +```bash +wails build +``` + +You should see something like the following: + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +This has compiled the application and saved it in the `build/bin` directory. + +## Run the application + +If we view the `build/bin` directory in Windows Explorer, we should see our project binary: + +```mdx-code-block +
+ +
+
+``` + +We can run it by simply double-clicking the `helloworld.exe` file. + +On Mac, Wails generates a `helloworld.app` file which can be run by double-clicking it. + +On Linux, you can run the application using `./helloworld` from the `build/bin` directory. + +You should see the application working as expected: + +```mdx-code-block +
+ +
+
+``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx new file mode 100644 index 00000000000..64c3f23b5ca --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# Ссылки + +Эта страница служит списком ссылок на сообщество. Пожалуйста, отправьте PR (нажмите `Редактировать эту страницу` внизу) для отправки ссылок. + +## Awesome Wails + +[Полный список](https://github.com/wailsapp/awesome-wails) ссылок, связанных с Wails. + +## Каналы поддержки + +- [Wails Discord Server](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [v2 Beta Discussion Board](https://github.com/wailsapp/wails/discussions/828) + +## Социальные сети + +- [Twitter](https://twitter.com/wailsapp) +- [Wails группа в QQ китайского сообщества](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - Номер группы: 1067173054 + +## Другие руководства и статьи + +- [Создание доски объявлений](https://blog.customct.com/building-bulletin-board) diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..37be75135ed --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +

+ +
+

+``` + +The [BulletinBoard](https://github.com/raguay/BulletinBoard) application is a versital message board for static messages or dialogs to get information from the user for a script. It has a TUI for creating new dialogs that can latter be used to get information from the user. It's design is to stay running on your system and show the information as needed and then hide away. I have a process for watching a file on my system and sending the contents to BulletinBoard when changed. It works great with my workflows. There is also an [Alfred workflow](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) for sending information to the program. The workflow is also for working with [EmailIt](https://github.com/raguay/EmailIt). diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx new file mode 100644 index 00000000000..c1817b70fff --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/) is a Wails 2 program that is a markdown based email sender only with nine notepads, scripts to manipulate the text, and templates. It also has a scripts terminal to run scripts in EmailIt on files in your system. The scripts and templates can be used from the commandline itself or with the Alfred, Keyboard Maestro, Dropzone, or PopClip extensions. It also supports scripts and themes downloaded form GitHub. Documentation is not complete, but the programs works. It’s built using Wails2 and Svelte, and the download is a universal macOS application. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..a03cc9bedc6 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app) - это простой и простой в использовании PGP инструмент шифрования, управления вашими ключами и ключами ваших контактов. Шифрование не должно быть сложным процессом для пользователя. Разработано на Wails.** + +Шифрование сообщений с помощью PGP является стандартом отрасли. У каждого есть приватный и публичный ключ. Ваш приватный ключ должен оставаться в секрете, для того, чтобы только вы могли читать сообщения. Ваш публичный ключ передается всем, кто хочет отправлять вам секретные и зашифрованные сообщения. Управление ключами, шифрованием сообщений и расшифровкой сообщений должно бесперебойно работать. EncryptEasy - это то, что позволяет сделать это с лёгкостью. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx new file mode 100644 index 00000000000..22b6d90816e --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# FileHound утилита для экспорта данных + +```mdx-code-block +

+ +
+

+``` + +[FileHound утилита для экспорта данных](https://www.filehound.co.uk/) FileHound это облачная платформа управления документами для безопасного сохранения файлов, автоматизации бизнес-процессов и SmartCapture возможностей. + +Утилита экспорта данных из FileHound позволяет администраторам запустить безопасное извлечение документов и данных для альтернативного варианта создания бэкапов и восстановления данных. Это приложение загруит все документы и/или метаданные, сохраненные в FileHound основываясь на выбранных фильтрах. Метадата будет извлечена в форматах XML и JSON. + +Бэкенд: Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +Frontend with: Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..87e5837d32f --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter) is a simple and efficient http API testing client tool. Based on Wails, Go and sveltejs. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..b79e123cab4 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Minecraft Updater + +```mdx-code-block +

+ +
+

+``` + +[Minecraft Updater](https://github.com/Gurkengewuerz/MinecraftModUpdater) — утилита для обновления и синхронизации Minecraft модов для вашей пользовательской базы. Она построена с помощью Wails2 и React с [antd](https://ant.design/) в качестве фронтенд фреймворка. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..ed470b8bd55 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Modal File Manager + +```mdx-code-block +

+ +
+

+``` + +[Modal File Manager](https://github.com/raguay/ModalFileManager) — это двухпанельный файловый менеджер с использованием веб-технологий. Мой оригинальный дизайн был основан на NW.js, и его можно найти [здесь](https://github.com/raguay/ModalFileManager-NWjs). В этой версии используется тот же код интерфейса Svelte (но с момента выхода из NW.), но бэкенд - это реализация [Wails 2](https://wails.io/). By using this implementation, I no longer use command line `rm`, `cp`, etc. commands, but a git install has to be on the system to download themes and extensions. Он полностью закодирован с помощью Go и работает гораздо быстрее, чем предыдущие версии. + +Этот файловый менеджер построен вокруг того же принципа, что и Vim: управление состоянием клавиатурных действий. Количество состояний не фиксировано, но очень программируется. Поэтому может быть создано и использовано бесконечное количество конфигураций клавиатуры. Это главное отличие от других файловых менеджеров. There are themes and extensions available to download from GitHub. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..2795fc6241c --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molly Wallet](https://github.com/grvlle/constellation_wallet/) официальный кошелек $DAG сети Constellation. Это позволит пользователям взаимодействовать с Hypergraph Network различными способами, не ограничиваясь $DAG транзакциями. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx new file mode 100644 index 00000000000..b9dbea1f888 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[October](https://october.utf9k.net) - это небольшая программа Wails, которая позволяет извлечь выделения из [Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader) и затем переслать их в [Readwise](https://readwise.io). + +Он обладает относительно небольшим охватом со всеми версиями платформы, весом менее 10 МБ, и это без включения [UPX сжатия](https://upx.github.io/)! + +В отличие от этого, предыдущие попытки автора с Electron быстро вылились в несколько сот мегабайт. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx new file mode 100644 index 00000000000..24a41cbd6cd --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# Optimus + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus) - приложение для оптимизации изображений. Он поддерживает конвертацию и сжатие изображений между форматами WebP, JPEG и PNG. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx new file mode 100644 index 00000000000..1a9fa1eb7d5 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# Portfall + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - Портпад k8s портал для перенаправления портов для легкого доступа ко всем интерфейсам кластера diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..5461ccf1af5 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - Простой кросс-платформенный интерфейс для [restic](https://github.com/restic/restic) для просмотра и восстановления оставшихся репозиториев. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..1b234072e13 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +Простой, безопасный и бесплатный обмен файлами для всех. Узнайте больше на [Riftshare.app](https://riftshare.app) + +## Возможности + +- Легкий обмен файлами между компьютерами как в локальной сети, так и через Интернет +- Поддерживает безопасную отправку файлов или директорий через [magic wormhole протокол](https://magic-wormhole.readthedocs.io/en/latest/) +- Совместимо со всеми приложениями, использующими magic wormhole (magic-wormhole или wormhole-william CLI, wormhole-gui и т. д.) +- Автоматическое сжатие нескольких выбранных файлов для одновременной отправки +- Полная анимация, индикатор прогресса и поддержка отмены для отправки и получения +- Нативное окно выбора файла +- Открытие файлов в один клик после получения +- Автоматическое обновление - не беспокойтесь о получении последней версии! diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..a512448fe40 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) is a program to show the output of scripts or [Node-Red](https://nodered.org) server. It runs scripts defined in EmailIt program and shows the output. Scripts from xBar or TextBar can be used, but currently on the TextBar scripts work well. Он также отображает вывод скриптов в вашей системе. ScriptBar не помещает их в меню, но все они в удобном окне для удобного просмотра. Вы можете открыть несколько вкладок, чтобы показать множество различных вещей. Вы также можете держать ссылки на наиболее посещаемые вами веб-сайты. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx new file mode 100644 index 00000000000..8f77bfb8a14 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/) это p2p приложение для обмена файлов, разработанное для использования блокчейн технологий для 100% анонимной передачи файлов. Surge зашифрован сквозным шифрованием, децентрализован и с открытым исходным кодом. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx new file mode 100644 index 00000000000..1d112481d06 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally) это официальное приложение для прошивки [Ergodox](https://ergodox-ez.com/) клавиатур. Это великолепный и фантастический пример того, чего вы можете достичь с помощью Wails: возможности сочетать мощь Go и богатые графические инструменты мира веб-разработки. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx new file mode 100644 index 00000000000..950dc3f3db1 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# Minecraft launcher for WarMine + +```mdx-code-block +

+ + +
+

+``` + +[Minecraft launcher for WarMine](https://warmine.ru/) is a Wails application, that allows you to easily join modded game servers and manage your game accounts. + +The Launcher downloads the game files, checks their integrity and launches the game with a wide range of customization options for the launch arguments from the backend. + +Frontend is written in Svelte, whole launcher fits in 9MB and supports Windows 7-11. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx new file mode 100644 index 00000000000..f526142ec28 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# Wombat + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat) - это кроссплатформенный клиент gRPC. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx new file mode 100644 index 00000000000..15be62e01e2 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails) это приложение для скачивания треков с youtube, создания оффлайн плейлистов и обмена ими с друзьями, ваши друзья смогут воспроизвести ваши плейлисты или загрузить их для оффлайн прослушивания, имеет встроенный плеер. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx new file mode 100644 index 00000000000..0b86bb8d9bd --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# Шаблоны + +Эта страница служит списком шаблонов, поддерживаемых сообществом. Пожалуйста, отправьте PR (нажмите `Изменить эту страницу` внизу) чтобы включить ваши шаблоны. Чтобы создать свой собственный шаблон, см. [руководство по шаблонам](../guides/templates.mdx). + +Чтобы использовать эти шаблоны, запустите `wails init -n "Название вашего проекта" -t [ссылка ниже[@version]]` + +Если нет суффикса версии, по умолчанию используется основной шаблон кода ветки. При наличии суффикса версии используется кодовый шаблон, соответствующий тегу этой версии. + +Пример: `wails init -n "Project Name" -t https://github.com/misitebao/wails-template-vue` + +::warning Внимание + +**Проект Wails не поддерживает и не несет ответственности за сторонние шаблоны!** + +Если вы не уверены в шаблоне, проверьте `package.json` и `wails.json` на какие скрипты запускаются и какие пакеты установлены. + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - шаблон, основанный на Vue ecology (ИнтегрированTypeScript, Темная тема, интернационализация, маршрутизация для одной страницы, TailwindCSS) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - Vue 3 TypeScript с Vite (и инструкции для добавления функций) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - Vue 3 TypeScript с Vite, Vuex, Vue Router, Sass, и ESLint + Prettier +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - A template using JavaScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier) +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - A template using TypeScript + Quasar V2 (Vue 3, Vite, Sass, Pinia, ESLint, Prettier, Composition API with <script setup>) +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - Wails template based on Naive UI (A Vue 3 Component Library) + +## Angular + +- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - Angular 15+ action packed & ready to roll to production. +- [wails-угловый-template](https://github.com/TAINCER/wails-angular-template) - Angular с TypeScript, Sass, Hot-Reload, Code-Splitting и i18n + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - Шаблон с использованием reactjs +- [wails-react-template](https://github.com/flin7/wails-react-template) - минимальный шаблон для React, который поддерживает живую разработку +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - шаблон с использованием Next.js и TypeScript +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - шаблон для React + TypeScript + Vite + TailwindCSS +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - шаблон с использованием Svelte +- [wails-vite-template](https://github.com/BillBuilt/wails-vite-svelte-template) - шаблон с использованием Svelte и Vite +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - шаблон с использованием Svelte и Vite с TailwindCSS v3 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - шаблон с использованием SvelteKit + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - A template using Solid + Ts + Vite +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - A template using Solid + Js + Vite + +## Elm + +- [wails-elm-шаблон](https://github.com/benjamin-thomas/wails-elm-template) - Разработайте ваше GUI приложение с функциональным программированием и **snappy** настройками горячей перезагрузки :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Объединение возможностей :muscle: Elm + Tailwind CSS + Wails! Поддерживается горячая перезагрузка. + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + +## Чистый JavaScript (Vanilla) + +- [wails-pure-js-шаблон](https://github.com/KiddoV/wails-pure-js-template) - шаблон, содержащий только базовый JavaScript, HTML и CSS diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx new file mode 100644 index 00000000000..11122314cdc --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# Компиляция проекта + +Из каталога проекта запустите `wails build`. Это позволит скомпилировать ваш проект и сохранить готовый к выпуску бинарный файл в директории `build/bin`. + +Если вы запустите исполняемый файл, вы увидите приложение по умолчанию: + +```mdx-code-block +
+ +
+
+``` + +Для получения более подробной информации о параметрах компиляции обратитесь к [CLI Reference](../reference/cli.mdx#build). diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx new file mode 100644 index 00000000000..e7c6351e782 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# Разработка вашего приложения + +Вы можете запустить ваше приложение в режиме разработки, выполнив `wails dev` в вашем каталоге проекта. Это сделает следующие вещи: + +- Соберет ваше приложение и запустит +- Bind your Go code to the frontend so it can be called from JavaScript +- С помощью [Vite](https://vitejs.dev/), будет наблюдать за изменениями в ваших Go файлах и пересобирать/перезапускать при изменении +- Устанавливает [веб-сервер](http://localhost:34115), который будет обслуживать ваше приложение через браузер. Это позволит вам использовать ваши любимые расширения браузера. Вы даже можете вызвать ваш Go-код из консоли + +Для начала запустите `wails dev` в каталоге проекта. Более подробную информацию об этом можно найти [здесь](../reference/cli.mdx#dev). + +Скоро: Инструкция diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..52b58fa7757 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# Создание проекта + +## Генерация проекта + +Теперь, когда CLI установлен, вы можете создать новый проект, используя команду `wails init`. + +Выберите Ваш любимый фреймворк: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Generate a Svelte project using JavaScript with:

+ + wails init -n myproject -t svelte + +If you would rather use TypeScript:
+ + wails init -n myproject -t svelte-ts + +
+ + Generate a React project using JavaScript with:

+ + wails init -n myproject -t react + +If you would rather use TypeScript:
+ + wails init -n myproject -t react-ts + +
+ + Generate a Vue project using JavaScript with:

+ + wails init -n myproject -t vue + +If you would rather use TypeScript:
+ + wails init -n myproject -t vue-ts + +
+ + Generate a Preact project using JavaScript with:

+ + wails init -n myproject -t preact + +If you would rather use TypeScript:
+ + wails init -n myproject -t preact-ts + +
+ + Generate a Lit project using JavaScript with:

+ + wails init -n myproject -t lit + +If you would rather use TypeScript:
+ + wails init -n myproject -t lit-ts + +
+ + Generate a Vanilla project using JavaScript with:

+ + wails init -n myproject -t vanilla + +If you would rather use TypeScript:
+ + wails init -n myproject -t vanilla-ts + +
+
+``` + +
+ +Доступны также [шаблоны сообщества](../community/templates.mdx), которые предлагают различные возможности и фреймворки. + +Чтобы увидеть другие доступные опции, вы можете запустить `wails init -help`. Более подробную информацию можно найти в [CLI Reference](../reference/cli.mdx#init). + +## Структура проекта + +Проекты Wails имеют следующую структуру: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### Краткое описание структуры проекта + +- `/main.go` - основное приложение +- `/frontend/` - фронтенд файлы проекта +- `/build/` - директория сборки проекта +- `/build/appicon.png` - значок приложения +- `/build/darwin/` - файлы проекта для Mac +- `/build/windows/` - файлы проектов, специфичных для Windows +- `/wails.json` - Конфигурация проекта +- `/go.mod` - Go module файл +- `/go.sum` - Go module проверочная сумма + +Каталог `frontend` ничего не имеет специфического для Wails и может быть любым пользовательским проектом по вашему выбору. + +Каталог `build` используется в процессе сборки. Эти файлы можно обновить, чтобы настроить ваши сборки. Если удалить файлы из директории, они будут заново сгенерированы с версией по-умолчанию. + +Название модуля по умолчанию в `go.mod` - это "changeme". Вы должны изменить это на нечто более подходящее. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx new file mode 100644 index 00000000000..b69ab2d4c76 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx @@ -0,0 +1,90 @@ +--- +sidebar_position: 1 +--- + +# Инструкция по установке + +## Поддерживаемые платформы + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## Зависимости + +Wails имеет ряд общих зависимостей, которые необходимы перед установкой: + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +Скачайте Go с [Go Downloads Page](https://go.dev/dl/). + +Убедитесь, что вы следуете официальным [Инструкциям по установке](https://go.dev/doc/install). Вам также нужно убедиться, что ваша переменная окружения `PATH` также включает путь к вашему каталогу `~/go/bin`. Перезапустите терминал и выполните следующие шаги: + +- Проверьте то, что Go установлен правильно: `go version` +- Проверьте "~/go/bin" в переменной PATH: `echo $PATH | grep go/bin` + +### NPM + +Загрузите NPM отсюда: [Node Downloads Page](https://nodejs.org/en/download/). Лучше использовать последнюю версию, так как это то, что мы её обычно тестируем. + +Запустите `npm --version` для проверки. + +## Зависимости платформы + +Вам также нужно установить специфичные для платформы зависимости: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wails требует установки инструментов xcode command line tools. Их можно установить, + запустив xcode-select --install. + + + Wails требует установки WebView2. В некоторых установках Windows это уже установлено. Вы можете проверить используя команду wails doctor. + + + Linux requires the standard gcc build tools plus libgtk3 and libwebkit. Вместо того чтобы перечислять огромное количество команд для разных дистрибутивов, Wails может попробовать определить команды установки, специфичные для вашего дистрибутива. Запустите wails doctor после установки, чтобы узнать, как установить зависимости. Если ваш дистрибутив или менеджер пакетов не поддерживается, пожалуйста, обратитесь к руководству {" "} Добавление дистрибутива Linux. + + +``` + +## Необязательные зависимости + +- [UPX](https://upx.github.io/) для сжатия приложений. +- [NSIS](https://wails.io/docs/guides/windows-installer/) for generating Windows installers. + +## Установка Wails + +Выполните `go install github.com/wailsapp/wails/v2/cmd/wails@latest` для установки Wails CLI. + +Примечание: Если вы получите ошибку, похожую на эту: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +пожалуйста, убедитесь, что у вас установлен Go 1.18+: +```shell +go version +``` + +## Проверка системы + +Запуск `wails doctor` проверит установлены ли у вас правильные зависимости. Если нет, то он покажет что не хватает, и покажет как исправить какие-либо проблемы. + +## Отсутствует команда `wails`? + +Если ваша система пишет, что команда `wails` отсутствует, удостоверьтесь, что вы корректно следовали инструкции по установке Go. Обычно это значит, что папка `go/bin`, находящаяся в домашней папке пользователя не добавлена в переменную окружения `PATH`. Обычно после изменения переменных окружения нужно переоткрыть командную строку, чтобы изменения применились в ней. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx new file mode 100644 index 00000000000..4d56b65bba1 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +Хотя у Wails нет шаблона с Angular, тем не менее Angular использовать возможно. + +## Режим разработчика + +Чтобы получить режим dev работы с Angular, вам нужно добавить следующие значения к `wails.json`: + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx new file mode 100644 index 00000000000..f9e723fee41 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx @@ -0,0 +1,215 @@ +# Разработка приложения + +Нет жестких и быстрых правил для разработки приложений с помощью Wails, но есть некоторые основные принципы. + +## Application Setup + +The pattern used by the default templates are that `main.go` is used for configuring and running the application, whilst `app.go` is used for defining the application logic. + +The `app.go` file will define a struct that has 2 methods which act as hooks into the main application: + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- The startup method is called as soon as Wails allocates the resources it needs and is a good place for creating resources, setting up event listeners and anything else the application needs at startup. It is given a `context.Context` which is usually saved in a struct field. This context is needed for calling the [runtime](../reference/runtime/intro.mdx). If this method returns an error, the application will terminate. In dev mode, the error will be output to the console. + +- The shutdown method will be called by Wails right at the end of the shutdown process. This is a good place to deallocate memory and perform any shutdown tasks. + +The `main.go` file generally consists of a single call to `wails.Run()`, which accepts the application configuration. The pattern used by the templates is that before the call to `wails.Run()`, an instance of the struct we defined in `app.go` is created and saved in a variable called `app`. This configuration is where we add our callbacks: + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +More information on application lifecycle hooks can be found [here](../howdoesitwork.mdx#application-lifecycle-callbacks). + +## Binding Methods + +It is likely that you will want to call Go methods from the frontend. This is normally done by adding public methods to the already defined struct in `app.go`: + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +In the main application configuration, the `Bind` key is where we can tell Wails what we want to bind: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +This will bind all public methods in our `App` struct (it will never bind the startup and shutdown methods). + +### Dealing with context when binding multiple structs + +If you want to bind methods for multiple structs but want each struct to keep a reference to the context so that you can use the runtime functions, a good pattern is to pass the context from the `OnStartup` method to your struct instances : + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +More information on Binding can be found [here](../howdoesitwork.mdx#method-binding). + +## Application Menu + +Wails supports adding a menu to your application. This is done by passing a [Menu](../reference/menus.mdx#menu) struct to application config. It's common to use a method that returns a Menu, and even more common for that to be a method on the `App` struct used for the lifecycle hooks. + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## Assets + +The great thing about the way Wails v2 handles assets is that it doesn't! The only thing you need to give Wails is an `embed.FS`. How you get to that is entirely up to you. You can use vanilla html/css/js files like the vanilla template. You could have some complicated build system, it doesn't matter. + +When `wails build` is run, it will check the `wails.json` project file at the project root. There are 2 keys in the project file that are read: + +- "frontend:install" +- "frontend:build" + +The first, if given, will be executed in the `frontend` directory to install the node modules. The second, if given, will be executed in the `frontend` directory to build the frontend project. + +If these 2 keys aren't given, then Wails does absolutely nothing with the frontend. It is only expecting that `embed.FS`. + +### AssetsHandler + +A Wails v2 app can optionally define a `http.Handler` in the `options.App`, which allows hooking into the AssetServer to create files on the fly or process POST/PUT requests. GET requests are always first handled by the `assets` FS. If the FS doesn't find the requested file the request will be forwarded to the `http.Handler` for serving. Any requests other than GET will be directly processed by the `AssetsHandler` if specified. It's also possible to only use the `AssetsHandler` by specifiy `nil` as the `Assets` option. + +## Built in Dev Server + +Running `wails dev` will start the built in dev server which will start a file watcher in your project directory. By default, if any file changes, wails checks if it was an application file (default: `.go`, configurable with `-e` flag). If it was, then it will rebuild your application and relaunch it. If the changed file was in the assets, it will issue a reload after a short amount of time. + +The dev server uses a technique called "debouncing" which means it doesn't reload straight away, as there may be multiple files changed in a short amount of time. When a trigger occurs, it waits for a set amount of time before issuing a reload. If another trigger happens, it resets to the wait time again. By default this value is `100ms`. If this value doesn't work for your project, it can be configured using the `-debounce` flag. If used, this value will be saved to your project config and become the default. + +## External Dev Server + +Some frameworks come with their own live-reloading server, however they will not be able to take advantage of the Wails Go bindings. In this scenario, it is best to run a watcher script that rebuilds the project into the build directory, which Wails will be watching. For an example, see the default svelte template that uses [rollup](https://rollupjs.org/guide/en/). + +### Create React App + +The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: + +```html + + + + + +``` + +This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. + +## Go Module + +The default Wails templates generate a `go.mod` file that contains the module name "changeme". You should change this to something more appropriate after project generation. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..975916524b4 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# Динамические ресурсы + +Если вы хотите загрузить или генерировать ресурсы для вашего фронтенда динамически, вы можете это сделать, используя опцию [AssetsHandler](../reference/options#assetshandler). AssetsHandler это общий `http.Handler`, который будет вызываться для любого запроса, не являющегося GET, на сервере ресурсов и для запросов GET, которые не могут быть обработаны из бандла ресурсов, поскольку файл не найден. + +Установив пользовательские AssetsHandler, вы можете предоставлять свои собственные ресурсы с помощью пользовательского ресурсного сервера. + +## Пример + +В нашем примере мы создадим простой обработчик ресурсов, который загрузит файлы с диска: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +Когда мы запустим приложение в dev режиме, с помощью команды `wails dev`, мы увидим следующий вывод: + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +Как видите, обработчик ресурсов вызывается, когда стандартный сервер ресурсов не может предоставить файл `favicon.ico`. + +Если вы щелкните правой кнопкой мыши по окну приложения и выберите "Исследовать элемент" для вызова devtools, вы сможете протестировать эту функцию, введя в консоль следующую строку: + +``` +let response = await fetch('does-not-exist.txt'); +``` + +Это сгенерирует ошибку в devtools. Мы видим ошибку обработчика пользовательских ресурсов, которую мы ожидали увидеть: + +```mdx-code-block +

+ +

+``` + +Однако, если мы вызовем `go.mod`, мы увидим следующий вывод: + +```mdx-code-block +

+ +

+``` + +А так мы можем разместить изображение прямо на страницу. Если мы обновим шаблон по умолчанию и заменим изображение логотипа: + +```html + +``` + +на: + +```html + +``` + +Тогда мы увидим следующее: + +```mdx-code-block +

+ +

+``` + +:::warning + +Раскрытие структуры файловой системы таким образом сопряжено с рисками безопасности. Рекомендуется правильно управлять доступом к файловой системе. + +::: diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..167955b12d7 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,244 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx new file mode 100644 index 00000000000..1c7a661e794 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx @@ -0,0 +1,87 @@ +# Безрамочные приложения + +Wails поддерживает приложения, не имеющие рамок. Этого можно добиться, используя поле [frameless](../reference/options.mdx#frameless) в [опциях приложения](../reference/options.mdx#application-options). + +Wails предоставляет легкое решение для перемещения окна: любой HTML элемент имеющий стиль `--wails-draggable:drag` будет работать как "ручка для перетаскивания". Это свойство применяется для всех дочерних элементов. Если вам необходимо указать, что вложенный элемент не должен перестаскивать окно, то используйте атрибут '--wails-draggable:no-drag' на этом элементе. + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +Для некоторых проектов использование CSS переменной может оказаться невозможным из-за динамических стилей. В этом случае вы можете использовать параметры `CSSDragProperty` и `CSSDragValue` в опциях приожения для определения свойства и значения, которые будут использоваться для указания перетаскиваемых областей: + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // Создаем экземляр структуры приложения + app := NewApp() + + // Создаем приложение с опциями + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info Fullscreen + +Если вы разрешите приложению перейти в полноэкранный режим, функция перетаскивания будет отключена. + +::: diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx new file mode 100644 index 00000000000..ac087ee4514 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx @@ -0,0 +1,72 @@ +# Frontend + +## Script Injection + +When Wails serves your `index.html`, by default, it will inject 2 script entries into the `` tag to load `/wails/ipc.js` and `/wails/runtime.js`. These files install the bindings and runtime respectively. + +The code below shows where these are injected by default: + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### Overriding Default Script Injection + +To provide more flexibility to developers, there is a meta tag that may be used to customise this behaviour: + +```html + +``` + +The options are as follows: + +| Value | Description | +| ------------------- | ------------------------------------------------ | +| noautoinjectruntime | Disable the autoinjection of `/wails/runtime.js` | +| noautoinjectipc | Disable the autoinjection of `/wails/ipc.js` | +| noautoinject | Disable all autoinjection of scripts | + +Multiple options may be used provided they are comma seperated. + +This code is perfectly valid and operates the same as the autoinjection version: + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx new file mode 100644 index 00000000000..9319fbb2371 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx @@ -0,0 +1,127 @@ +# IDEs + +Wails aims to provide a great development experience. To that aim, we now support generating IDE specific configuration to provide smoother project setup. + +Currently, we support [Visual Studio Code](https://code.visualstudio.com/) but aim to support other IDEs such as Goland. + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +When generating a project using the `-ide vscode` flags, IDE files will be created alongside the other project files. Эти файлы помещаются в директорию `.vscode` и предоставляют правильную конфигурацию для отладки вашего приложения. + +The 2 files generated are `tasks.json` and `launch.json`. Below are the files generated for the default vanilla project: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### Configuring the install and build steps + +The `tasks.json` file is simple for the default project as there is no `npm install` or `npm run build` step needed. For projects that have a frontend build step, such as the svelte template, we would need to edit `tasks.json` to add the install and build steps: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +:::info Future Enhancement + +In the future, we hope to generate a `tasks.json` that includes the install and build steps automatically. + +::: diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..10750baaf09 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Поддержка Linux + +## Обзор + +Wails предлагает поддержку Linux, но составить инструкции по установке для всех доступных дистрибутивов - это невыполнимая задача. Вместо этого Wails пытается определить, доступны ли пакеты для разработки через системный менеджер пакетов. В настоящее время мы поддерживаем следующие менеджеры пакетов: + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## Добавление имен пакетов + +Может возникнуть ситуация, когда ваш дистрибутив использует один из поддерживаемых менеджеров пакетов, но имя пакета отличается от названия. Например, вы можете использовать производную от Ubuntu систему, но имя пакета для gtk может быть другим. Wails пытается найти правильный пакет, ища по списку имен пакетов. Список пакетов хранится в файле (packagemanager) в каталоге `v2/internal/system/packagemanager`. В нашем примере это `v2/internal/system/packagemanager/apt.go`. + +В этом файле список пакетов определяется методом `Packages()`: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +Предположим, что в нашем Linux-дистрибутиве `libgtk-3` упакован под именем `lib-gtk3-dev`. Мы могли бы добавить поддержку для этого, добавив следующую строку: + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## Добавление новых менеджеров пакетов + +Чтобы добавить новый менеджер пакетов, выполните следующие действия: + +- Создайте новый файл в `v2/internal/system/packagemanager` под названием `.go`, где `` это имя менеджера пакетов. +- Определите структуру, которая соответствует интерфейсу менеджера пакетов, определенному в `pm.go`: + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()` должен возвращать имя менеджера пакетов +- `Packages()` должен возвращать `packagemap`, который предоставляет имена файлов для зависимостей +- `PackageInstalled()` должен возвращать `true`, если данный пакет установлен +- `PackageAvailable()` должен возвращать `true`, если данный пакет не установлен, но доступен для установки +- `InstallCommand()` должен возвращать точную команду для установки данного пакета + +Взгляните на другие коды менеджеров пакетов, чтобы получить представление о том, как это работает. + +::info Помните + +Если вы добавите поддержку для нового менеджера пакетов, не забудьте также обновить эту страницу! + +::: diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx new file mode 100644 index 00000000000..229c282bf55 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +This page has miscellaneous guides related to developing Wails applications for Linux. + +## Video tag doesn't fire "ended" event + +When using a video tag, the "ended" event is not fired when the video is finished playing. This is a bug in WebkitGTK, however you can use the following workaround to fix it: + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +Source: [Lyimmi](https://github.com/Lyimmi) on the [discussions board](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx new file mode 100644 index 00000000000..4ba1f34c37d --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx @@ -0,0 +1,55 @@ +# Local Development + +## Overview + +Wails is in constant development and new releases are regularly "tagged". This usually happens when all the newer code on `master` has been tested and confirmed working. If you need a bugfix or feature that has not yet made it to a release, it's possible to use the latest "bleeding edge" version using the following steps: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +NOTE: The directory that you cloned the project into will now be called "clonedir". + +The Wails CLI will now be at the very latest version. + +### Updating your project + +To update projects to use the latest version of the Wails library, update the project's `go.mod` and ensure the following line is at the bottom of the file: + +`replace github.com/wailsapp/wails/v2 => ` + +Example: + +On Windows: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +On 'nix: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +To revert to a stable version, run: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## Testing a Branch + +If you want to test a branch, follow the instructions above, but ensure you switch the branch you want to test before installing: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +Make sure you [update your project](#updating-your-project) as described above. + +## Testing a PR + +If you want to test a PR, follow the instructions above, but ensure you fetch the PR and switch the branch before installing. Please replace `[IDofThePR]` with the ID of the PR shown on github.com: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +Make sure you [update your project](#updating-your-project) as described above. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx new file mode 100644 index 00000000000..961595711c7 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Mac App Store Guide + +This page gives a brief overview of how to submit your Wails App to the Mac App Store. + +## Prerequisites + +- You will need to have an Apple Developer account. Please find more information on the [Apple Developer Program](https://developer.apple.com/support/compare-memberships/) site +- You will need to have your Certificates, Identifiers, and App created on the developer portal. More on this below +- Xcode command line tools will need to be installed on your local machine + +#### Create Certificates and Identifiers + +1. Go to your [Apple Developer Account](https://developer.apple.com/account/) +2. Under `Certificates, Identifiers & Profiles`, click `Identifiers` and Register a New App ID. Use the format (com.example.app) +3. Under the same page click `Certificates` and generate new Certificates for Mac App Store Distribution. Download them and import the certificates into Keychain on your local machine. + +#### Create App Submission + +1. Go to the [App Store Connect Site](https://appstoreconnect.apple.com/apps) +2. Register a new application and link the bundle ID that you created in the previous step +3. Populate your app with the correct screen shots, descriptions, etc. as required by Apple +4. Create a new version of your app + +#### Create Provisioning Profile +1. Go to the [Apple Developer Profiles](https://developer.apple.com/account/resources/profiles/list) page +2. Add a new provisioning profile for Mac App Store Distribution +3. Set the Profile Type as Mac and select the App ID for the application created above +4. Select the Mac App Distribution certificate +5. Name the Provisioning Profile embedded and download the created profile. + +## Mac App Store Process + +#### Enable Apple's App Sandbox + +Apps submitted to the Mac App Store must run under Apple's [App Sandbox](https://developer.apple.com/app-sandboxing/). You must create an `entitlements.plist` file for this to work. The recommendation is to create this file under this path `{PROJECT_DIR}/build/darwin/entitlements.plist`. + +**Example Entitlements File** + +This is an example entitlements file from the [RiftShare](https://github.com/achhabra2/riftshare) app. For reference please put in the entitlements your app requires. Refer to [this site](https://developer.apple.com/documentation/bundleresources/entitlements) for more information. You will need to replace the Team ID and Application Name with the ones you registered above. + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +**Add the Embedded Provisioning Profile** The Provisioning Profile created above needs to be added to the root of the applicaton. It needs to be named embedded.provisionprofile. + +#### Build and Sign the App Package + +The following is an example script for building and signing your app for Mac App Store submission. It assumes you are running the script from your root project directory. + +Note the certificates for signing the app and signing the installer are different. Please make sure both are imported into Keychain. Find the strings in Keychain and insert them below. Populate your certificate names, and app name below. Running the following script will generate a signed `app.pkg` file in the root directory of your app. + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### Upload App Bundle + +You will need to upload the generated package file and associate it to your Application before you will be able to submit it for review. + +1. Download the [Transporter App](https://apps.apple.com/us/app/transporter/id1450874784) from the Mac App Store +2. Open it and sign in with your Apple ID +3. Click the + sign and select the `APP_NAME.pkg` file that you generated in the previous step. Upload it +4. Go back to the [App Store Connect](https://appstoreconnect.apple.com/apps) site and navigate back into your app submission. Select the version that you are ready to make available on the App Store. Under `Build` select the package that you uploaded via Transporter. + +That's it! You can now use the site to submit your App for review. After a few business days if all goes well you should see your App live on the Mac App Store. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx new file mode 100644 index 00000000000..dcf192d337f --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# Manual Builds + +The Wails CLI does a lot of heavy lifting for the project, but sometimes it's desirable to manually build your project. This document will discuss the different operations the CLI does and how this may be achieved in different ways. + +## Build Process + +When either `wails build` or `wails dev` are used, the Wails CLI performs a common build process: + + - Install frontend dependencies + - Build frontend project + - Generate build assets + - Compile application + - [optional] Compress application + +### Install frontend dependencies + +#### CLI Steps + +- If the `-s` flag is given, this step is skipped +- Checks `wails.json` to see if there is an install command in the key `frontend:install` +- If there isn't, it skips this step +- If there is, it checks if `package.json` exists in the frontend directory. If it doesn't exist, it skips this step +- An MD5 sum is generated from the `package.json` file contents +- It checks for the existence of `package.json.md5` and if it exists, will compare the contents of it (an MD5 sum) with the one generated to see if the contents have changed. If they are the same, this step is skipped +- If `package.json.md5` does not exist, it creates it using the generated MD5 sum +- If a build is now required, or `node_modules` does not exist, or the `-f` flag is given, the install command is executed in the frontend directory + +#### Manual Steps + +This step could be done from the command line or a script with `npm install`. + +### Build frontend project + +#### Wails CLI + +- If the `-s` flag is given, this step is skipped +- Checks `wails.json` to see if there is a build command in the key `frontend:build` +- If there isn't, it skips this step +- If there is, it is executed in the frontend directory + +#### Manual Steps + +This step could be done from the command line or a script with `npm run build` or whatever the frontend build script is. + +### Generate assets + +#### Wails CLI + +- If `-nopackage` flag is set, this stage is skipped +- If the `build/appicon.png` file does not exist, a default one is created +- For Windows, see [Bundling for Windows](#windows) +- If `build/windows/icon.ico` does not exist, it will create it from the `build/appicon.png` image. + +##### Windows + +- If `build/windows/icon.ico` does not exist, it will create it from `build/appicon.png` using icon sizes of 256, 128, 64, 48, 32 and 16. This is done using [winicon](https://github.com/leaanthony/winicon). +- If the `build/windows/.manifest` file does not exist, it creates it from a default version. +- Compiles the application as a production build (above) +- Uses [winres](https://github.com/tc-hib/winres) to bundle the icon and manifest into a `.syso` file ready for linking. + +#### Manual Steps + +- Create `icon.ico` using the [winicon](https://github.com/leaanthony/winicon) CLI tool (or any other tool). +- Create / Update a `.manifest` file for your application +- Use the [winres CLI](https://github.com/tc-hib/go-winres) to generate a `.syso` file. + +### Compile application + +#### Wails CLI + +- If the `-clean` flag is provided, the `build` directory is deleted and recreated +- For `wails dev`, the following default Go flags are used: `-tags dev -gcflags "all=-N -l"` +- For `wails build`, the following default Go flags are used: `-tags desktop,production -ldflags "-w -s"` + - On Windows, `-ldflags "-w -h -H windowsgui"` +- Additional tags passed to the CLI using `-tags` are added to the defaults +- Additional ldflags passed to the CLI using `-ldflags` are added to the defaults +- The `-o` flag is passed through +- The Go compiler specified by `-compiler` will be used for compilation + +#### Manual steps + +- For dev build, the minimum command would be: `go build -tags dev -gcflags "all=-N -l"` +- For production build, the minimum command would be: `go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- Ensure that you compile in the same directory as the `.syso` file + +### Compress application + +#### Wails CLI + +- If the `-upx` flag has been given, the `upx` program will be run to compress the application with the default settings +- If `-upxflags` is also passed, these flags are used instead of the default ones + +#### Manual steps + +- Run `upx [flags]` manually to compress the application. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx new file mode 100644 index 00000000000..7123cbe6b60 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx @@ -0,0 +1,191 @@ +# Migrating from v1 + +## Overview + +Wails v2 is a significant change from v1. This document aims to highlight the changes and the steps in migrating an existing project. + +### Creating the Application + +In v1, the main application is created using `wails.CreateApp`, bindings are added with `app.Bind`, then the application is run using `app.Run()`. + +Example: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +In v2, there is just a single method, `wails.Run()`, that accepts [application options](../reference/options.mdx#application-options). + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### Binding + +In v1, it was possible to bind both arbitrary functions and structs. In v2, this has been simplified to only binding structs. The struct instances that were previously passed to the `Bind()` method in v1, are now specified in the `Bind` field of the [application options](../reference/options.mdx#application-options): + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +In v1, bound methods were available to the frontend at `window.backend`. This has changed to `window.go`.`` + +### Application Lifecycle + +In v1, there were 2 special methods in a bound struct: `WailsInit()` and `WailsShutdown()`. These have been replaced with 3 lifecycle hooks as part of the [application options](../reference/options.mdx#application-options): + +- [OnStartup](../reference/options.mdx#onstartup) +- [OnShutdown](../reference/options.mdx#onshutdown) +- [OnDomReady](../reference/options.mdx#ondomready) + +Note: [OnDomReady](../reference/options.mdx#ondomready) replaces the `wails:ready` system event in v1. + +These methods can be standard functions, but a common practice is to have them part of a struct: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### Runtime + +The runtime in v2 is much richer than v1 with support for menus, window manipulation and better dialogs. The signature of the methods has changed slightly - please refer the the [Runtime Reference](../reference/runtime/intro.mdx). + +In v1, the [runtime](../reference/runtime/intro.mdx) was available via a struct passed to `WailsInit()`. In v2, the runtime has been moved out to its own package. Each method in the runtime takes the `context.Context` that is passed to the [OnStartup](../reference/options.mdx#onstartup) method. + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startup is called at application startup +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} + +``` + +### Assets + +The _biggest_ change in v2 is how assets are handled. + +In v1, assets were passed via 2 application options: + +- `JS` - The application's JavaScript +- `CSS` - The application's CSS + +This meant that the responsibility of generating a single JS and CSS file was on the developer. This essentially required the use of complicated packers such as webpack. + +In v2, Wails makes no assumptions about your frontend assets, just like a webserver. All of your application assets are passed to the application options as an `embed.FS`. + +**This means there is no requirement to bundle your assets, encode images as Base64 or attempt the dark art of bundler configuration to use custom fonts**. + +At startup, Wails will scan the given `embed.FS` for `index.html` and use its location as the root path for all the other application assets - just like a webserver would. + +Example: An application has the following project layout. All final assets are placed in the `frontend/dist` directory: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +Those assets may be used by the application by simply creating an `embed.FS`: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +Of course, bundlers can be used if you wish to. The only requirement is to pass the final application assets directory to Wails using an `embed.FS` in the `Assets` key of the [application options](../reference/options.mdx#application-options). + +### Project Configuration + +In v1, the project configuration was stored in the `project.json` file in the project root. In v2, the project configuration is stored in the `wails.json` file in the project root. + +The format of the file is slightly different. Here is a comparison: + +

+ +| v1 | v2 | Notes | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | Removed | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | Changed | +| frontend / dir | | Removed | +| frontend / install | frontend:install | Changed | +| frontend / build | frontend:build | Changed | +| frontend / bridge | | Removed | +| frontend / serve | | Removed | +| tags | | Removed | +| | wailsjsdir | The directory to generate wailsjs modules | +| | assetdir | The directory of the compiled frontend assets for `dev` mode. This is normally inferred and could be left empty. | +| | reloaddirs | Comma separated list of additional directories to watch for changes and to trigger reloads in `dev` mode. This is only needed for some more advanced asset configurations. | + +

diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..1655b380018 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# Клавиши мыши + +Время выполнения кода Wails перехватывает щелчки мыши для определения необходимости изменения размера окна или перемещения окна. Был задан вопрос о том, как определить, когда произошел клик мышки, потому что `window.onclick` не сообщит о кнопках мыши. Следующий код показывает, как обнаружить клики мышкой: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +Источник: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx new file mode 100644 index 00000000000..8094a0324f1 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# Обфускация кода + +Wails поддерживает обфускацию вашего приложения с помощью [grable](https://github.com/burrowers/garble). + +Чтобы обфусцировать ваше приложение, вы можете добавить флаг `obfuscate` к команде `wails build`: + +```bash +wails build -obfuscated +``` + +Для изменения настроек обфускации, вы можете использовать флаг `garbleargs`: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +These settings may be persisted in your [project config](../reference/project-config). + +## Как это работает + +При стандартной сборке, все привязанные методы доступны в фронтенде через переменную `window.go`. При вызове этих методов происходит вызов соответствущих методов в бекенде, для этого используется имя функции. При использовании обфускации, методы привязываются по ID, а не по имени. Привязки, сгенерированные в папке `wailsjs` используют эти ID для вызова метода в бекенде. + +:::note + +Чтобы убедиться, что ваше приложение будет работать в режиме обфускации, используйте сгенерированные в папке `wailsjs` привязки. + +::: + +## Example + +Импортирование метода "Greet" из биндинга: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +обеспечит корректную работу с включенным режимом обфускации, так как привязки будут сгенерированы заново, используя новый механизм вызова метода по ID. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx new file mode 100644 index 00000000000..915633d0cea --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# Overscroll + +[Overscroll](https://developer.mozilla.org/en-US/docs/Web/CSS/overscroll-behavior) - это "эффект отскоков", который иногда получается, когда вы прокручиваете за границы содержимого страницы. Это часто используется в мобильных приложениях. Это можно отключить с помощью CSS: + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx new file mode 100644 index 00000000000..acb1eb656d6 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx @@ -0,0 +1,47 @@ +# Маршрутизация + +Маршрутизация - это популярный способ переключения шаблонов в приложении. Эта страница предлагает некоторые рекомендации о том, как это сделать. + +## Vue + +Рекомендуемый подход к маршрутизации в Vue — [Hash Mode](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode): + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +Рекомендуемый подход к маршрутизации в Angular - [HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies#_hashlocationstrategy): + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +The recommended approach for routing in React is [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx new file mode 100644 index 00000000000..4c7cf45ba99 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx @@ -0,0 +1,387 @@ +# Code Signing + +This is a guide on how you can sign your binaries generated with Wails on MacOS and Windows. The guide will target CI environments, more specifically GitHub Actions. + +## Windows + +First off you need a code signing certificate. If you do not already have one, Microsoft's info page lists some providers [here](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate). Please note that an EV certificate is not required unless you need to write kernel-level software such as device drivers. For signing your Wails app, a standard code signing certificate will do just fine. + +It may be a good idea to check with your certificate provider how to sign your binaries on your local machine before targeting automated build systems, just so you know if there are any special requirements. For instance, [here](https://www.ssl.com/how-to/using-your-code-signing-certificate/) is SSL.com's code signing guide for Windows. If you know how to sign locally, it will be easier to troubleshoot any potential issues in a CI environment. For instance, SSL.com code signing certificates require the `/tr` flag for [SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool) while other providers may only need the `/t` flag for providing the timestamping server. Popular GitHub Actions for signing Windows binaries like [this one](https://github.com/Dana-Prajea/code-sign-action) does not support the `/tr` flag on SignTool.exe. Therefore this guide will focus on signing our app manually with PowerShell commands, but you can use actions like the [code-sign-action](https://github.com/Dana-Prajea/code-sign-action) Action if you prefer. + +First off, let's make sure we are able to build our Wails app in our GitHub CI. Here is a small workflow template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend manually here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +Next we need to give the GitHub workflow access to our signing certificate. This is done by encoding your .pfx or .p12 certificate into a base64 string. To do this in PowerShell, you can use the following command assuming your certificate is called 'my-cert.p12': + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +You should now have your .txt file with the base64 encoded certificate. It should start with _-----BEGIN CERTIFICATE-----_ and end with _-----END CERTIFICATE-----_. Now you need to make two action secrets on GitHub. Navigate to _Settings -> Secrets -> Actions_ and create the two following secrets: + +- **WIN_SIGNING_CERT** with the contents of your base64 encoded certificate text. +- **WIN_SIGNING_CERT_PASSWORD** with the contents of your certificate password. + +Now we're ready to implement the signing in our workflow using one of the two methods: + +### Method 1: signing with commands + +This method uses PowerShell commands to sign our app, and leaves you control over the entire signing process. + +After the `"Build Wails app"` step, we can add the following step to our workflow: + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +This script creates a new directory for your certificate file, creates the certificate file from our base64 secret, converts it to a .pfx file, and finally signs the binary. The following variables needs to be replaced in the last line: + +- **signing algorithm**: usually sha256. +- **timestamping server**: URL to the timestamping server to use with your certificate. +- **path to binary**: path to the binary you want to sign. + +Given that our Wails config has `outputfilename` set to "app.exe" and that we have a certificate from SSL.com, this would be our workflow: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### Method 2: automatically signing with Action + +It is possible to use a Windows code signing Action like [this](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) one, but note it requires a SHA1 hash for the certificate and a certificate name. View an example of how to configure it on the Action's [marketplace](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate). + +--- + +## MacOS + +First off you need your code signing certificate from Apple. If you do not have one, a simple Google search will help you acquire one. Once you have your certificate, you need to export it and encode it to base64. [This tutorial](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions) shows you how to do that in an easy manner. Once you have exported your .p12 certificate file, you can encode it to base64 as seen in the tutorial with the following command: + +```bash +base64 Certificates.p12 | pbcopy +``` + +Now you're ready to create some GitHub project secrets, just as with Windows: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64** with the contents of your newly copied base64 certificate. +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD** with the contents of your certificate password. +- **APPLE_PASSWORD** with the contents of an App-Specific password to your Apple-ID account which you can generate [here](https://appleid.apple.com/account/manage). + +Let's make sure we are able to build our Wails app in our GitHub Action workflow. Here is a small template: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +For code signing on macOS, [gon](https://github.com/mitchellh/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and will be used in this guide. + +After the `Build Wails app` step, add the following to the workflow: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +Now we need to configure some gon config files in our `build/darwin` directory: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +Where `source` is your Wails binary, `bundle_id` is your bundle ID, `apple_id` contains your Apple ID username and App-Specific password which you created earlier, and `sign.application_identity` is your identity which you can find by running the following command: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +In this file you configure the entitlements you need for you app, e.g. camera permissions if your app uses the camera. Read more about entitlements [here](https://developer.apple.com/documentation/bundleresources/entitlements). + +Make sure you have updated your `Info.plist` file with the same bundle ID as you entered in `gon-sign.json`. Here's an example `Info.plist` file: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +Now we're ready to add the signing step in our workflow after building the Wails app: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +Please note that signing binaries with Apple could take anywhere from minutes to hours. + +## Combined workflow file: + +Here is our GitHub workflow file with Windows + macOS combined: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# End notes + +This guide inspired by the RiftShare project and its workflow, which is highly recommended to check out [here](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml). diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..4651c422ed1 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +This guide will go into: + +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. + +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. + +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. + +- `npm create svelte@latest frontend` + +##### Modify wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. + +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Install/remove dependencies using your favorite package manager. + +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js + +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. + +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. + +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. + +- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! + +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html + +``` + +... + + + +... + +``` + +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. + +- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling + +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions + +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx new file mode 100644 index 00000000000..23e24c4c8dc --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx @@ -0,0 +1,97 @@ +# Templates + +Wails генерирует проекты из заранее созданных шаблонов. В v1 это было трудно поддерживать набор проектов, которые были подвержены устареванию. В v2 для расширения прав и возможностей сообщества добавлено несколько новых возможностей для шаблонов: + +- Возможность создания проектов из [удаленных шаблонов](../reference/cli.mdx#remote-templates) +- Инструменты, которые помогут создать собственные шаблоны + +## Создание шаблонов + +Для создания шаблона вы можете использовать команду `wails generate template`. Для создания шаблона по умолчанию запустите: + +`wails generate template -name mytemplate` + +Это создает каталог "mytemplate" с файлами по умолчанию: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### Обзор шаблона + +Шаблон по умолчанию состоит из следующих файлов и директорий: + +| Имя файла / Каталог | Описание | +| ------------------- | ------------------------------------ | +| NEXTSTEPS.md | Инструкции по заполнению шаблона | +| README.md | README публикуемый с шаблоном | +| app.tmpl.go | `app.go` файл шаблона | +| frontend/ | Каталог, содержащий файлы интерфейса | +| go.mod.tmpl | `go.mod` файл шаблона | +| main.tmpl.go | `main.go` файл шаблона | +| template.json | Метаданные шаблона | +| wails.tmpl.json | `wails.json` файл шаблона | + +На данный момент желательно выполнить шаги в `NEXTSTEPS.md`. + +## Создать шаблон из существующего проекта + +Можно создать шаблон из существующего проекта фронтенда, передав путь к проекту при генерации шаблона. Теперь мы рассмотрим как создать шаблон Vue 3: + +- Установите vue cli: `npm install -g @vue/cli` +- Создайте проект по умолчанию: `vue create vue3-base` + - Выберите `Default (Vue 3) ([Vue 3] babel, eslint)` +- После создания проекта, выполните: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- Шаблон теперь может быть настроен как указано в файле `NEXTSTEPS.md` +- Как только файлы будут готовы, их можно протестировать, запустив: `wails init -n my-vue3-project -t .\wails-vue3-template\` +- Для тестирования нового проекта выполните: `cd my-vue3-project` затем `wails build` +- Как только проект будет скомпилирован, запустите его: `.\build\bin\my-vue3-project.exe` +- У вас должно получиться полностью работающее приложение Vue3: + +```mdx-code-block +
+ +
+``` + +## Публикация шаблона + +Для публикации шаблона просто загрузите файлы на GitHub. Поощряется следующая лучшая практика: + +- Удалите любые нежелательные файлы и каталоги (например, `.git`) из вашей frontend директории +- Убедитесь, что `template.json` завершен, особенно `helpurl` +- Отправить файлы на GitHub +- Создайте PR на странице [Шаблоны сообщества](../community/templates.mdx) +- Расскажите о своем шаблоне на [Template Announcement](https://github.com/wailsapp/wails/discussions/825) форуме diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..1c3b6a9cc52 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# Troubleshooting + +An assortment of troubleshooting tips. + +## The `wails` command appears to be missing? + +If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment variable. You will also normally need to close and reopen any open command prompts so that changes to the environment made by the installer are reflected at the command prompt. + +## My application is displaying a white/blank screen + +Check that your application includes the assets from the correct directory. In your `main.go` file, you will have something similar to the following code: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +Check that `frontend/dist` contains your application assets. + +### Mac + +If this happens on Mac, try adding the following to your `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Mac application not valid + +If your built application looks like this in finder: + +```mdx-code-block +

+ +

+``` + +it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to the `build/darwin` directory. + +## My application is not displaying the correct icon in Windows Explorer + +If your application is not displaying the correct icon, try deleting the hidden `IconCache.db` file located in the `C:\Users\<your username>\AppData\Local` directory. This will force Windows to rebuild the icon cache. + +Source: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## Cannot call backend method from frontend with variadic arguments + +If you have a backend method defined with variadic parameters, eg: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +calling this method from the frontend like this will fail: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Workaround: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Credit: https://github.com/wailsapp/wails/issues/1186 + +## I'm having getting proxy errors when trying to install Wails + +If you are getting errors like this: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +it's probably because the official Go Proxy is being blocked (Users in China have reported this). The solution is to set up the proxy manually, eg: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +Source: https://github.com/wailsapp/wails/issues/1233 + +## The generated TypeScript doesn't have the correct types + +Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, it is possible to specify what types should be generated using the `ts_type` struct tag. For more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). + +## When I navigate away from `index.html`, I am unable to call methods on the frontend + +If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding the following imports to the `` section of any new page you navigate to: + +```html + + + + +``` + +Source: https://github.com/wailsapp/wails/discussions/1512 + +## I get `too many open files` errors on my Mac when I run `wails dev` + +By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. This limit can be increased by running: `ulimit -n 1024` in the terminal. + +FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). + +## My Mac app gives me weird compilation errors + +A few users have reported seeing compilation errors such as the following: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. + +If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: + +`xcode-select -p` + +If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" + +It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine that may have different versions of Node installed, you may not be able to run your application. + +If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. + +## Build process stuck on "Generating bindings" + +Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx new file mode 100644 index 00000000000..539c52ba582 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +Эта страница предназначена для разных советов и трюков при использовании Visual Studio Code с Wails. + +## Конфигурация Vetur + +Большое спасибо [@Lyimmi](https://github.com/Lyimmi) за этот совет. Изначально оставил сообщение [здесь](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349). + +Vetur - это популярный плагин для кода Visual Studio, который предоставляет подсветку синтаксиса и завершение кода для Vue проектов. При загрузке проекта Wails в VSCode, Ветур выбросит ошибку, ожидая найти проект фронтенда в корневом каталоге. Чтобы исправить это, можно сделать следующее: + +Создайте файл `vetur.config.js` в корне проекта. + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +Затем настройте `frontend/tsconfig.json`: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +Это должно позволить теперь использовать Vetur как ожидалось. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx new file mode 100644 index 00000000000..b332c5f1dc0 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# NSIS installer + +```mdx-code-block +

+ +
+

+``` + +Wails supports generating Windows installers using the [NSIS installer](https://nsis.sourceforge.io/). + +## Installing NSIS + +### Windows + +The installer is available on the [NSIS Download](https://nsis.sourceforge.io/Download) page. + +If you use the chocolatey package manager, run the following script: + +``` +choco install nsis +``` + +If you install NSIS manually, you need to add the _Bin_ folder, which contains `makensis.exe`, in your NSIS installation to your path. [Here](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/) is a good tutorial on how to add to path on Windows. + +### Linux + +Пакет `nsis` должен быть доступен через менеджер пакетов вашего дистрибутива. + +### MacOS + +NSIS is available to install through homebrew: `brew install nsis`. + +## Generating the installer + +When a new project is created, Wails generates the NSIS configuration files in `build/windows/installer`. The config data is read from `installer/info.json` and that is configured to use the project's `wails.json` Info section: + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +To generate an installer for your application, use the `-nsis` flag with `wails build`: + +``` +wails build -nsis +``` + +The installer will now be available in the `build/bin` directory. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx new file mode 100644 index 00000000000..821808c0b8d --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +This page has miscellaneous guides related to developing Wails applications for Windows. + +## Handling the WebView2 Runtime Dependency + +Wails applications built for Windows have a runtime requirement on the Microsoft [WebView2 Runtime](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). Windows 11 will have this installed by default, but some machines won't. Wails offers an easy approach to dealing with this dependency. + +By using the `-webview2` flag when building, you can decide what your application will do when a suitable runtime is not detected (including if the installed runtime is too old). The four options are: + +1. Download +2. Embed +3. Browser +4. Error + +### Download + +This option will prompt the user that no suitable runtime has been found and then offer to download and run the official bootstrapper from Microsoft's WebView2 site. If the user proceeds, the official bootstrapper will be downloaded and run. + +### Embed + +This option embeds the official bootstrapper within the application. If no suitable runtime has been found, the application will offer to run the bootstrapper. This adds ~150k to the binary size. + +### Browser + +This option will prompt the user that no suitable runtime has been found and then offer to open a browser to the official WebView2 page where the bootstrapper can be downloaded and installed. The application will then exit, leaving the installation up to the user. + +### Error + +If no suitable runtime is found, an error is given to the user and no further action taken. + +## Fixed version runtime + +Another way of dealing with webview2 dependency is shipping it yourself. You can download [fixed version runtime](https://developer.microsoft.com/microsoft-edge/webview2/#download-section) and bundle or download it with your application. + +Also, you should specify path to fixed version of webview2 runtime in the `windows.Options` structure when launching wails. + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +Note: When `WebviewBrowserPath` is specified, `error` strategy will be forced in case of minimal required version mismatch or invalid path to a runtime. + +## Spawning other programs + +When spawning other programs, such as scripts, you will see the window appear on the screen. To hide the window, you can use the following code: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +Solution provided by [sithembiso](https://github.com/sithembiso) on the [discussions board](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172). diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx new file mode 100644 index 00000000000..44fa130cc74 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# How does it work? + +A Wails application is a standard Go application, with a webkit frontend. The Go part of the application consists of the application code and a runtime library that provides a number of useful operations, like controlling the application window. The frontend is a webkit window that will display the frontend assets. Also available to the frontend is a JavaScript version of the runtime library. Finally, it is possible to bind Go methods to the frontend, and these will appear as JavaScript methods that can be called, just as if they were local JavaScript methods. + +```mdx-code-block +
+ +
+``` + +## The Main Application + +### Overview + +The main application consists of a single call to `wails.Run()`. It accepts the application configuration which describes the size of the application window, the window title, what assets to use, etc. A basic application might look like this: + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### Options rundown + +This example has the following options set: + +- `Title` - The text that should appear in the window's title bar +- `Width` & `Height` - The dimensions of the window +- `Assets` - The application's frontend assets +- `OnStartup` - A callback for when the window is created and is about to start loading the frontend assets +- `OnShutdown` - A callback for when the application is about to quit +- `Bind` - A slice of struct instances that we wish to expose to the frontend + +A full list of application options can be found in the [Options Reference](reference/options). + +#### Assets + +The `Assets` option is mandatory as you can't have a Wails application without frontend assets. Those assets can be any files you would expect to find in a web application - html, js, css, svg, png, etc. **There is no requirement to generate asset bundles** - plain files will do. When the application starts, it will attempt to load `index.html` from your assets and the frontend will essentially work as a browser from that point on. It is worth noting that there is no requirement on where in the `embed.FS` the files live. It is likely that the embed path uses a nested directory relative to your main application code, such as `frontend/dist`: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +At startup, Wails will iterate the embedded files looking for the directory containing `index.html`. All other assets will be loaded relative to this directory. + +As production binaries use the files contained in `embed.FS`, there are no external files required to be shipped with the application. + +When running in development mode using the `wails dev` command, the assets are loaded off disk, and any changes result in a "live reload". The location of the assets will be inferred from the `embed.FS`. + +More details can be found in the [Application Development Guide](guides/application-development.mdx). + +#### Application Lifecycle Callbacks + +Just before the frontend is about to load `index.html`, a callback is made to the function provided in [OnStartup](reference/options.mdx#onstartup). A standard Go context is passed to this method. This context is required when calling the runtime so a standard pattern is to save a reference to in this method. Just before the application shuts down, the [OnShutdown](reference/options.mdx#onshutdown) callback is called in the same way, again with the context. There is also an [OnDomReady](reference/options.mdx#ondomready) callback for when the frontend has completed loading all assets in `index.html` and is equivalent of the [`body onload`](https://www.w3schools.com/jsref/event_onload.asp) event in JavaScript. It is also possible to hook into the window close (or application quit) event by setting the option [OnBeforeClose](reference/options.mdx#onbeforeclose). + +#### Method Binding + +The `Bind` option is one of the most important options in a Wails application. It specifies which struct methods to expose to the frontend. Think of structs like "controllers" in a traditional web application. When the application starts, it examines the struct instances listed in the `Bind` field in the options, determines which methods are public (starts with an uppercase letter) and will generate JavaScript versions of those methods that can be called by the frontend code. + +:::info Note + +Wails requires that you pass in an _instance_ of the struct for it to bind it correctly + +::: + +In this example, we create a new `App` instance and then add this instance to the `Bind` option in `wails.Run`: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +You may bind as many structs as you like. Just make sure you create an instance of it and pass it in `Bind`: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +When you run `wails dev` (or `wails generate module`), a frontend module will be generated containing the following: + +- JavaScript bindings for all bound methods +- TypeScript declarations for all bound methods +- TypeScript definitions for all Go structs used as inputs or outputs by the bound methods + +This makes it incredibly simple to call Go code from the frontend, using the same strongly typed datastructures. + +## The Frontend + +### Overview + +The frontend is a collection of files rendered by webkit. It's like a browser and webserver in one. There is virtually[^1] no limit to which frameworks or libraries you can use. The main points of interaction between the frontend and your Go code are: + +- Calling bound Go methods +- Calling runtime methods + +### Calling bound Go methods + +When you run your application with `wails dev`, it will automatically generate JavaScript bindings for your structs in a directory called `wailsjs/go` (You can also do this by running `wails generate module`). The generated files mirror the package names in your application. In the example above, we bind `app`, which has one public method `Greet`. This will lead to the generation of the following files: + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +Here we can see that there is a `main` package that contains the JavaScript bindings for the bound `App` struct, as well as the TypeScript declaration file for those methods. To call `Greet` from our frontend, we simply import the method and call it like a regular JavaScript function: + +```javascript +// ... +import { Greet } from "../wailsjs/go/main/App"; + +function doGreeting(name) { + Greet(name).then((result) => { + // Do something with result + }); +} +``` + +The TypeScript declaration file gives you the correct types for the bound methods: + +```ts +export function Greet(arg1: string): Promise; +``` + +The generated methods return a Promise. A successful call will result in the first return value from the Go call to be passed to the `resolve` handler. An unsuccessful call is when a Go method that has an error type as it's second return value, passes an error instance back to the caller. This is passed back via the `reject` handler. In the example above, `Greet` only returns a `string` so the JavaScript call will never reject - unless invalid data is passed to it. + +All data types are correctly translated between Go and JavaScript. Even structs. If you return a struct from a Go call, it will be returned to your frontend as a JavaScript class. + +:::info Note + +Struct fields _must_ have a valid `json` tag to be included in the generated TypeScript. + +Anonymous nested structs are not supported at this time. + +::: + +It is possible to send structs back to Go. Any JavaScript map/class passed as an argument that is expecting a struct, will be converted to that struct type. To make this process a lot easier, in `dev` mode, a TypeScript module is generated, defining all the struct types used in bound methods. Using this module, it's possible to construct and send native JavaScript objects to the Go code. + +There is also support for Go methods that use structs in their signature. All Go structs specified by a bound method (either as parameters or return types) will have TypeScript versions auto generated as part of the Go code wrapper module. Using these, it's possible to share the same data model between Go and JavaScript. + +Example: We update our `Greet` method to accept a `Person` instead of a string: + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +The `wailsjs/go/main/App.js` file will still have the following code: + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +But the `wailsjs/go/main/App.d.ts` file will be updated with the following code: + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +As we can see, the "main" namespace is imported from a new "models.ts" file. This file contains all the struct definitions used by our bound methods. In this example, this is a `Person` struct. If we look at `models.ts`, we can see how the models are defined: + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +So long as you have TypeScript as part of your frontend build configuration, you can use these models in the following way: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +The combination of generated bindings and TypeScript models makes for a powerful development environment. + +More information on Binding can be found in the [Binding Methods](guides/application-development.mdx#binding-methods) section of the [Application Development Guide](guides/application-development.mdx). + +### Calling runtime methods + +The JavaScript runtime is located at `window.runtime` and contains many methods to do various tasks such as emit an event or perform logging operations: + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +More details about the JS runtime can be found in the [Runtime Reference](reference/runtime/intro). + +[^1]: There is a very small subset of libraries that use features unsupported in WebViews. There are often alternatives and workarounds for such cases. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx new file mode 100644 index 00000000000..9afebec4d31 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# Введение + +Wails - это проект, позволяющий писать настольные приложения с использованием Go и web технологий. + +Считайте, что это легкая и быстрая альтернатива Electron для Go. Вы можете легко создавать приложения с гибкостью и мощностью Go, в сочетании с богатым современным фронтендом. + +### Возможности + +- Нативные меню, диалоги, темы и прозрачность +- Поддержка Windows, macOS и linux +- Встроенные шаблоны для Svelte, React, Preact, Vue, Lit и Vanilla JS +- Easily call Go methods from JavaScript +- Automatic Go struct to TypeScript model generation +- Для Windows не требуется CGO или внешние DLL +- Горячая перезагрузка, используя мощь [Vite](https://vitejs.dev/) +- Мощный CLI для простого создания, сборки и упаковки приложений +- Богатая [runtime библиотека](/docs/reference/runtime/intro) +- Приложения, созданные с помощью Wails совместимы с Apple и Microsoft Store + +Например, [varly](https://varly.app) - настольное приложение для MacOS и Windows, написанное с помощью Wails. Оно не только великолепно выглядит, но и использует системные меню и полупрозрачность - все, что можно ожидать от современного нативного приложения. + +```mdx-code-block +

+ + + +

+``` + +### Шаблоны для быстрого начала + +Wails поставляется с рядом предварительно настроенных шаблонов, которые позволяют вам быстро создать и запустить ваше приложение. Есть шаблоны для следующих фреймворков: Svelte, React, Vue, Preact, Lit и Vanilla. There are both JavaScript and TypeScript versions for each template. + +### Системные элементы + +Wails использует специально созданную библиотеку для обработки системных элементов, таких как окна, меню, диалоги и так далее, чтобы вы могли создавать хорошо выглядящие, богатые функционалом приложения. + +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. На Windows это новая библиотека Microsoft Webview2, основанная на Chromium. + +### Go & JavaScript Interoperability + +Wails automatically makes your Go methods available to JavaScript, so you can call them by name from your frontend! It even generates TypeScript models for the structs used by your Go methods, so you can pass the same data structures between Go and JavaScript. + +### Библиотека среды выполнения + +Wails provides a runtime library, for both Go and JavaScript, that handles a lot of the things modern applications need, like Eventing, Logging, Dialogs, etc. + +### Опыт разработки в реальном времени + +#### Автоматическая пересборка + +Когда вы запускаете ваше приложение в режиме разработки, Wails будет собирать его, но читать ресурсы с диска. Он будет отслеживать любые изменения в вашем коде на Go и автоматически пересобирать и перезапускать приложение. + +#### Автоматическая перезагрузка + +Когда будут обнаружены изменения ресурсов вашего приложения, ваше запущенное приложение перезагрузится, почти сразу, отражая ваши изменения. + +#### Разрабатывайте приложение в браузере + +Если вы предпочитаете отладку и разработку в браузере, то Wails это умеет. Запущенное приложение также имеет веб-сервер, который запустит ваше приложение в любом, подключенном к нему, браузере. Он будет перезапускаться когда ресурсы на вашем диске изменятся. + +### Бинарные файлы, готовые к выпуску + +Когда вы готовы сделать финальную сборку вашего приложения, CLI соберет всё в один исполняемый файл со всеми ресурсами внутри. На Windows и MacOS есть возможность создать нативный установочный пакет для распространения. Ресурсы, используемые при упаковке (иконка, info.plist, файл manifest'а, и т.д.) являются частью вашего проекта и могут быть изменены, что даёт вам полный контроль над тем, как вы создаете приложение. + +### Инструментарий + +Wails CLI предоставляет легкий способ генерировать, собирать и упаковывать ваши приложения. Оно создаст иконки, соберет ваше приложение с оптимальными параметрами и создаст готовый для распространения исполняемый файл. Выберите из нескольких стартовых шаблонов для быстрого старта! diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx new file mode 100644 index 00000000000..c7aea10a64d --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx @@ -0,0 +1,240 @@ +--- +sidebar_position: 2 +--- + +# CLI + +The Wails CLI has a number of commands that are used for managing your projects. All commands are run in the following way: + +`wails ` + +## init + +`wails init` is used for generating projects. + +| Flag | Description | Default | +|:------------------ |:----------------------------------------------------------------------------------------------------------------------- |:-------------------:| +| -n "project name" | Name of the project. **Mandatory**. | | +| -d "project dir" | Project directory to create | Name of the project | +| -g | Initialise git repository | | +| -l | List available project templates | | +| -q | Suppress output to console | | +| -t "template name" | The project template to use. This can be the name of a default template or a URL to a remote template hosted on github. | vanilla | +| -ide | Generate IDE project files | | +| -f | Force build application | false | + +Example: `wails init -n test -d mytestproject -g -ide vscode -q` + +This will generate a a project called "test" in the "mytestproject" directory, initialise git, generate vscode project files and do so silently. + +More information on using IDEs with Wails can be found [here](../guides/ides.mdx). + +### Remote Templates + +Remote templates (hosted on GitHub) are supported and can be installed by using the template's project URL. + +Example: `wails init -n test -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +A list of community maintained templates can be found [here](../community/templates.mdx) + +:::warning Attention + +**The Wails project does not maintain, is not responsible nor liable for 3rd party templates!** + +If you are unsure about a template, inspect `package.json` and `wails.json` for what scripts are run and what packages are installed. + +::: + +## build + +`wails build` is used for compiling your project to a production-ready binary. + +| Flag | Description | Default | +|:-------------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:--------------------------------------------------------------------------------------------------------------------------------------------------- | +| -clean | Cleans the `build/bin` directory | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Force build application | | +| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -m | Skip mod tidy before compile | | +| -nopackage | Do not package application | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o filename | Output filename | | +| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | +| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Build with Go's race detector | | +| -s | Skip building the frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | +| -trimpath | Remove all file system paths from the resulting executable. | | +| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | +| -upx | Compress final binary using "upx" | | +| -upxflags | Flags to pass to upx | | +| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | +| -windowsconsole | Keep the console window for Windows builds | | + +For a detailed description of the `webview2` flag, please refer to the [Windows](../guides/windows.mdx) Guide. + +If you prefer to build using standard Go tooling, please consult the [Manual Builds](../guides/manual-builds.mdx) guide. + +Example: + +`wails build -clean -o myproject.exe` + +:::info + +On Mac, the application will be bundled with `Info.plist`, not `Info.dev.plist`. + +::: + +:::info UPX on Apple Silicon + +There are [issues](https://github.com/upx/upx/issues/446) with using UPX with Apple Silicon. + +::: + +:::info UPX on Windows + +Some Antivirus vendors false positively mark `upx` compressed binaries as virus, see [issue](https://github.com/upx/upx/issues/437). + +::: + +### Platforms + +Supported platforms are: + +| Platform | Description | +|:---------------- |:--------------------------------------------- | +| darwin | MacOS + architecture of build machine | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | MacOS AMD64+ARM64 universal application | +| windows | Windows 10/11 + architecture of build machine | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + architecture of build machine | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## doctor + +`wails doctor` will run diagnostics to ensure that your system is ready for development. + +Example: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## dev + +`wails dev` is used to run your application in a "live development" mode. This means: + +- The application's `go.mod` will be updated to use the same version of Wails as the CLI +- The application is compiled and run automatically +- A watcher is started and will trigger a rebuild of your dev app if it detects changes to your go files +- A webserver is started on `http://localhost:34115` which serves your application (not just frontend) over http. This allows you to use your favourite browser development extensions +- All application assets are loaded from disk. If they are changed, the application will automatically reload (not rebuild). All connected browsers will also reload +- A JS module is generated that provides the following: +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- A second JS module is generated that provides a wrapper + TS declaration for the runtime +- On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. + +| Flag | Description | Default | +|:---------------------------- |:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |:--------------------- | +| -appargs "args" | Arguments passed to the application in shell style | | +| -assetdir "./path/to/assets" | Serve assets from the given directory instead of using the provided asset FS | Value in `wails.json` | +| -browser | Opens a browser to `http://localhost:34115` on startup | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debounce | The time to wait for reload after an asset change is detected | 100 (milliseconds) | +| -devserver "host:port" | The address to bind the wails dev server to | "localhost:34115" | +| -extensions | Extensions to trigger rebuilds (comma separated) | go | +| -forcebuild | Force build of application | | +| -frontenddevserverurl "url" | Use 3rd party dev server url to serve assets, EG Vite | "" | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -loglevel "loglevel" | Loglevel to use - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | Turn off colour cli output | false | +| -noreload | Disable automatic reload when assets change | | +| -nosyncgomod | Do not sync go.mod with the Wails version | false | +| -race | Build with Go's race detector | false | +| -reloaddirs | Additional directories to trigger reloads (comma separated) | Value in `wails.json` | +| -s | Skip building the frontend | false | +| -save | Saves the given `assetdir`, `reloaddirs`, `wailsjsdir`, `debounce`, `devserver` and `frontenddevserverurl` flags in `wails.json` to become the defaults for subsequent invocations. | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to compiler (quoted and space separated) | | +| -v | Verbosity level (0 - silent, 1 - standard, 2 - verbose) | 1 | +| -wailsjsdir | The directory to generate the generated Wails JS modules | Value in `wails.json` | + +Example: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +This command will do the following: + +- Build the application and run it (more details [here](../guides/manual-builds.mdx) +- Generate the Wails JS modules in `./frontend/src` +- Watch for updates to files in `./frontend/dist` and reload on any change +- Open a browser and connect to the application + +There is more information on using this feature with existing framework scripts [here](../guides/application-development.mdx#live-reloading). + +## generate + +### template + +Wails uses templates for project generation. The `wails generate template` command helps scaffold a template so that it may be used for generating projects. + +| Flag | Description | +|:---------------- |:------------------------------------------- | +| -name | The template name (Mandatory) | +| -frontend "path" | Path to frontend project to use in template | + +For more details on creating templates, consult the [Templates guide](../guides/templates.mdx). + +### module + +The `wails generate module` command allows you to manually generate the `wailsjs` directory for your application. + +## update + +`wails update` will update the version of the Wails CLI. + +| Flag | Description | +|:------------------ |:------------------------------------- | +| -pre | Update to latest pre-release version | +| -version "version" | Install a specific version of the CLI | + +## version + +`wails version` will simply output the current CLI version. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx new file mode 100644 index 00000000000..ff9a2442281 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# Menus + +It is possible to add an application menu to Wails projects. This is achieved by defining a [Menu](#menu) struct and setting it in the [`Menu`](../reference/options.mdx#menu) application config, or by calling the runtime method [MenuSetApplicationMenu](../reference/runtime/menu.mdx#menusetapplicationmenu). + +An example of how to create a menu: + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +It is also possible to dynamically update the menu, by updating the menu struct and calling [MenuUpdateApplicationMenu](../reference/runtime/menu.mdx#menuupdateapplicationmenu). + +The example above uses helper methods, however it's possible to build the menu structs manually. + +## Menu + +A Menu is a collection of MenuItems: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +For the Application menu, each MenuItem represents a single menu such as "Edit". + +A simple helper method is provided for building menus: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +This makes the layout of the code more like that of a menu without the need to add the menu items manually after creating them. Alternatively, you can just create the menu items and add them to the menu manually. + +## MenuItem + +A MenuItem represents an item within a Menu. + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| Field | Type | Notes | +| ----------- | ------------------------------------ | ------------------------------------------------------------- | +| Label | string | The menu text | +| Accelerator | [\*keys.Accelerator](#accelerator) | Key binding for this menu item | +| Type | [Type](#type) | Type of MenuItem | +| Disabled | bool | Disables the menu item | +| Hidden | bool | Hides this menu item | +| Checked | bool | Adds check to item (Checkbox & Radio types) | +| SubMenu | [\*Menu](#menu) | Sets the submenu | +| Click | [Callback](#callback) | Callback function when menu clicked | +| Role | string | Defines a [role](#role) for this menu item. Mac only for now. | + +### Accelerator + +Accelerators (sometimes called keyboard shortcuts) define a binding between a keystroke and a menu item. Wails defines an Accelerator as a combination or key + [Modifier](#modifier). They are available in the `"github.com/wailsapp/wails/v2/pkg/menu/keys"` package. + +Example: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +Keys are any single character on a keyboard with the exception of `+`, which is defined as `plus`. Some keys cannot be represented as characters so there are a set of named characters that may be used: + +| | | | | +|:-----------:|:-----:|:-----:|:---------:| +| `backspace` | `f1` | `f16` | `f31` | +| `tab` | `f2` | `f17` | `f32` | +| `return` | `f3` | `f18` | `f33` | +| `enter` | `f4` | `f19` | `f34` | +| `escape` | `f5` | `f20` | `f35` | +| `left` | `f6` | `f21` | `numlock` | +| `right` | `f7` | `f22` | | +| `up` | `f8` | `f23` | | +| `down` | `f9` | `f24` | | +| `space` | `f10` | `f25` | | +| `delete` | `f11` | `f36` | | +| `home` | `f12` | `f37` | | +| `end` | `f13` | `f38` | | +| `page up` | `f14` | `f39` | | +| `page down` | `f15` | `f30` | | + +Wails also supports parsing accelerators using the same syntax as Electron. This is useful for storing accelerators in config files. + +Example: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### Modifier + +The following modifiers are keys that may be used in combination with the accelerator key: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +A number of helper methods are available to create Accelerators using modifiers: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +Modifiers can be combined using `keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)`: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### Type + +Each menu item must have a type and there are 5 types available: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +For convenience, helper methods are provided to quickly create a menu item: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +You can also create menu items directly on a menu by using the "Add" helpers: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +A note on radio groups: A radio group is defined as a number of radio menu items that are next to each other in the menu. This means that you do not need to group items together as it is automatic. However, that also means you cannot have 2 radio groups next to each other - there must be a non-radio item between them. + +### Callback + +Each menu item may have a callback that is executed when the item is clicked: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +The function is given a `CallbackData` struct which indicates which menu item triggered the callback. This is useful when using radio groups that may share a callback. + +### Role + +:::info Roles + +Roles are currently supported on Mac only. + +::: + +A menu item may have a role, which is essentially a pre-defined menu item. We currently support the following roles: + +| Role | Description | +| ------------ | ------------------------------------------------------------------------ | +| AppMenuRole | The standard Mac application menu. Can be created using `menu.AppMenu()` | +| EditMenuRole | The standard Mac edit menu. Can be created using `menu.EditMenu()` | diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx new file mode 100644 index 00000000000..46d1c071b60 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx @@ -0,0 +1,853 @@ +--- +sidebar_position: 3 +--- + +# Options + +## Application Options + +The `Options.App` struct contains the application configuration. It is passed to the `wails.Run()` method: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // User messages that can be customised + Messages *windows.Messages + // OnSuspend is called when Windows enters low power mode + OnSuspend func() + // OnResume is called when Windows resumes from low power mode + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### Title + +The text shown in the window's title bar. + +Name: Title
Type: `string` + +### Width + +The initial width of the window. + +Name: Width
Type: `int`
Default: 1024. + +### Height + +The initial height of the window. + +Name: Height
Type: `int`
Default: 768 + +### DisableResize + +By default, the main window is resizable. Setting this to `true` will keep it a fixed size. + +Name: DisableResize
Type: `bool` + +### Fullscreen + +Deprecated: Please use [WindowStartState](#windowstartstate). + +### WindowStartState + +Defines how the window should present itself at startup. + +| Value | Win | Mac | Lin | +| ---------- | --- | --- | --- | +| Fullscreen | ✅ | ✅ | ✅ | +| Maximised | ✅ | ✅ | ✅ | +| Minimised | ✅ | ❌ | ✅ | + +Name: WindowStartState
Type: `options.WindowStartState` + +### Frameless + +When set to `true`, the window will have no borders or title bar. Also see [Frameless Windows](../guides/frameless.mdx). + +Name: Frameless
Type: `bool` + +### MinWidth + +This sets the minimum width for the window. If the value given in `Width` is less than this value, the window will be set to `MinWidth` by default. + +Name: MinWidth
Type: `int` + +### MinHeight + +This sets the minimum height for the window. If the value given in `Height` is less than this value, the window will be set to `MinHeight` by default. + +Name: MinHeight
Type: `int` + +### MaxWidth + +This sets the maximum width for the window. If the value given in `Width` is more than this value, the window will be set to `MaxWidth` by default. + +Name: MaxWidth
Type: `int` + +### MaxHeight + +This sets the maximum height for the window. If the value given in `Height` is more than this value, the window will be set to `MaxHeight` by default. + +Name: MaxHeight
Type: `int` + +### StartHidden + +When set to `true`, the application will be hidden until [WindowShow](../reference/runtime/window.mdx#windowshow) is called. + +Name: StartHidden
Type: `bool` + +### HideWindowOnClose + +By default, closing the window will close the application. Setting this to `true` means closing the window will + +hide the window instead. + +Name: HideWindowOnClose
Type: `bool` + +### BackgroundColour + +This value is the default background colour of the window. Example: options.NewRGBA(255,0,0,128) - Red at 50% transparency + +Name: BackgroundColour
Type: `*options.RGBA`
Default: white + +### AlwaysOnTop + +Indicates that the window should stay above other windows when losing focus. + +Name: AlwaysOnTop
Type: `bool` + +### Assets + +Deprecated: Please use Assets on [AssetServer specific options](#assetserver). + +### AssetsHandler + +Deprecated: Please use AssetsHandler on [AssetServer specific options](#assetserver). + +### AssetServer + +This defines AssetServer specific options. It allows to customize the AssetServer with static assets, serving assets dynamically with an `http.Handler` or hook into the request chain with an `assetserver.Middleware`. + +Not all features of an `http.Request` are currently supported, please see the following feature matrix: + +| Feature | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +Name: AssetServer
Type: `*assetserver.Options` + +#### Assets + +The static frontend assets to be used by the application. + +A GET request is first tried to be served from this `fs.FS`. If the `fs.FS` returns `os.ErrNotExist` for that file, the request handling will fallback to the [Handler](#handler) and tries to serve the GET request from it. + +If set to nil, all GET requests will be forwarded to [Handler](#handler). + +Name: Assets
Type: `fs.FS` + +#### Handler + +The assets handler is a generic `http.Handler` for fallback handling of assets that can't be found. + +The handler will be called for every GET request that can't be served from [Assets](#assets), due to `os.ErrNotExist`. Furthermore all non GET requests will always be served from this Handler. If not defined, the result is the following in cases where the Handler would have been called: + +- GET request: `http.StatusNotFound` +- Other request: `http.StatusMethodNotAllowed` + +NOTE: When used in combination with a Frontend DevServer there might be limitations, eg. Vite serves the index.html on every path, that does not contain a file extension. + +Name: AssetsHandler
Type: `http.Handler` + +#### Middleware + +Middleware is a HTTP Middleware which allows to hook into the AssetServer request chain. It allows to skip the default request handler dynamically, e.g. implement specialized Routing etc. The Middleware is called to build a new `http.Handler` used by the AssetSever and it also receives the default handler used by the AssetServer as an argument. + +If not defined, the default AssetServer request chain is executed. + +Name: Middleware
Type: `assetserver.Middleware` + +### Menu + +The menu to be used by the application. More details about Menus in the [Menu Reference](../reference/runtime/menu.mdx). + +:::note + +On Mac, if no menu is specified, a default menu will be created. + +::: + +Name: Menu
Type: `*menu.Menu` + +### Logger + +The logger to be used by the application. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: Logger
Type: `logger.Logger`
Default: Logs to Stdout + +### LogLevel + +The default log level. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevel
Type: `logger.LogLevel`
Default: `Info` in dev mode, `Error` in production mode + +### LogLevelProduction + +The default log level for production builds. More details about logging in the [Log Reference](../reference/runtime/log.mdx). + +Name: LogLevelProduction
Type: `logger.LogLevel`
Default: `Error` + +### OnStartup + +This callback is called after the frontend has been created, but before `index.html` has been loaded. It is given the application context. + +Name: OnStartup
Type: `func(ctx context.Context)` + +### OnDomReady + +This callback is called after the frontend has loaded `index.html` and its resources. It is given the application context. + +Name: OnDomReady
Type: `func(ctx context.Context)` + +### OnShutdown + +This callback is called after the frontend has been destroyed, just before the application terminates. It is given the application context. + +Name: OnShutdown
Type: `func(ctx context.Context)` + +### OnBeforeClose + +If this callback is set, it will be called when the application is about to quit, either by clicking the window close button or calling `runtime.Quit`. Returning true will cause the application to continue, false will continue shutdown as normal. This is good for confirming with the user that they wish to exit the program. + +Example: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +Name: OnBeforeClose
Type: `func(ctx context.Context) bool` + +### CSSDragProperty + +Indicates the CSS property to use to identify which elements can be used to drag the window. Default: `--wails-draggable`. + +Name: CSSDragProperty
Type: `string` + +### CSSDragValue + +Indicates what value the `CSSDragProperty` style should have to drag the window. Default: `drag`. + +Name: CSSDragValue
Type: `string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenu enables the browser's default context-menu in production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : + +| CSS Style | Behavior | +| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `--default-contextmenu: auto;` | (**default**) will show the default context menu only if :
contentEditable is true OR text has been selected OR element is input or textarea | +| `--default-contextmenu: show;` | will always show the default context menu | +| `--default-contextmenu: hide;` | will always hide the default context menu | + +This rule is inherited like any normal CSS rule, so nesting works as expected. + +:::note +This filtering functionality is only enabled in production, so in development and in debug build, the full context-menu is always available everywhere. +::: + +:::warning +This filtering functionality is NOT a security measure, the developer should expect that the full context-menu could be leaked anytime which could contain commands like (Download image, Reload, Save webpage), if this is a concern, the developer SHOULD NOT enable the default context-menu. +::: + + +Name: EnableDefaultContextMenu
Type: `bool` + +### EnableFraudulentWebsiteDetection + +EnableFraudulentWebsiteDetection enables scan services for fraudulent content, such as malware or phishing attempts. These services might send information from your app like URLs navigated to and possibly other content to cloud services of Apple and Microsoft. + +Name: EnableFraudulentWebsiteDetection
Type: `bool` + +### ZoomFactor + +Name: ZoomFactor
Type: `float64` + +This defines the zoom factor for the WebView2. This is the option matching the Edge user activated zoom in or out. + +### IsZoomControlEnabled + +Name: IsZoomControlEnabled
Type: `bool` + +This enables the zoom factor to be changed by the user. Please note that the zoom factor can be set in the options while disallowing the user to change it at runtime (f.e. for a kiosk application or similar). + +### Bind + +A slice of struct instances defining methods that need to be bound to the frontend. + +Name: Bind
Type: `[]interface{}` + +### ErrorFormatter + +A function that determines how errors are formatted when returned by a JS-to-Go method call. The returned value will be marshalled as JSON. + +Name: ErrorFormatter
Type: `func (error) any` + +### Windows + +This defines [Windows specific options](#windows). + +Name: Windows
Type: `*windows.Options` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent). + +For Windows 11 versions before build 22621, this will use the [BlurBehind](https://learn.microsoft.com/en-us/windows/win32/dwm/blur-ovw) method for translucency, which can be slow. For Windows 11 versions after build 22621, this will enable the newer translucency types that are much faster. By default, the type of translucency used will be determined by Windows. To configure this, use the [BackdropType](#BackdropType) option. + +Name: WindowIsTranslucent
Type: `bool` + +#### BackdropType + +:::note + +Requires Windows 11 build 22621 or later. + +::: + +Sets the translucency type of the window. This is only applicable if [WindowIsTranslucent](#WindowIsTranslucent) is set to `true`. + +Name: BackdropType
Type `windows.BackdropType` + +The value can be one of the following: + +| Value | Description | +| ------- | ----------------------------------------------------------------------------------------- | +| Auto | Let Windows decide which backdrop to use | +| None | Do not use translucency | +| Acrylic | Use [Acrylic](https://learn.microsoft.com/en-us/windows/apps/design/style/acrylic) effect | +| Mica | Use [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) effect | +| Tabbed | Use Tabbed. This is a backdrop that is similar to Mica. | + +#### DisableWindowIcon + +Setting this to `true` will remove the icon in the top left corner of the title bar. + +Name: DisableWindowIcon
Type: `bool` + +#### DisableFramelessWindowDecorations + +Setting this to `true` will remove the window decorations in [Frameless](#Frameless) mode. This means there will be no 'Aero Shadow' and no 'Rounded Corners' shown for the window. Please note that 'Rounded Corners' are only supported on Windows 11. + +Name: DisableFramelessWindowDecorations
Type: `bool` + +#### WebviewUserDataPath + +This defines the path where the WebView2 stores the user data. If empty `%APPDATA%\[BinaryName.exe]` will be used. + +Name: WebviewUserDataPath
Type: `string` + +#### WebviewBrowserPath + +This defines the path to a directory with WebView2 executable files and libraries. If empty, webview2 installed in the system will be used. + +Important information about distribution of fixed version runtime: + +- [How to get and extract runtime](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [Known issues for fixed version](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [The path of fixed version of the WebView2 Runtime should not contain \Edge\Application\.](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +Name: WebviewBrowserPath
Type: `string` + +#### Theme + +Minimum Windows Version: Windows 10 2004/20H1 + +This defines the theme that the application should use: + +| Value | Description | +| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------- | +| SystemDefault | _Default_. The theme will be based on the system default. If the user changes their theme, the application will update to use the new setting | +| Dark | The application will use a dark theme exclusively | +| Light | The application will use a light theme exclusively | + +Name: Theme
Type: `windows.Theme` + +#### CustomTheme + +:::note + +Minimum Windows Version: Windows 10/11 2009/21H2 Build 22000 + +::: + +Allows you to specify custom colours for TitleBar, TitleText and Border for both light and dark mode, as well as when the window is active or inactive. + +Name: CustomTheme
Type: `windows.CustomTheme` + +##### CustomTheme type + +The CustomTheme struct uses `int32` to specify the colour values. These are in the standard(!) Windows format of: `0x00BBGGAA`. A helper function is provided to do RGB conversions into this format: `windows.RGB(r,g,b uint8)`. + +NOTE: Any value not provided will default to black. + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +Example: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### Messages + +A struct of strings used by the webview2 installer if a valid webview2 runtime is not found. + +Name: Messages
Type: `*windows.Messages` + +Customise this for any language you choose to support. + +#### ResizeDebounceMS + +ResizeDebounceMS is the amount of time to debounce redraws of webview2 when resizing the window. The default value (0) will perform redraws as fast as it can. + +Name: ResizeDebounceMS
Type: `uint16` + +#### OnSuspend + +If set, this function will be called when Windows initiates a switch to low power mode (suspend/hibernate) + +Name: OnSuspend
Type: `func()` + +#### OnResume + +If set, this function will be called when Windows resumes from low power mode (suspend/hibernate) + +Name: OnResume
Type: `func()` + +#### WebviewGpuIsDisabled + +Setting this to `true` will disable GPU hardware acceleration for the webview. + +Name: WebviewGpuIsDisabled
Type: `bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +This defines [Mac specific options](#mac). + +Name: Mac
Type: `*mac.Options` + +#### TitleBar + +The TitleBar struct provides the ability to configure the look and feel of the title bar. + +Name: TitleBar
Type: [`*mac.TitleBar`](#titlebar-struct) + +##### Titlebar struct + +The titlebar of the application can be customised by using the TitleBar options: + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| Name | Description | +| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| TitlebarAppearsTransparent | Makes the titlebar transparent. This has the effect of hiding the titlebar and the content fill the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | Hides the title of the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | Removes [NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/) from the style mask | +| FullSizeContent | Makes the webview fill the entire window. [Apple Docs](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | Adds a default toolbar to the window. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | Removes the line beneath the toolbar. [Apple Docs](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +Preconfigured titlebar settings are available: + +| Setting | Example | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +Example: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +Click [here](https://github.com/lukakerr/NSWindowStyles) for some inspiration on customising the titlebar. + +#### Appearance + +Appearance is used to set the style of your app in accordance with Apple's [NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc) names. + +Name: Appearance
Type: [`mac.AppearanceType`](#appearance-type) + +##### Appearance type + +You can specify the application's [appearance](https://developer.apple.com/documentation/appkit/nsappearance?language=objc). + +| Value | Description | +| ----------------------------------------------------- | --------------------------------------------------------------- | +| DefaultAppearance | DefaultAppearance uses the default system value | +| NSAppearanceNameAqua | The standard light system appearance | +| NSAppearanceNameDarkAqua | The standard dark system appearance | +| NSAppearanceNameVibrantLight | The light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastAqua | A high-contrast version of the standard light system appearance | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | A high-contrast version of the standard dark system appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | A high-contrast version of the light vibrant appearance | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | A high-contrast version of the dark vibrant appearance | + +Example: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### WebviewIsTransparent + +Setting this to `true` will make the webview background transparent when an alpha value of `0` is used. This means that if you use `rgba(0,0,0,0)` for `background-color` in your CSS, the host window will show through. Often combined with [WindowIsTranslucent](#WindowIsTranslucent) to make frosty-looking applications. + +Name: WebviewIsTransparent
Type: `bool` + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Often combined with [WebviewIsTransparent](#WebviewIsTransparent) to make frosty-looking applications. + +Name: WindowIsTranslucent
Type: `bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### About + +This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. + +Name: About
Type: [`*mac.AboutInfo`](#about-struct) + +##### About struct + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +If these settings are provided, an "About" menu item will appear in the app menu (when using the `AppMenu` role). Given this configuration: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +The "About" menu item will appear in the app menu: + +```mdx-code-block +
+ +
+
+``` + +When clicked, that will open an about message box: + +```mdx-code-block +
+ +
+
+``` + +### Linux + +This defines [Linux specific options](#linux). + +Name: Linux
Type: `*linux.Options` + +#### Icon + +Sets up the icon representing the window. This icon is used when the window is minimized (also known as iconified). + +Name: Icon
Type: `[]byte` + +Some window managers or desktop environments may also place it in the window frame, or display it in other contexts. On others, the icon is not used at all, so your mileage may vary. + +NOTE: Gnome on Wayland at least does not display this icon. To have a application icon there, a `.desktop` file has to be used. On KDE it should work. + +The icon should be provided in whatever size it was naturally drawn; that is, don’t scale the image before passing it. Scaling is postponed until the last minute, when the desired final size is known, to allow best quality. + +#### WindowIsTranslucent + +Setting this to `true` will make the window background translucent. Some window managers may ignore it, or result in a black window. + +Name: WindowIsTranslucent
Type: `bool` + +#### WebviewGpuPolicy + +This option is used for determining the webview's hardware acceleration policy. + +Name: WebviewGpuPolicy
Type: [`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
Default: `WebviewGpuPolicyAlways` + +##### WebviewGpuPolicy type + +| Value | Description | +| ------------------------ | -------------------------------------------------------------------- | +| WebviewGpuPolicyAlways | Hardware acceleration is always enabled | +| WebviewGpuPolicyOnDemand | Hardware acceleration is enabled/disabled as request by web contents | +| WebviewGpuPolicyNever | Hardware acceleration is always disabled | + +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` property differs form the executable's filename. + +Name: ProgramName
Type: string
+ +### Debug + +This defines [Debug specific options](#Debug) that apply to debug builds. + +Name: Debug
Type: `options.Debug` + +#### OpenInspectorOnStartup + +Setting this to `true` will open the WebInspector on startup of the application. + +Name: OpenInspectorOnStartup
Type: `bool` + +[^1]: This requires WebKit2GTK 2.36+ support and your app needs to be build with the build tag `webkit2_36` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.36 for your app. +[^2]: This requires WebKit2GTK 2.40+ support and your app needs to be build with the build tag `webkit2_40` to activate support for this feature. This also bumps the minimum requirement of WebKit2GTK to 2.40 for your app. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx new file mode 100644 index 00000000000..8e763502bcc --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx @@ -0,0 +1,92 @@ +--- +sidebar_position: 5 +--- + +# Project Config + +The project config resides in the `wails.json` file in the project directory. The structure of the config is: + +```json5 +{ + // Project config version + "version": "", + // The project name + "name": "", + // Relative path to the directory containing the compiled assets, this is normally inferred and could be left empty + "assetdir": "", + // Additional directories to trigger reloads (comma separated), this is only used for some advanced asset configurations + "reloaddirs": "", + // The directory where the build files reside. Defaults to 'build' + "build:dir": "", + // Relative path to the frontend directory. Defaults to 'frontend' + "frontend:dir": "", + // The command to install node dependencies, run in the frontend directory - often `npm install` + "frontend:install": "", + // The command to build the assets, run in the frontend directory - often `npm run build` + "frontend:build": "", + // This command has been replaced by frontend:dev:build. If frontend:dev:build is not specified will falls back to this command. \nIf this command is also not specified will falls back to frontend:build + "frontend:dev": "", + // This command is the dev equivalent of frontend:build. If not specified falls back to frontend:dev + "frontend:dev:build": "", + // This command is the dev equivalent of frontend:install. If not specified falls back to frontend:install + "frontend:dev:install": "", + // This command is run in a separate process on `wails dev`. Useful for 3rd party watchers or starting 3d party dev servers + "frontend:dev:watcher": "", + // URL to a 3rd party dev server to be used to serve assets, EG Vite. \nIf this is set to 'auto' then the devServerUrl will be inferred from the Vite output + "frontend:dev:serverUrl": "", + // Relative path to the directory that the auto-generated JS modules will be created + "wailsjsdir": "", + // The name of the binary + "outputfilename": "", + // The default time the dev server waits to reload when it detects a change in assets + "debounceMS": 100, + // Address to bind the wails dev sever to. Default: localhost:34115 + "devServer": "", + // Arguments passed to the application in shell style when in dev mode + "appargs": "", + // Defines if build hooks should be run though they are defined for an OS other than the host OS. + "runNonNativeBuildHooks": false, + "preBuildHooks": { + // The command that will be executed before a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed before a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed before every build: ${platform} is replaced with the "GOOS/GOARCH". + "*/*": "" + }, + "postBuildHooks": { + // The command that will be executed after a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed after a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed after every build: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. + "*/*": "" + }, + // Data used to populate manifests and version info. + "info": { + // The company name. Default: [The project name] + "companyName": "", + // The product name. Default: [The project name] + "productName": "", + // The version of the product. Default: '1.0.0' + "productVersion": "", + // The copyright of the product. Default: 'Copyright.........' + "copyright": "", + // A short comment of the app. Default: 'Built using Wails (https://wails.app)' + "comments": "" + }, + // 'multiple': One installer per architecture. 'single': Single universal installer for all architectures being built. Default: 'multiple' + "nsisType": "", + // Whether the app should be obfuscated. Default: false + "obfuscated": "", + // The arguments to pass to the garble command when using the obfuscated flag + "garbleargs": "" +} +``` + +This file is read by the Wails CLI when running `wails build` or `wails dev`. + +The `assetdir`, `reloaddirs`, `wailsjsdir`, `debounceMS`, `devserver` and `frontenddevserverurl` flags in `wails build/dev` will update the project config +and thus become defaults for subsequent runs. + +The JSON Schema for this file is located [here](https://wails.io/schemas/config.v2.json). diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx new file mode 100644 index 00000000000..f45c13c46a6 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# Браузер + +Эти методы относятся к системному браузеру. + +### BrowserOpenURL + +Открывает данный URL-адрес в системном браузере. + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..805f68ed917 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# Clipboard + +This part of the runtime provides access to the operating system's clipboard.
The current implementation only handles text. + +### ClipboardGetText + +This method reads the currently stored text from the clipboard. + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
Returns: a string (if the clipboard is empty an empty string will be returned) or an error. + +JS: `ClipboardGetText(): Promise`
Returns: a promise with a string result (if the clipboard is empty an empty string will be returned). + +### ClipboardSetText + +This method writes a text to the clipboard. + +Go: `ClipboardSetText(ctx context.Context, text string) error`
Returns: an error if there is any. + +JS: `ClipboardSetText(text: string): Promise`
Returns: a promise with true result if the text was successfully set on the clipboard, false otherwise. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..9b3e6cf145c --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# Dialog + +This part of the runtime provides access to native dialogs, such as File Selectors and Message boxes. + +:::info JavaScript + +Dialog is currently unsupported in the JS runtime. + +::: + +### OpenDirectoryDialog + +Opens a dialog that prompts the user to select a directory. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Returns: Selected directory (blank if the user cancelled) or an error + +### OpenFileDialog + +Opens a dialog that prompts the user to select a file. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +Returns: Selected file (blank if the user cancelled) or an error + +### OpenMultipleFilesDialog + +Opens a dialog that prompts the user to select multiple files. Can be customised using [OpenDialogOptions](#opendialogoptions). + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +Returns: Selected files (nil if the user cancelled) or an error + +### SaveFileDialog + +Opens a dialog that prompts the user to select a filename for the purposes of saving. Can be customised using [SaveDialogOptions](#savedialogoptions). + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +Returns: The selected file (blank if the user cancelled) or an error + +### MessageDialog + +Displays a message using a message dialog. Can be customised using [MessageDialogOptions](#messagedialogoptions). + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +Returns: The text of the selected button or an error + +## Options + +### OpenDialogOptions + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| Field | Description | Win | Mac | Lin | +| -------------------------- | ---------------------------------------------- | --- | --- | --- | +| DefaultDirectory | The directory the dialog will show when opened | ✅ | ✅ | ✅ | +| DefaultFilename | The default filename | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | A list of file filters | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Show files hidden by the system | | ✅ | ✅ | +| CanCreateDirectories | Allow user to create directories | | ✅ | | +| ResolvesAliases | If true, returns the file not the alias | | ✅ | | +| TreatPackagesAsDirectories | Allow navigating into packages | | ✅ | | + +### SaveDialogOptions + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| Field | Description | Win | Mac | Lin | +| -------------------------- | ---------------------------------------------- | --- | --- | --- | +| DefaultDirectory | The directory the dialog will show when opened | ✅ | ✅ | ✅ | +| DefaultFilename | The default filename | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | A list of file filters | ✅ | ✅ | ✅ | +| ShowHiddenFiles | Show files hidden by the system | | ✅ | ✅ | +| CanCreateDirectories | Allow user to create directories | | ✅ | | +| TreatPackagesAsDirectories | Allow navigating into packages | | ✅ | | + +### MessageDialogOptions + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| Field | Description | Win | Mac | Lin | +| ------------- | -------------------------------------------------------------------------- | -------------- | --- | --- | +| Type | The type of message dialog, eg question, info... | ✅ | ✅ | ✅ | +| Title | Title for the dialog | ✅ | ✅ | ✅ | +| Message | The message to show the user | ✅ | ✅ | ✅ | +| Buttons | A list of button titles | | ✅ | | +| DefaultButton | The button with this text should be treated as default. Bound to `return`. | ✅[*](#windows) | ✅ | | +| CancelButton | The button with this text should be treated as cancel. Bound to `escape` | | ✅ | | + +#### Windows + +Windows has standard dialog types in which the buttons are not customisable. The value returned will be one of: "Ok", "Cancel", "Abort", "Retry", "Ignore", "Yes", "No", "Try Again" or "Continue". + +For Question dialogs, the default button is "Yes" and the cancel button is "No". This can be changed by setting the `DefaultButton` value to `"No"`. + +Example: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linux has standard dialog types in which the buttons are not customisable. The value returned will be one of: "Ok", "Cancel", "Yes", "No" + +#### Mac + +A message dialog on Mac may specify up to 4 buttons. If no `DefaultButton` or `CancelButton` is given, the first button is considered default and is bound to the `return` key. + +For the following code: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, +}) +``` + +the first button is shown as default: + +```mdx-code-block +
+ +
+
+``` + +And if we specify `DefaultButton` to be "two": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +the second button is shown as default. When `return` is pressed, the value "two" is returned. + +```mdx-code-block +
+ +
+
+``` + +If we now specify `CancelButton` to be "three": + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +the button with "three" is shown at the bottom of the dialog. When `escape` is pressed, the value "three" is returned: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### DialogType + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### FileFilter + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +Windows allows you to use multiple file filters in dialog boxes. Each FileFilter will show up as a separate entry in the dialog: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linux allows you to use multiple file filters in dialog boxes. Each FileFilter will show up as a separate entry in the dialog: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +Mac dialogs only have the concept of a single set of patterns to filter files. If multiple FileFilters are provided, Wails will use all the Patterns defined. + +Example: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +This will result in the Open File dialog using `*.png,*.jpg,*.mov,*.mp4` as a filter. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx new file mode 100644 index 00000000000..856ba6f0c26 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# Events + +The Wails runtime provides a unified events system, where events can be emitted or received by either Go or JavaScript. Optionally, data may be passed with the events. Listeners will receive the data in the local data types. + +### EventsOn + +This method sets up a listener for the given event name. When an event of type `eventName` is [emitted](#EventsEmit), the callback is triggered. Any additional data sent with the emitted event will be passed to the callback. It returns a function to cancel the listener. + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff + +This method unregisters the listener for the given event name, optionally multiple listeneres can be unregistered via `additionalEventNames`. + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce + +This method sets up a listener for the given event name, but will only trigger once. It returns a function to cancel the listener. + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple + +This method sets up a listener for the given event name, but will only trigger a maximum of `counter` times. It returns a function to cancel the listener. + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit + +This method emits the given event. Optional data may be passed with the event. This will trigger any event listeners. + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx new file mode 100644 index 00000000000..aae8efccccf --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# Introduction + +The runtime is a library that provides utility methods for your application. There is both a Go and JavaScript runtime and the aim is to try and keep them at parity where possible. + +It has utility methods for: + +- [Window](window.mdx) +- [Menu](menu.mdx) +- [Dialog](dialog.mdx) +- [Events](events.mdx) +- [Browser](browser.mdx) +- [Log](log.mdx) +- [Clipboard](clipboard.mdx) + +The Go Runtime is available through importing `github.com/wailsapp/wails/v2/pkg/runtime`. All methods in this package take a context as the first parameter. This context should be obtained from the [OnStartup](../options.mdx#onstartup) or [OnDomReady](../options.mdx#ondomready) hooks. + +:::info Note + +Whilst the context will be provided to the [OnStartup](../options.mdx#onstartup) method, there's no guarantee the runtime will work in this method as the window is initialising in a different thread. If you wish to call runtime methods at startup, use [OnDomReady](../options.mdx#ondomready). + +::: + +The JavaScript library is available to the frontend via the `window.runtime` map. There is a runtime package generated when using `dev` mode that provides TypeScript declarations for the runtime. This should be located in the `wailsjs` directory in your frontend directory. + +### Hide + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +Hides the application. + +:::info Note + +On Mac, this will hide the application in the same way as the `Hide` menu item in standard Mac applications. This is different to hiding the window, but the application still being in the foreground. For Windows and Linux, this is currently the same as `WindowHide`. + +::: + +### Show + +Shows the application. + +:::info Note + +On Mac, this will bring the application back into the foreground. For Windows and Linux, this is currently the same as `WindowShow`. + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### Quit + +Quits the application. + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### Environment + +Returns details of the current environment. + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### EnvironmentInfo + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx new file mode 100644 index 00000000000..06f0423b0a8 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# Log + +The Wails runtime provides a logging mechanism that may be called from Go or JavaScript. Like most loggers, there are a number of log levels: + +- Trace +- Debug +- Info +- Warning +- Error +- Fatal + +The logger will output any log message at the current, or higher, log level. Example: The `Debug` log level will output all messages except `Trace` messages. + +### LogPrint + +Logs the given message as a raw message. + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf + +Logs the given message as a raw message. + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace + +Logs the given message at the `Trace` log level. + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef + +Logs the given message at the `Trace` log level. + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug + +Logs the given message at the `Debug` log level. + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf + +Logs the given message at the `Debug` log level. + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo + +Logs the given message at the `Info` log level. + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof + +Logs the given message at the `Info` log level. + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning + +Logs the given message at the `Warning` log level. + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf + +Logs the given message at the `Warning` log level. + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError + +Logs the given message at the `Error` log level. + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf + +Logs the given message at the `Error` log level. + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal + +Logs the given message at the `Fatal` log level. + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf + +Logs the given message at the `Fatal` log level. + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel + +Sets the log level. In JavaScript, the number relates to the following log levels: + +| Value | Log Level | +| ----- | --------- | +| 1 | Trace | +| 2 | Debug | +| 3 | Info | +| 4 | Warning | +| 5 | Error | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## Using a Custom Logger + +A custom logger may be used by providing it using the [Logger](../options.mdx#logger) application option. The only requirement is that the logger implements the `logger.Logger` interface defined in `github.com/wailsapp/wails/v2/pkg/logger`: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx new file mode 100644 index 00000000000..6b6ceb77d69 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# Меню + +Эти методы относятся к меню приложения. + +:::info JavaScript + +В настоящее время меню не поддерживается в JS. + +::: + +### MenuSetApplicationMenu + +Задает меню приложения для данного [меню](../menus.mdx). + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu + +Обновляет меню приложения, поднимая все изменения в меню, переданное `MenuSetationMenu`. + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx new file mode 100644 index 00000000000..457c92ebff9 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# Screen + +These methods provide information about the currently connected screens. + +### ScreenGetAll + +Returns a list of currently connected screens. + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### Screen + +Go struct: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript interface: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx new file mode 100644 index 00000000000..15f555c5a79 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# Window + +These methods give control of the application window. + +### WindowSetTitle + +Sets the text in the window title bar. + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen + +Makes the window full screen. + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen + +Restores the previous window dimensions and position prior to full screen. + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen + +Returns true if the window is full screen. + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter + +Centers the window on the monitor the window is currently on. + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS + +Executes arbitrary JS code in the window. + +This method runs the code in the browser asynchronously and returns immediately. If the script causes any errors, they will only be available in the browser console. + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload + +Performs a "reload" (Reloads current page). + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp + +Reloads the application frontend. + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme + +Windows only. + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +Sets window theme to system default (dark/light). + +### WindowSetLightTheme + +Windows only. + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +Sets window theme to light. + +### WindowSetDarkTheme + +Windows only. + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +Sets window theme to dark. + +### WindowShow + +Shows the window, if it is currently hidden. + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowHide + +Hides the window, if it is currently visible. + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal + +Returns true if the window not minimised, maximised or fullscreen. + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize + +Sets the width and height of the window. + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowGetSize + +Gets the width and height of the window. + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize + +Sets the minimum window size. Will resize the window if the window is currently smaller than the given dimensions. + +Setting a size of `0,0` will disable this constraint. + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize + +Sets the maximum window size. Will resize the window if the window is currently larger than the given dimensions. + +Setting a size of `0,0` will disable this constraint. + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop + +Sets the window AlwaysOnTop or not on top. + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition + +Sets the window position relative to the monitor the window is currently on. + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition + +Gets the window position relative to the monitor the window is currently on. + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximise + +Maximises the window to fill the screen. + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise + +Restores the window to the dimensions and position prior to maximising. + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised + +Returns true if the window is maximised. + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise + +Toggles between Maximised and UnMaximised. + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise + +Minimises the window. + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise + +Restores the window to the dimensions and position prior to minimising. + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised + +Returns true if the window is minimised. + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour + +Sets the background colour of the window to the given RGBA colour definition. This colour will show through for all transparent pixels. + +Valid values for R, G, B and A are 0-255. + +:::info Windows + +On Windows, only alpha values of 0 or 255 are supported. Any value that is not 0 will be considered 255. + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## TypeScript Object Definitions + +### Position + +```ts +interface Position { + x: number; + y: number; +} +``` + +### Size + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..1af16f7745a --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# Dogs API + +```mdx-code-block +
+ +
+
+``` + +:::note + +This tutorial has been kindly provided by [@tatadan](https://twitter.com/tatadan) and forms part of their [Wails Examples Repository](https://github.com/tataDan/wails-v2-examples). + +::: + +In this tutorial we are going to develop an application that retrieves photos of dogs from the web and then displays them. + +### Create the project + +Let's create the application. From a terminal enter: `wails init -n dogs-api -t svelte` + +Note: We could optionally add `-ide vscode` or `-ide goland` to the end of this command if you wanted to add IDE support. + +Now let's `cd dogs-api` and start editing the project files. + +### Remove unused code + +We will start by removing some elements that we know we will not use: + +- Open `app.go` and remove the following lines: + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- Open `frontend/src/App.svelte` and delete all lines. +- Delete the `frontend/src/assets/images/logo-universal.png` file + +### Creating our application + +Now let's add our new Go code. + +Add the following struct declarations to `app.go` before the function definitions: + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +Add the following functions to `app.go` (perhaps after the existing function definitions): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +Modify the `import` section of `app.go` to look like this: + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +Add the following lines to `frontend/src/App.svelte`: + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### Testing the application + +To generate the bindings and test the application, run `wails dev`. + +### Compiling the application + +To compile the application to a single, production grade binary, run `wails build`. diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx new file mode 100644 index 00000000000..77d128b188c --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# Hello World + +Цель этого урока — запустить наиболее базовое приложение, использующее Wails. Вы сможете: + +- Создавать новое Wails приложение +- Собирать приложение +- Запускать приложение + +:::note + +В этом уроке в качестве целевой платформы используется Windows. Вывод будет варьироваться в зависимости от вашей операционной системы. + +::: + +## Создаем новое Wails приложение + +Чтобы создать новое Wails приложение, использующее стандартный шаблон JS, вам нужно выполнить следующую команду: + +```bash +wails init -n helloworld +``` + +Вы должны увидеть что-то похожее на следующее: + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +Это создаст новый каталог под названием `helloworld` в текущей директории. В этом каталоге вы найдете несколько файлов: + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## Сборка приложения + +Чтобы собрать приложение, перейдите в новую директорию `helloworld` и запустите следующую команду: + +```bash +wails build +``` + +Вы должны увидеть что-то похожее на следующее: + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +Приложение собрано и сохранено в папке `build/bin`. + +## Запуск приложения + +Если мы откроем папку `build/bin` в Проводнике, то увидим исполняемый файл проекта: + +```mdx-code-block +
+ +
+
+``` + +Мы можем запустить его, просто дважды щелкнув по файлу `helloworld.exe`. + +На Mac, Wails генерирует файл `helloworld.app` который может быть запущен двойным щелчком. + +На Linux вы можете запустить приложение с помощью файла `./helloworld` из папки `build/bin`. + +Вы должны видеть приложение, работающее так, как ожидалось: + +```mdx-code-block +
+ +
+
+``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/appendix/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/appendix/_category_.json new file mode 100644 index 00000000000..1374f0d5518 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/appendix/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Runtime", + "position": 70 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/_category_.json new file mode 100644 index 00000000000..9827bf0c0fb --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "社区", + "position": 50 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx new file mode 100644 index 00000000000..8f8a54fcb3d --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# 链接 + +此页面用于列出社区相关的链接。 请提交 PR(点击页面底部的 `编辑此页`)增加链接。 + +## Awesome Wails + +Wails 相关的 [优秀列表](https://github.com/wailsapp/awesome-wails)。 + +## 支持的频道 + +- [Wails Discord 服务器](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [v2 测试版讨论板](https://github.com/wailsapp/wails/discussions/828) + +## 社交媒体 + +- [Twitter](https://twitter.com/wailsapp) +- [Wails 中文社区 QQ 群](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - 群号:1067173054 + +## 其他教程和文章 + +- [Building of Bulletin Board](https://blog.customct.com/building-bulletin-board) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/_category_.json new file mode 100644 index 00000000000..276e283b71f --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Showcase", + "position": 1 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..2bba85cb82d --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +

+ +
+

+``` + +[BulletinBoard](https://github.com/raguay/BulletinBoard) 应用程序是一个用于静态消息或对话框的通用消息板,用于获取用户的脚本信息。 它有一个用于创建新对话框的 TUI,后者可以使用这些对话框从用户那里获取信息。 它的设计是在您的系统上保持运行并根据需要显示信息,然后隐藏起来。 我有一个在我的系统上监视文件并在更改时将内容发送到 BulletinBoard 的进程。 它与我的工作流配合的很好。 还有一个用于向程序发送信息的 [Alfred 工作流](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow)。 该工作流也适用于 [EmailIt](https://github.com/raguay/EmailIt)。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx new file mode 100644 index 00000000000..701e96b5a95 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/) 是一个 Wails v2 程序,它是一个基于 Markdown 的电子邮件发送器,只有九个记事本、用于处理文本的脚本和模板。 它还有一个脚本终端,用于在 EmailIt 中对系统中的文件运行脚本。 脚本和模板可以从命令行本身使用,也可以与 Alfred、Keyboard Maestro、Dropzone 或 PopClip 扩展一起使用。 它还支持从 GitHub 下载的脚本和主题。 文档不完整,但程序有效。 它是使用 Wails v2 和 Svelte 构建的,下载的是一个通用的 macOS 应用程序。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx new file mode 100644 index 00000000000..df788c69cbd --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx @@ -0,0 +1,12 @@ +# EncryptEasy + +```mdx-code-block +

+ +
+

+``` + +**[EncryptEasy](https://www.encrypteasy.app) 是一个简单易用的 PGP 加密工具,管理您和您的所有联系人密钥。 加密应该很简单。 使用 Wails 开发。** + +使用 PGP 加密消息是行业标准。 每个人都有私钥和公钥。 您的私钥需要保密,因此只有您可以阅读消息。 您的公钥会分发给任何想向您发送秘密加密消息的人。 管理密钥、加密消息和解密消息应该是一种流畅的体验。 EncryptEasy 就是要让它变得简单。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx new file mode 100644 index 00000000000..a0566cccf7a --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx @@ -0,0 +1,16 @@ +# FileHound Export Utility + +```mdx-code-block +

+ +
+

+``` + +[FileHound Export Utility](https://www.filehound.co.uk/) FileHound 是一个云文档管理平台,用于安全文件保留、业务流程自动化和 SmartCapture 功能。 + +FileHound Export Utility 允许 FileHound 管理员运行安全文档和数据提取任务以备选备份和恢复目的。 此应用程序将根据您选择的过滤器下载保存在 FileHound 中的所有文档和/或元数据。 元数据将以 JSON 和 XML 格式导出。 + +后端构建使用:Go 1.15 Wails 1.11.0 go-sqlite3 1.14.6 go-linq 3.2 + +前端使用:Vue 2.6.11 Vuex 3.4.0 TypeScript Tailwind 1.9.6 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx new file mode 100644 index 00000000000..61153563564 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx @@ -0,0 +1,10 @@ +# hiposter + +```mdx-code-block +

+ +
+

+``` + +[hiposter](https://github.com/obity/hiposter) 是一个简单高效的 HTTP API 测试客户端工具。 基于 Wails、Go 和 sveltejs。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx new file mode 100644 index 00000000000..f5552ed5ad7 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx @@ -0,0 +1,14 @@ +# Minecraft Updater + +```mdx-code-block +

+ +
+

+``` + +[Minecraft Updater](https://github.com/Gurkengewuerz/MinecraftModUpdater) 是一个可以为你的用户更新和同步 Minecraft 模组的实用工具。 它使用 Wails2 和 React 构建,采用 [antd](https://ant.design/) 作为前端框架。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx new file mode 100644 index 00000000000..634ef3329bb --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx @@ -0,0 +1,14 @@ +# Modal File Manager + +```mdx-code-block +

+ +
+

+``` + +[Modal File Manager](https://github.com/raguay/ModalFileManager) 是一个使用网络技术的双面板文件管理器。 我最初的设计是基于 NW.js 的,可以在 [这里](https://github.com/raguay/ModalFileManager-NWjs) 找到。 此版本使用相同的基于 Svelte 的前端代码(但自从 NW.js 脱离以来已进行了很大修改),但后端是 [Wails v2](https://wails.io/) 实现。 通过使用这个实现,我不再使用命令行 `rm`、`cp` 等命令,而是必须在系统上安装 git 才能下载主题和扩展。 它完全使用 Go 编码,运行速度比以前的版本快得多。 + +这个文件管理器是围绕与 Vim 相同的原则设计的:状态控制键盘操作。 状态的数量不是固定的,但非常可编程。 因此,可以创建和使用无数种键盘配置。 这是与其他文件管理器的主要区别。 可以从 GitHub 下载主题和扩展。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx new file mode 100644 index 00000000000..1df94bb1139 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx @@ -0,0 +1,10 @@ +# Molley Wallet + +```mdx-code-block +

+ +
+

+``` + +[Molly Wallet](https://github.com/grvlle/constellation_wallet/) 是 Constellation 网络的官方 $DAG 钱包。 它允许用户以各种方式与 Hypergraph 网络进行交互,而不仅仅是产生 $DAG 交易。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx new file mode 100644 index 00000000000..861e8d43c2b --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[October](https://october.utf9k.net) 是一个小型的 Wails 应用程序,它可以非常容易地从 [Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader) 中提取精彩片段,然后将它们转发到 [Readwise](https://readwise.io) 。 + +它的代码规模相对较小,所有平台版本的大小都不到10 MB ,而且这还是在没有启用 [UPX 压缩](https://upx.github.io/) 的情况下! + +相比之下,作者之前尝试使用 Electron 很快就膨胀到了几百兆字节。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx new file mode 100644 index 00000000000..baea5ba9218 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx @@ -0,0 +1,10 @@ +# Optimus + +```mdx-code-block +

+ +
+

+``` + +[Optimus](https://github.com/splode/optimus) 是一款桌面图像优化应用程序。 它支持 WebP、JPEG 和 PNG 图像格式之间的转换和压缩。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx new file mode 100644 index 00000000000..40180200bef --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx @@ -0,0 +1,10 @@ +# Portfall + +```mdx-code-block +

+ +
+

+``` + +[Portfall](https://github.com/rekon-oss/portfall) - 桌面 k8s 端口转发门户,可轻松访问你的所有集群 UI 。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..a7f07f4f5b6 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - 一个简单的、跨平台的 [restic](https://github.com/restic/restic) 备份 GUI ,用于浏览和恢复 restic 存储库。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..29514c8e5ba --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +为每个人提供简单、安全和免费的文件共享。 请访问 [Riftshare.app](https://riftshare.app) 了解更多信息。 + +## 功能 + +- 可以在本地网络和互联网之间轻松安全地共享文件 +- 支持通过 [魔法虫洞(magic-wormhole)](https://magic-wormhole.readthedocs.io/en/latest/) 安全地发送文件或目录 +- 与所有其他使用魔法虫洞(magic-wormhole)的应用程序兼容( magic-wormhole 或 wormhole-william CLI,wormhole-gui 等。) +- 自动压缩多个选定文件以便一次性发送 +- 发送和接收的完整动画、进度条和取消支持 +- 原生的操作系统文件选择功能 +- 接收文件后一键打开 +- 自动更新 - 不用担心有最新的版本! diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..85409d3c7bd --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) 是一个用于显示脚本或 [Node-Red](https://nodered.org) 服务器输出的程序。 它运行 EmailIt 程序中定义的脚本,并显示输出结果。 可以使用来自 xBar 或 TextBar 的脚本,但目前只有 TextBar 的脚本运行良好。 它还可以显示你系统中的脚本输出。 ScriptBar 并没有把它们放在菜单栏中,而是提供一个方便的窗口,方便用户查看。 你可以使用多个标签页来显示不同的内容。 还可以保留访问次数最多的网站链接。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx new file mode 100644 index 00000000000..67e0930beeb --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/) 是一个 P2P 文件共享应用程序,旨在利用区块链技术实现100%匿名文件传输。 Surge 是端到端加密的、去中心化的以及开源的。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx new file mode 100644 index 00000000000..349b8ce215e --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally) 是 [Ergodox](https://ergodox-ez.com/) 键盘的官方固件刷写程序。 它看起来很不错,并且是一个使用 Wails 实现的绝妙示例:结合了 Go 语言的强大功能和 Web 开发世界丰富的图形工具。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx new file mode 100644 index 00000000000..1172a8c3366 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx @@ -0,0 +1,19 @@ +# WarMine Minecraft 启动器 + +```mdx-code-block +

+ + +
+

+``` + +[Warmine Minecraft 启动器](https://warmine.ru/) 是一款 Wails 应用,它使您可以轻松地加入服务器并管理您的游戏账户。 + +启动器能够下载游戏文件, 检查游戏的完整性,使用自定义选项从后端启动游戏。 + +前端使用 Svelte 编写, 整个启动器大小约 9MB 并支持 Windows 7-11. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx new file mode 100644 index 00000000000..d8d74d360ad --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx @@ -0,0 +1,10 @@ +# Wombat + +```mdx-code-block +

+ +
+

+``` + +[Wombat](https://github.com/rogchap/wombat) 是一个跨平台的 gRPC 客户端。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx new file mode 100644 index 00000000000..12cefd304f5 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails) 是一款可以从 YouTube 下载音乐,创建离线播放列表并与朋友分享的应用程序。你的朋友们可以播放你的播放列表或将其下载到本地离线收听,并且该应用程序还内置了播放器。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx new file mode 100644 index 00000000000..fb487e87c79 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx @@ -0,0 +1,69 @@ +--- +sidebar_position: 1 +--- + +# 模板 + +此页面用作社区支持的模板列表。 请提交一个包含您的模板的 PR(点击页面底部的 `编辑此页`)。 要构建您自己的模板,请参考 [模板](../guides/templates) 指南。 + +要使用这些模板,请运行 `wails init -n "您的项目名" -t [下面的链接[@版本]]` + +如果不带版本后缀,默认使用的是主分支代码模板。 如果带有版本后缀,则使用该版本对应标签的代码模板。 + +示例:`wails init -n "Your Project Name" -t https://github.com/misitebao/wails-template-vue` + +:::warning 注意 + +**Wails 项目不维护也不对第 3 方模板负责** + +如果您不确定某个模板,请检查 `package.json` 和 `wails.json` 中安装的模块和运行的脚本。 + +::: + +## Vue + +- [wails-template-vue](https://github.com/misitebao/wails-template-vue) - 基于 Vue 生态的 Wails 模板(集成 TypeScript、黑暗主题、国际化、单页路由、TailwindCSS) +- [wails-vite-vue-ts](https://github.com/codydbentley/wails-vite-vue-ts) - 使用 Vite 的 Vue 3 TypeScript(以及添加功能的说明) +- [wails-vite-vue-the-works](https://github.com/codydbentley/wails-vite-vue-the-works) - 使用 Vite, Vuex, Vue Router, Sass, 和 ESLint + Prettier 的 Vue 3 TypeScript +- [wails-template-quasar-js](https://github.com/sgosiaco/wails-template-quasar-js) - 使用 JavaScript + Quasar V2(Vue 3, Vite, Sass, Pinia, ESLint, Prettier)的模板 +- [wails-template-quasar-ts](https://github.com/sgosiaco/wails-template-quasar-ts) - 使用 TypeScript + Quasar V2(Vue 3、Vite、Sass、Pinia、ESLint、Prettier、带 <script setup> 的Composition API)的模板 +- [wails-template-naive](https://github.com/tk103331/wails-template-naive) - 基于 Naive UI(一款 Vue 3 组件库)的 Wails 模板 + +## Angular + +- [wails-template-angular](https://github.com/mateothegreat/wails-template-angular) - Angular 15+ 已打包并准备投入生产。 +- [wails-angular-template](https://github.com/TAINCER/wails-angular-template) - 带有 TypeScript, Sass, 热重载, 代码拆分和 i18n 的 Angular + +## React + +- [wails-react-template](https://github.com/AlienRecall/wails-react-template) - 基于 reactjs 的模板 +- [wails-react-template](https://github.com/flin7/wails-react-template) - 基于 React 并支持实时开发模式的轻量级模板 +- [wails-vite-react-ts](https://github.com/lontten/wails-vite-react-ts) - 基于 Vite + React + TypeScript 的模板 +- [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - 一个 React + TypeScript + Vite + TailwindCSS 模板 +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui + +## Svelte + +- [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - 基于 Svelte 的模板 +- [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - 使用 Svelte 和 Vite 的模板 +- [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - 使用 Svelte 和 Vite 和 TailwindCSS v3 的模板 +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 +- [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - 基于 Next.js + TypeScript 的模板 + +## Solid + +- [wails-template-vite-solid-ts](https://github.com/xijaja/wails-template-solid-ts) - 使用 Solid + Ts + Vite 的模版。 +- [wails-template-vite-solid-js](https://github.com/xijaja/wails-template-solid-js) - 使用 Solid + Js + Vite 的模版。 + +## Elm + +- [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - 使用函数式编程和 **快速** 的热重载设置开发您的 GUI 应用程序 :tada: :rocket: +- [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - 结合 Elm + Tailwind CSS + Wails 的力量 :muscle: ! 支持热重载。 + +## HTMX + +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms + +## 纯 JavaScript (Vanilla) + +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - 一个只有基本 JavaScript、HTML 和 CSS 的模板 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/developing-new-features.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/developing-new-features.mdx new file mode 100644 index 00000000000..9fc9025bd4b --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/developing-new-features.mdx @@ -0,0 +1,34 @@ +--- +sidebar_position: 20 +--- + +# 开发新功能 + +We are always keen to add features to Wails and expand on what the project can do. The process for adding new features are as follows: The process for adding new features are as follows: We are always keen to add features to Wails and expand on what the project can do. The process for adding new features are as follows: The process for adding new features are as follows: The process for adding new features are as follows: + +- Pick an enhancement ticket with the "TODO" label. Pick an enhancement ticket with the "TODO" label. Pick an enhancement ticket with the "TODO" label. It's preferable to select one from the current [Backlog](https://github.com/orgs/wailsapp/projects/1/views/1) but the choice is yours. +- Before developing, check that the ticket includes the following information: +- The purpose of the enhancement +- What is out of scope for the enhancement +- What platforms the enhancement targets (most features should be cross-platform unless there's a very specific reason) +- If the ticket does not include this information, feel free to request the information from the person who opened the ticket. Sometimes placeholder tickets are created and require more details Sometimes placeholder tickets are created and require more details Sometimes placeholder tickets are created and require more details +- Comment on the ticket stating you wish to develop the feature +- Clone the repository and create a branch with the format `feature/_` +- New features often require documentation so please ensure you have also added or updated the documentation as part of the changes +- Once the feature is ready for testing, create a draft PR. Once the feature is ready for testing, create a draft PR. Once the feature is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. Once the feature is ready for testing, create a draft PR. Once the feature is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. Once the feature is ready for testing, create a draft PR. Once the feature is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. +- Once all the testing is completed, please update the status of the PR from draft and leave a message. + +:::note +There is nothing stopping you from opening a ticket and working on it yourself, but please be aware that all +enhancement requests are reviewed for good fit. Not all ideas will be selected so it's best to have discussion +on the ticket first. +::: Not all ideas will be selected so it's best to have discussion +on the ticket first. +::: Not all ideas will be selected so it's best to have discussion +on the ticket first. +::: + +:::warning +Any PRs opened without a corresponding ticket may be rejected. +::: ::: +::: diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/documenting.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/documenting.mdx new file mode 100644 index 00000000000..84e47290358 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/documenting.mdx @@ -0,0 +1,34 @@ +--- +sidebar_position: 40 +--- + +# 文档 + +This website is also the main documentation site for the project. Sometimes this gets out of date and needs some slight adjustments. Some of the documentation isn't written to the best standards either. Developing documentation is hard and so any contribution to this is greatly appreciated. Features without documentation are unfinished so to the project, it's _as important_ as the code. + +We generally do not create tickets for updating documentation so if there is text you think should be updated or rephrased then feel free to submit a PR for that. This site is in the main repository under the `website` directory. We use [Docusaurus](https://docusaurus.io/) to create the site so there is plenty of existing documentation and tutorials around to get started. This site is in the main repository under the `website` directory. We use [Docusaurus](https://docusaurus.io/) to create the site so there is plenty of existing documentation and tutorials around to get started. This site is in the main repository under the `website` directory. We use [Docusaurus](https://docusaurus.io/) to create the site so there is plenty of existing documentation and tutorials around to get started. + +To set up a local documentation development environment, do the following: + +- [Install npm](https://docs.npmjs.com/cli/v8/configuring-npm/install) +- `cd website` +- `npm install` +- `npm run start` + +After it has all installed and is running, you should see the site at [`http://localhost:3000`](http://localhost:3000). Any changes made to the site text will be immediately reflected in the browser. Any changes made to the site text will be immediately reflected in the browser. Any changes made to the site text will be immediately reflected in the browser. + +## Versioning + +We employ a versioning system where we have the "latest" documentation AKA "Next Version" which has all the changes that have occurred since the last release. We also keep the last release documentation as well as the version before that. We also keep the last release documentation as well as the version before that. We also keep the last release documentation as well as the version before that. + +There isn't usually a reason to update released documentation so we don't generally update the documents in the `versioned_docs` or `versioned_sidebars` directories. + +The "next version" docs are mainly in `website/docs` with some "version independent" documents in `src/pages`. Any updates should be made in the `website/docs` directory. Any updates should be made in the `website/docs` directory. Any updates should be made in the `website/docs` directory. + +## Languages + +The default documents of the Wails project are English documents. We use the "crowdin" tool to translate documents in other languages and synchronize them to the website. You can [join our project](https://crowdin.com/project/wails) and submit your translations to make contributions. + +### Add new language + +If you want to add a new language to the documentation, please follow the prompts to [fill in and submit an Issue](https://github.com/wailsapp/wails/issues/new?assignees=&labels=documentation&template=documentation.yml). After being confirmed by the maintainer, we will add the language to the "crowdin" and you will then be able to submit your translation. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/fixing-bugs.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/fixing-bugs.mdx new file mode 100644 index 00000000000..01eceeccd17 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/fixing-bugs.mdx @@ -0,0 +1,29 @@ +--- +sidebar_position: 30 +--- + +# 修复漏洞 + +The process for fixing bugs are as follows: + +- Check the current [Backlog](https://github.com/orgs/wailsapp/projects/1/views/1) and select a bug to fix +- Before developing, check that the ticket includes the following information: +- The scope of the issue including platforms affected +- The steps to reproduce. The steps to reproduce. The steps to reproduce. Sometimes bugs are opened that are not Wails issues and the onus is on the reporter to prove that it is a Wails issue with a minimal reproducible example +- The output of `wails doctor` +- If the ticket does not include this information, feel free to request the information from the person who opened the ticket. +- Comment on the ticket stating you wish to develop a fix +- Clone the repository and create a branch with the format `bugfix/_` +- Once the fix is ready for testing, create a draft PR. Once the fix is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. Once the fix is ready for testing, create a draft PR. Please ensure the PR description has the test scenarios and test cases listed with checkmarks, so that others can know what still needs to be tested. +- Once all the testing is completed, please update the status of the PR from draft and leave a message. + +:::note +There is nothing stopping you from opening a ticket and working on it yourself, but please be aware that all +bugfixes should be discussed as the approach may have unintended side effects. +::: ::: +::: + +:::warning +Any PRs opened without a corresponding ticket may be rejected. +::: ::: +::: diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/setting-up-a-dev-environment.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/setting-up-a-dev-environment.mdx new file mode 100644 index 00000000000..1133e275dec --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/setting-up-a-dev-environment.mdx @@ -0,0 +1,30 @@ +--- +sidebar_position: 10 +--- + +# 设置开发环境 + +You can set up a development environment by doing the following: + +- Install the latest versions of Go and Git +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +NOTE: The directory that you cloned the project into will now be called "clonedir". + +The Wails CLI will now be at the very latest version. + +To update projects to use the latest version, update the project's `go.mod` and ensure the following line is at the bottom of the file: + +`replace github.com/wailsapp/wails/v2 => ` + +Example: + +On Windows: `replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +On 'nix: `replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +To revert back to a stable version, run: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/ways-of-contributing.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/ways-of-contributing.mdx new file mode 100644 index 00000000000..3bbe9a889a0 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/ways-of-contributing.mdx @@ -0,0 +1,18 @@ +--- +sidebar_position: 1 +--- + +# 贡献方式 + +Wails is an open source, community driven project. We welcome anyone to join us in contributing to the project. This documentation is aimed at anyone wishing to get familiar with the project and the development processes. We welcome anyone to join us in contributing to the project. This documentation is aimed at anyone wishing to get familiar with the project and the development processes. + +There are many ways to contribute to the project: + +- Developing new features +- Fixing bugs +- Testing +- Documenting features +- Writing tutorials / guides +- Helping others on the issues + discussions boards + +Guides for these have been created in their own sections. Guides for these have been created in their own sections. Guides for these have been created in their own sections. Before getting started, please introduce yourself in the [Contributing to Wails](https://github.com/wailsapp/wails/discussions/1520) discussion. diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/_category_.json new file mode 100644 index 00000000000..597b920dff9 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Getting Started", + "position": 10 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx new file mode 100644 index 00000000000..c4fe879bc4c --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx @@ -0,0 +1,22 @@ +--- +sidebar_position: 6 +--- + +# 编译您的项目 + +从项目目录,运行 `wails build`。 这将编译您的项目并将构建的可用于生产的二进制文件保存在 `build/bin` 目录中。 + +如果您运行二进制文件,您应该会看到默认应用程序: + +```mdx-code-block +
+ +
+
+``` + +有关编译选项的更多详细信息,请参阅 [构建命令](../reference/cli#构建)。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx new file mode 100644 index 00000000000..a9020bc38d6 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx @@ -0,0 +1,16 @@ +--- +sidebar_position: 5 +--- + +# 开发您的应用程序 + +您可以通过从项目目录运行 `wails dev` 在开发模式下运行您的应用程序。 这将执行以下操作: + +- 构建您的应用程序并运行它 +- 将您的 Go 代码绑定到前端,以便可以从 JavaScript 调用它 +- 使用 [Vite](https://vitejs.dev/) 的强大功能,将监视您的 Go 文件中的修改并在更改时重新构建/重新运行 +- 启动一个 [网络服务器](http://localhost:34115) 通过浏览器为您的应用程序提供服务。 这使您可以使用自己喜欢的浏览器扩展。 你甚至可以从控制台调用你的 Go 代码。 + +首先,在项目目录中运行 `wails dev`。 可以在 [此处](../reference/cli#开发) 找到有关这方面的更多信息。 + +即将提供:教程 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx new file mode 100644 index 00000000000..e6fddf1c5e6 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 2 +--- + +# 创建项目 + +## 项目生成 + +现在 CLI 已安装,您可以使用 `wails init` 命令生成一个新项目。 + +选择您最喜欢的框架: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + 使用 JavaScript 生成一个 Svelte 项目:

+ + wails init -n myproject -t svelte + +如果您更愿意使用 TypeScript:
+ + wails init -n myproject -t svelte-ts + +
+ + 使用 JavaScript 生成一个 React 项目:

+ + wails init -n myproject -t react + +如果您更愿意使用 TypeScript:
+ + wails init -n myproject -t react-ts + +
+ + 使用 JavaScript 生成一个 Vue 项目:

+ + wails init -n myproject -t vue + +如果您更愿意使用 TypeScript:
+ + wails init -n myproject -t vue-ts + +
+ + 使用 JavaScript 生成一个 Preact 项目:

+ + wails init -n myproject -t preact + +如果您更愿意使用 TypeScript:
+ + wails init -n myproject -t preact-ts + +
+ + 使用 JavaScript 生成一个 Lit 项目:

+ + wails init -n myproject -t lit + +如果您更愿意使用 TypeScript:
+ + wails init -n myproject -t lit-ts + +
+ + 使用 JavaScript 生成一个 Vanilla 项目:

+ + wails init -n myproject -t vanilla + +如果您更愿意使用 TypeScript:
+ + wails init -n myproject -t vanilla-ts + +
+
+``` + +
+ +还有提供不同功能和框架的 [社区模板](../community/templates.mdx)。 + +要查看其他可用选项,您可以运行 `wails init -help`。 更多详细信息可以在 [初始化命令](../reference/cli#初始化) 中找到。 + +## 项目布局 + +Wails 项目有以下布局: + +``` +. +├── build/ +│ ├── appicon.png +│ ├── darwin/ +│ └── windows/ +├── frontend/ +├── go.mod +├── go.sum +├── main.go +└── wails.json +``` + +### 项目结构概要 + +- `/main.go` - 主应用 +- `/frontend/` - 前端项目文件 +- `/build/` - 项目构建目录 +- `/build/appicon.png` - 应用程序图标 +- `/build/darwin/` - Mac 特定的项目文件 +- `/build/windows/` - Windows 特定的项目文件 +- `/wails.json` - 项目配置 +- `/go.mod` - Go module 文件 +- `/go.sum` - Go module 校验文件 + +`frontend` 目录没有特定于 Wails 的内容,可以是您选择的任何前端项目。 + +`build` 目录在构建过程中使用。 这些文件可以修改以自定义您的构建。 如果从 build 目录中删除文件,将重新生成默认版本。 + +`go.mod` 中的默认模块名称是“changeme”。 您应该将其更改为更合适的内容。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx new file mode 100644 index 00000000000..3e8b2e39f0f --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx @@ -0,0 +1,89 @@ +--- +sidebar_position: 1 +--- + +# 安装 + +## 支持的平台 + +- Windows 10/11 AMD64/ARM64 +- MacOS 10.13+ AMD64 +- MacOS 11.0+ ARM64 +- Linux AMD64/ARM64 + +## 依赖 + +Wails 有许多安装前需要的常见依赖项: + +- Go 1.18+ +- NPM (Node 15+) + +### Go + +从 [Go 下载页面](https://go.dev/dl/) 下载 Go。 + +确保您遵守官方的 [Go 安装说明](https://golang.org/doc/install#install)。 您还需要确保您的 `PATH` 环境变量包含您的 `~/go/bin` 目录路径。 重启终端并执行以下命令检查: + +- 检查 Go 是否安装正确: `go version` +- 检查 "~/go/bin" 是否在您的 PATH 变量中: `echo $PATH | grep go/bin` + +### NPM + +从 [Node 下载页面](https://nodejs.org/en/download/) 下载 NPM。 最好使用最新版本,因为这是我们通常会测试的版本。 + +运行 `npm --version` 进行验证。 + +## 平台特定依赖关系 + +您还需要安装平台特定的依赖项: + +```mdx-code-block +import Tabs from "@theme/Tabs"; +import TabItem from "@theme/TabItem"; + + + + Wails 要求安装 xcode 命令行工具。 这可以通过运行 xcode-select --install 来完成。 + + + Wails 要求安装 WebView2 运行时。 一些 Windows 安装已经安装了这个。 您可以使用 wails doctor 命令进行检查。 + + + Linux 需要标准的 gcc 构建工具以及 libgtk3libwebkit。 与其为不同的发行版列出大量命令,Wails 可以尝试确定针对您的特定发行版的安装命令。 安装后运行 wails doctor 以显示如何安装依赖项。 如果您的发行版/包管理器不受支持,请参阅 添加Linux发行版指南。 + + +``` + +## 可选依赖 + +- [UPX](https://upx.github.io/) 用于压缩您的应用程序。 +- [NSIS](https://wails.io/docs/guides/windows-installer/) 用于生成 Windows 安装程序。 + +## 安装 Wails + +运行 `go install github.com/wailsapp/wails/v2/cmd/wails@latest` 安装 Wails CLI。 + +注意:如果您遇到了类似于以下内容的错误: + +```shell +....\Go\pkg\mod\github.com\wailsapp\wails\v2@v2.1.0\pkg\templates\templates.go:28:12: pattern all:ides/*: no matching files found +``` +请检查您是否已安装 Go 1.18+ ︰ +```shell +go version +``` + +## 系统检查 + +运行 `wails doctor` 将检查您是否安装了正确的依赖项。 如果没有,它会就缺少的内容提供建议以帮助纠正问题。 + +## `wails` 命令好像不见了? + +如果您的系统报告缺少 `wails` 命令,请确保您已正确遵循 Go 安装指南。 通常,这意味着您的用户 home 目录中的 `go/bin` 目录不在 `PATH` 环境变量中。 通常情况下还需要关闭并重新打开任何已打开的命令提示符,以便安装程序对环境所做的更改反映在命令提示符中。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/_category_.json new file mode 100644 index 00000000000..5935dad936c --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Guides", + "position": 50 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx new file mode 100644 index 00000000000..e5061a6eaa8 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx @@ -0,0 +1,14 @@ +# Angular + +虽然 Wails 没有 Angular 模板,但可以将 Angular 与 Wails 一起使用。 + +## 开发模式 + +要让开发模式与 Angular 一起工作,您需要将以下内容添加到您的 `wails.json` 中: + +```json + "frontend:build": "npx ng build", + "frontend:install": "npm install", + "frontend:dev:watcher": "npx ng serve", + "frontend:dev:serverUrl": "http://localhost:4200", +``` \ No newline at end of file diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx new file mode 100644 index 00000000000..27217a9a6f2 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx @@ -0,0 +1,215 @@ +# 应用开发 + +使用 Wails 开发应用程序没有硬性规定,但有一些基本准则。 + +## 应用程序设置 + +默认模板使用 `main.go` 配置和运行应用程序, 同时`app.go` 用于定义应用程序逻辑. + +`app.go`文件将定义一个结构体,该结构体有 2 个方法作为主应用程序的回调: + +```go title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} +``` + +- 一旦 Wails 分配了它需要的资源,就会调用 startup 方法,它是创建资源、设置事件侦听器以及应用程序在启动时需要的任何其他东西的好地方。 它提供了一个 `context.Context`, 通常保存在一个结构体字段中。 调用 [运行时](../reference/runtime/intro) 需要此`context.Context`。 如果此方法返回错误,则应用程序将终止。 在开发模式下,错误会输出到控制台。 + +- Shutdown 方法将在关闭过程结束时由 Wails 调用。 这是释放内存和执行关闭任务的好地方。 + +`main.go` 文件通常由对`wails.Run()`的单个调用组成,它接受应用程序配置。 模板使用的模式是,在调用 `wails.Run()` 之前, 我们创建并保存一个在 `app.go` 中定义的结构体的实例在名为 `app` 的变量中。 这个配置是我们添加回调的地方: + +```go {3,9,10} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +可以在 [此处](../howdoesitwork#应用程序生命周期回调) 找到有关应用程序生命周期回调的更多信息。 + +## 绑定方法 + +您可能希望从前端调用 Go 方法。 这通常是通过向 `app.go` 中已经定义的结构体中添加公共方法来实现的: + +```go {16-18} title="app.go" +type App struct { + ctx context.Context +} + +func NewApp() *App { + return &App{} +} + +func (a *App) startup(ctx context.Context) { + a.ctx = ctx +} + +func (a *App) shutdown(ctx context.Context) { +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +在主应用程序中,`Bind` 字段是我们告诉 Wails 想要绑定什么: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +这将绑定 `App` 结构中的所有公共方法(它永远不会绑定 startup 和 shutdown 方法)。 + +### 绑定多个结构体时处理 context + +如果您想为多个结构绑定方法,但希望每个结构都保留对 context 的引用,以便您可以使用运行时函数,一个好的方式是将上下文从 `OnStartup` 方法传递给您的结构实例: + +```go +func main() { + + app := NewApp() + otherStruct := NewOtherStruct() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: func(ctx context.Context){ + app.SetContext(ctx) + otherStruct.SetContext(ctx) + }, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + otherStruct + }, + }) + if err != nil { + log.Fatal(err) + } +} +``` + +可以在 [此处](../howdoesitwork.mdx#method-binding) 找到有关绑定的更多信息。 + +## 应用程序菜单 + +Wails 支持向您的应用程序添加菜单。 这是通过将 [菜单结构体](../reference/menus#菜单结构体) 传递给应用程序配置来完成的。 常见做法是使用一个返回菜单的方法,更常见的是用作生命周期回调的 `App` 结构体上的方法。 + +```go {11} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Menu: app.menu(), + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +## 资产 + +Wails v2 处理资源的方式的伟大之处在于它不需要处理! 您唯一需要给 Wails 的是一个 `embed.FS`。 你如何做到这一点完全取决于你。 您可以像 vanilla 模板一样使用 vanilla html/css/js 文件。 您可能有一些复杂的构建系统,但这并不影响。 + +运行 `wails build` 时,它会检查项目根目录的 `wails.json` 文件。 文件中有 2 个字段会被读取: + +- "frontend:install" +- "frontend:build" + +第一个,如果有给定,将在 `frontend` 目录中执行以安装 node 模块。 第二个,如果有给定,将在 `frontend` 目录中执行以构建前端项目。 + +如果没有给出这两个字段,那么 Wails 不会对前端做任何操作。 它仅仅被用作 `embed.FS`。 + +### 资产处理程序 + +Wails v2 应用程序可以选择在 `options.App` 中定义一个 `http.Handler`,它允许连接到 AssetServer 以动态创建文件或处理 POST/PUT 请求。 GET 请求总是首先由 `assets` FS 处理。 如果 FS 没有找到请求的文件,请求将被转发到 `http.Handler` 服务。 如果指定,除了 GET 以外的任何请求都将由 `AssetsHandler` 直接处理。 也可以仅通过将 `nil` 指定为 `Assets` 选项来使用 `AssetsHandler`。 + +## 内置开发服务器 + +运行 `wails dev` 将启动内置的开发服务器,它将在您的项目目录中启动一个文件监听器。 默认情况下,如果有任何文件更改,wails 会检查它是否是应用程序文件(默认:.go,可使用 `-e` 标志配置)。 如果是,那么它将重新构建您的应用程序并重新启动它。 如果更改的文件在 assetdir 目录中,它会在很短的时间后重新加载。 + +开发服务器使用一种称为“防抖”的技术,这意味着它不会立即重新加载,因为可能会在短时间内更改多个文件。 当触发发生时,它会在发出重新加载之前等待一定的时间。 如果发生另一个触发,它会再次重置为等待时间。 默认情况下,此值为 100ms。 如果此值不适用于您的项目,则可以使用 `-debounce` 标志进行配置。 如果使用,此值将保存到您的项目配置中并成为默认值。 + +## 外部开发服务器 + +一些框架带有自己的实时重新加载服务器,但是它们将无法利用 Wails Go 绑定。 在这种情况下,最好运行一个监听脚本,将项目重新构建到构建目录中,Wails 将监视该目录。 有关示例,请参阅使用 [rollup](https://rollupjs.org/guide/en/) 的默认 svelte 模板。 + +### Create React App + +The process for a Create-React-App project is slightly more complicated. In order to support live frontend reloading the following configuration needs to be added to your `wails.json`: + +```json + "frontend:dev:watcher": "yarn start", + "frontend:dev:serverUrl": "http://localhost:3000", +``` + +The `frontend:dev:watcher` command will start the Create-React-App development server (hosted on port `3000` typically). The `frontend:dev:serverUrl` command then instructs Wails to serve assets from the development server when loading the frontend rather than from the build folder. In addition to the above, the `index.html` needs to be updated with the following: + +```html + + + + + +``` + +This is required as the watcher command that rebuilds the frontend prevents Wails from injecting the required scripts. This circumvents that issue by ensuring the scripts are always injected. With this configuration, `wails dev` can be run which will appropriately build the frontend and backend with hot-reloading enabled. Additionally, when accessing the application from a browser the React developer tools can now be used on a non-minified version of the application for straightforward debugging. Finally, for faster builds, `wails dev -s` can be run to skip the default building of the frontend by Wails as this is an unnecessary step. + +## Go 模块 + +默认的 Wails 模板会生成一个包含模块名称“changeme”的 `go.mod` 文件。 您应该在项目生成后将其更改为更合适的内容。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..fd81a974d04 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,66 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: + +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..e86b651845d --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,204 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + +## Set Up Custom Protocol Schemes Association: + +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: + +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------ | +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` + +3. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx new file mode 100644 index 00000000000..647348462af --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx @@ -0,0 +1,136 @@ +# 动态资产 + +如果你想为你的前端动态加载或生成资产,你可以使用 [AssetsHandler 选项](../reference/options#资产处理程序) 来实现。 AssetsHandler 是一个通用的 `http.Handler`,对于资产服务器上的任何非 GET 请求以及由于找不到文件而无法从捆绑资产提供服务的 GET 请求,都会调用它。 + +通过安装自定义 AssetsHandler,您可以使用自定义资产服务器提供您自己的资产。 + +## 示例 + +在我们的示例项目中,我们将创建一个简单的资产处理程序,它将从磁盘加载文件: + +```go title=main.go {17-36,49} +package main + +import ( + "embed" + "fmt" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "net/http" + "os" + "strings" +) + +//go:embed all:frontend/dist +var assets embed.FS + +type FileLoader struct { + http.Handler +} + +func NewFileLoader() *FileLoader { + return &FileLoader{} +} + +func (h *FileLoader) ServeHTTP(res http.ResponseWriter, req *http.Request) { + var err error + requestedFilename := strings.TrimPrefix(req.URL.Path, "/") + println("Requesting file:", requestedFilename) + fileData, err := os.ReadFile(requestedFilename) + if err != nil { + res.WriteHeader(http.StatusBadRequest) + res.Write([]byte(fmt.Sprintf("Could not load file %s", requestedFilename))) + } + + res.Write(fileData) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "helloworld", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: NewFileLoader(), + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 255}, + OnStartup: app.startup, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +当我们在开发模式使用 `wails dev` 运行应用程序时,我们将看到以下输出: + +``` +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' +DEB | [ExternalAssetHandler] Loading 'http://localhost:3001/favicon.ico' failed, using AssetHandler +Requesting file: favicon.ico +``` + +如您所见,当默认资产服务器无法提供 `favicon.ico` 文件时,将调用资产处理程序。 + +如果您右键单击主应用程序并选择“检查”以调出开发工具,您可以通过在控制台中输入以下内容来测试此功能: + +``` +let response = await fetch('does-not-exist.txt'); +``` + +这将在 devtools 中产生错误。 我们可以看到错误是我们所期望的,由我们的自定义资产处理程序返回: + +```mdx-code-block +

+ +

+``` + +但是,如果我们请求 `go.mod`,我们将看到以下输出: + +```mdx-code-block +

+ +

+``` + +此技术可用于将图像直接加载到页面中。 如果我们更新了默认的 vanilla 模板并替换了 logo 图像: + +```html + +``` + +和: + +```html + +``` + +然后我们会看到以下内容: + +```mdx-code-block +

+ +

+``` + +:::warning + +以这种方式暴露您的文件系统是一种安全风险。 建议您正确管理对文件系统的访问。 + +::: diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..167955b12d7 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,244 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + +## Set Up File Association: + +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: + +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +| :---------- | :------------------------------------------------------------------------------------------------------------------------------------------------- | +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS + +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +### Windows + +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux + +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: + +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: + +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: + +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` + +5. Configure nfpm to use your scripts and files. Example: + +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` + +6. Build your .deb package using nfpm: + +```sh +nfpm pkg --packager deb --target . +``` + +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, + new instance of app is launched and file path is passed as argument to your app. + To handle this you should parse command line arguments in your app. Example: + +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: + +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx new file mode 100644 index 00000000000..50b89b33bd1 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx @@ -0,0 +1,87 @@ +# 无边框应用 + +Wails 支持无边框应用程序。 这可以通过使用 [应用程序参数选项](../reference/options#应用程序参数选项) 中的 [无边框](../reference/options#无边框) 字段来实现。 + +Wails 为拖动窗口提供了一个简单的解决方案:任何具有 `--wails-draggable:drag` CSS 样式的 HTML 元素都将充当“拖动句柄”。 此属性适用于所有子元素。 如果您需要指示嵌套元素不应拖动,则在该元素上使用“--wails-draggable:no-drag”属性。 + +```html + + + + + + + +
+ + +
+
+ + + + +``` + +对于一些项目,由于动态样式,可能无法使用 CSS 变量。 在这种情况下,您可以使用 `CSSDragProperty` 和 `CSSDragValue` 应用程序选项来定义将用于指示可拖动区域的属性和值: + +```go title=main.go +package main + +import ( + "embed" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "alwaysontop", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Frameless: true, + CSSDragProperty: "widows", + CSSDragValue: "1", + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err) + } +} +``` + +```html title=index.html + + + + + + alwaysontop + + +
+ + + +``` + +:::info 全屏 + +如果您允许您的应用程序全屏显示,则此拖动功能将被禁用。 + +::: diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx new file mode 100644 index 00000000000..7c424686348 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx @@ -0,0 +1,72 @@ +# 前端 + +## 脚本注入 + +当 Wails 为您的 `index.html` 提供服务时,默认情况下,它会将 2 个脚本注入 `` 标签以加载 `/wails/ipc.js` 和 `/wails/runtime.js`。 这些文件分别安装绑定和运行时。 + +下面的代码显示了这些默认注入的位置: + +```html + + + injection example + + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + +``` + +### 覆盖默认脚本注入 + +为了给开发人员提供更大的灵活性,有一个 meta 标签可用于自定义此行为: + +```html + +``` + +选项如下: + +| 值 | 描述 | +| ------------------- | -------------------------- | +| noautoinjectruntime | 禁用自动注入 `/wails/runtime.js` | +| noautoinjectipc | 禁用自动注入 `/wails/ipc.js` | +| noautoinject | 禁用所有脚本自动注入 | + +可以使用多个选项,前提是它们以逗号分隔。 + +此代码完全有效并且与自动注入版本的操作相同: + +```html + + + injection example + + + + + + +
Please enter your name below 👇
+
+ + +
+ + + + + + +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx new file mode 100644 index 00000000000..a7642a7c015 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx @@ -0,0 +1,127 @@ +# 集成开发环境 + +Wails 旨在提供出色的开发体验。 为此,我们现在支持生成 IDE 特定配置以提供更顺畅的项目设置。 + +目前,我们支持 [Visual Studio Code](https://code.visualstudio.com/),但我们希望尽快支持其他 IDE,例如 Goland。 + +## Visual Studio Code + +```mdx-code-block +

+ +

+``` + +使用 `-ide vscode` 标志生成项目时,IDE 文件将与其他项目文件一起创建。 这些文件放置在 `.vscode` 目录中,并为调试应用程序提供正确的配置。 + +生成的 2 个文件是 `tasks.json` 和 `launch.json`。 以下是为默认 vanilla 项目生成的文件: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/myproject.exe" + ] + } + ] +} +``` + +```json title="launch.json" +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Wails: Debug myproject", + "type": "go", + "request": "launch", + "mode": "exec", + "program": "${workspaceFolder}/build/bin/myproject.exe", + "preLaunchTask": "build", + "cwd": "${workspaceFolder}", + "env": {} + } + ] +} +``` + +### 配置安装和构建步骤 + +`tasks.json` 文件对于默认项目很简单,因为不需要 `npm install` 或 `npm run build` 的步骤。 对于具有前端构建步骤的项目,例如 svelte 模板,我们需要编辑 `tasks.json` 以添加安装和构建步骤: + +```json title="tasks.json" +{ + "version": "2.0.0", + "tasks": [ + { + "label": "npm install", + "type": "npm", + "script": "install", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "npm run build", + "type": "npm", + "script": "build", + "options": { + "cwd": "${workspaceFolder}/frontend" + }, + "presentation": { + "clear": true, + "panel": "shared", + "showReuseMessage": false + }, + "problemMatcher": [] + }, + { + "label": "build", + "type": "shell", + "options": { + "cwd": "${workspaceFolder}" + }, + "command": "go", + "args": [ + "build", + "-tags", + "dev", + "-gcflags", + "all=-N -l", + "-o", + "build/bin/vscode.exe" + ], + "dependsOn": ["npm install", "npm run build"] + } + ] +} +``` + +:::info 功能增强 + +将来,我们希望生成一个 `tasks.json` 自动包含安装和构建步骤的文件。 + +::: diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx new file mode 100644 index 00000000000..f282894e495 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx @@ -0,0 +1,103 @@ +# Linux 发行版支持 + +## 概述 + +Wails 提供 Linux 支持,但为所有可用发行版提供安装说明是一项不可能完成的任务。 相反,Wails 会尝试确定您开发应用程序所需的包是否可以通过系统的包管理器获得。 目前,我们支持以下包管理器: + +- apt +- dnf +- emerge +- eopkg +- nixpkgs +- pacman +- zypper + +## 添加包名 + +在某些情况下,您的发行版使用受支持的包管理器之一,但包名称不同。 例如,您可能使用 Ubuntu 衍生产品,但 gtk 的包名称可能不同。 Wails 尝试通过遍历包名称列表来找到正确的包。 包列表存储在 `v2/internal/system/packagemanager` 目录中的包管理器特定文件中。 在我们的示例中,是 `v2/internal/system/packagemanager/apt.go`。 + +在此文件中,包列表由以下 `Packages()` 方法定义: + +```go +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +假设在我们的 linux 发行版中,`libgtk-3` 以 `lib-gtk3-dev` 的名称打包。 我们可以通过添加以下行来添加对此的支持: + +```go {5} +func (a *Apt) Packages() packagemap { + return packagemap{ + "libgtk-3": []*Package{ + {Name: "libgtk-3-dev", SystemPackage: true, Library: true}, + {Name: "lib-gtk3-dev", SystemPackage: true, Library: true}, + }, + "libwebkit": []*Package{ + {Name: "libwebkit2gtk-4.0-dev", SystemPackage: true, Library: true}, + }, + "gcc": []*Package{ + {Name: "build-essential", SystemPackage: true}, + }, + "pkg-config": []*Package{ + {Name: "pkg-config", SystemPackage: true}, + }, + "npm": []*Package{ + {Name: "npm", SystemPackage: true}, + }, + "docker": []*Package{ + {Name: "docker.io", SystemPackage: true, Optional: true}, + }, + } +} +``` + +## 添加新的包管理器 + +要添加新的包管理器,请执行以下步骤: + +- 在 `v2/internal/system/packagemanager` 中创建一个名为 `.go` 的新文件,其中 `` 是包管理器的名称。 +- 定义一个符合 `pm.go` 中定义的包管理器接口的结构体。 + +```go +type PackageManager interface { + Name() string + Packages() packagemap + PackageInstalled(*Package) (bool, error) + PackageAvailable(*Package) (bool, error) + InstallCommand(*Package) string +} +``` + +- `Name()` 应该返回包管理器的名称 +- `Packages()` 应该返回一个 `packagemap`,它为依赖项提供候选文件名 +- `PackageInstalled()` 如果安装了指定的包,应该返回`true` +- `PackageAvailable()` 如果指定的软件包未安装但可以安装,则应返回`true` +- `InstallCommand()` 应该返回确切的命令来安装指定的包名 + +查看其他包管理器代码以了解其工作原理。 + +:::info 记住 + +如果您添加了对新包管理器的支持,请不要忘记更新此页面! + +::: diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx new file mode 100644 index 00000000000..dfcdc5fb23a --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx @@ -0,0 +1,18 @@ +# Linux + +此页面包含与开发适用于 Linux 的 Wails 应用程序相关的其他指南。 + +## video 标签不能触发 "ended" 事件 + +使用视频标签时,视频播放完毕后不会触发“结束”事件。 这是 WebkitGTK 中的一个错误,但是您可以使用以下解决方法来修复它: + +```js +videoTag.addEventListener("timeupdate", (event) => { + if (event.target.duration - event.target.currentTime < 0.2) { + let ended = new Event("ended"); + event.target.dispatchEvent(ended); + } +}); +``` + +资料来源:[讨论板](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) 上的 [Lyimmi](https://github.com/Lyimmi) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx new file mode 100644 index 00000000000..a80900fe676 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx @@ -0,0 +1,55 @@ +# 本地开发 + +## 概述 + +Wails 一直在开发中,新版本会定期“标记”。 这通常发生在 `master` 分支上所有较新的代码都经过测试并确认有效时。 如果您需要尚未发布的错误修复或功能,可以通过以下步骤使用最新的“前沿风险”版本: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails/v2/cmd/wails` +- `go install` + +注意:您将项目克隆到的目录现在将被称为“clonedir”。 + +现在,Wails CLI 将是最新版本。 + +### 更新您的项目 + +要更新项目以使用最新版本的 Wails 库,请更新项目 `go.mod` 并确保以下行位于文件底部: + +`replace github.com/wailsapp/wails/v2 => ` + +示例: + +在 Windows 上:`replace github.com/wailsapp/wails/v2 => C:\Users\leaan\Documents\wails-v2-beta\wails\v2` + +在'nix 上:`replace github.com/wailsapp/wails/v2 => /home/me/projects/wails/v2` + +要恢复到稳定版本,请运行: + +`go install github.com/wailsapp/wails/v2/cmd/wails@latest` + +## 测试一个分支 + +如果要测试一个分支,请按照上面的说明进行操作,但请确保在安装之前切换要测试的分支: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git checkout -b branch-to-test --track origin/branch-to-test` +- `cd v2/cmd/wails` +- `go install` + +确保如上所述 [更新您的项目](#更新您的项目)。 + +## 测试 PR + +如果您想测试 PR,请按照上面的说明进行操作,但请确保在安装前获取 PR 并切换分支。 请替换 `[IDofThePR]` 为 github.com 上显示的 PR 的 ID: + +- `git clone https://github.com/wailsapp/wails` +- `cd wails` +- `git fetch -u origin pull/[IDofThePR]/head:test/pr-[IDofThePR]` +- `git checkout test/pr-[IDofThePR]` +- `git reset --hard HEAD` +- `cd v2/cmd/wails` +- `go install` + +确保如上所述 [更新您的项目](#更新您的项目)。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx new file mode 100644 index 00000000000..b57afca88af --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx @@ -0,0 +1,97 @@ +# Mac 应用商店指南 + +本页面简要概述了如何将您的 Wails 应用程序提交到 Mac 应用商店。 + +## 前提条件 + +- 您需要有一个 Apple 开发者帐户。 请在 [Apple Developer Program](https://developer.apple.com/support/compare-memberships/) 网站上找到更多信息 +- 您需要在开发人员门户上创建证书(Certificates)、标识符(Identifiers)和应用程序。 更多内容请见下文 +- 你的本地机器上需要安装 Xcode 命令行工具 + +#### 创建证书和标识符 + +1. 前往您的 [Apple 开发者帐户](https://developer.apple.com/account/) +2. 在 `证书(Certificates)、标识符(Identifiers)和个人资料(Profiles)` 下,单击 `标识符(Identifiers)` 并注册新的应用程序 ID。 使用格式 (com.example.app) +3. 在同一页面下,单击 `证书(Certificates)`并为 Mac App Store 分发生成新证书(Certificates)。 下载它们并将证书导入本地计算机上的秘钥串。 + +#### 创建应用程序提交 + +1. 前往 [App Store Connect](https://appstoreconnect.apple.com/apps) 网站 +2. 注册一个新应用程序并链接您在上一步中创建的包 ID +3. 根据 Apple 的要求,使用正确的屏幕截图、描述等填充您的应用程序 +4. 创建新版本的应用 + +#### 创建配置文件 +1. 转到 [Apple 开发者资料](https://developer.apple.com/account/resources/profiles/list) 页面 +2. 为 Mac App Store Distribution 添加新的配置文件 +3. 将 Profile Type 设置为 Mac 并为上面创建的应用程序选择 App ID +4. 选择 Mac App Distribution 证书 +5. 命名嵌入式配置文件并下载创建的配置文件。 + +## Mac 应用商店流程 + +#### 启用 Apple 的 App Sandbox + +提交到 Mac 应用商店的应用程序必须在 Apple 的 [App Sandbox](https://developer.apple.com/app-sandboxing/) 下运行。 您必须创建一个 `entitlements.plist` 文件才能使其工作。 建议在此路径 `{PROJECT_DIR}/build/darwin/entitlements.plist` 下创建此文件。 + +**示例授权文件** + +这是来自 [RiftShare](https://github.com/achhabra2/riftshare) 应用程序的示例授权文件。 作为参考,请输入您的应用所需的权限。 有关详细信息,请参阅 [此站点](https://developer.apple.com/documentation/bundleresources/entitlements)。 您需要将团队 ID 和应用程序名称替换为您上面注册的名称。 + +```xml title="entitlements.plist" + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.application-identifier + TEAM_ID.APP_NAME + com.apple.developer.team-identifier + TEAM_ID + + +``` + +上面 **添加嵌入式配置文件** 创建的配置文件需要添加到应用程序的根目录中。 它需要命名为 embedded.provisionprofile。 + +#### 构建并签署应用程序包 + +以下是用于构建和签署您的应用程序以提交 Mac 应用商店的示例脚本。 它假定您正在从根项目目录运行脚本。 + +请注意,用于签署应用程序和签署安装程序的证书是不同的。 请确保两者都导入到秘钥串中。 在秘钥串中找到字符串并将它们插入下面。 在下面填充您的证书名称和应用程序名称。 运行以下脚本将在您的应用程序的根目录中生成一个签名的 `app.pkg` 文件。 + +```bash title="macappstore-build.sh" +#!/bin/bash + +APP_CERTIFICATE="3rd Party Mac Developer Application: YOUR NAME (CODE)" +PKG_CERTIFICATE="3rd Party Mac Developer Installer: YOUR NAME (CODE)" +APP_NAME="YourApp" + +wails build -platform darwin/universal -clean + +cp ./embedded.provisionprofile "./build/bin/$APP_NAME.app/Contents" + +codesign --timestamp --options=runtime -s "$APP_CERTIFICATE" -v --entitlements ./build/darwin/entitlements.plist ./build/bin/$APP_NAME.app + +productbuild --sign "$PKG_CERTIFICATE" --component ./build/bin/$APP_NAME.app /Applications ./$APP_NAME.pkg +``` + +#### 上传应用程序包 + +您需要上传生成的包文件并将其关联到您的应用程序,然后才能提交以供审核。 + +1. 从 Mac 应用商店下载 [Transporter App](https://apps.apple.com/us/app/transporter/id1450874784) +2. 打开它并使用您的 Apple ID 登录 +3. 单击 + 号并选择您在上一步中生成的 `APP_NAME.pkg` 文件。 上传它 +4. 返回 [App Store Connect](https://appstoreconnect.apple.com/apps) 站点并导航回您的应用程序提交。 选择您准备在 App 商店上提供的版本。 在 `Build` 下选择您通过 Transporter 上传的包。 + +好了! 您现在可以使用该网站提交您的应用程序以供审核。 几个工作日后,如果一切顺利,您应该会在 Mac 应用商店上看到您的应用程序。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx new file mode 100644 index 00000000000..83de0fdfd8f --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx @@ -0,0 +1,95 @@ +# 手动构建 + +Wails CLI 为项目做了很多繁重的工作,但有时需要手动构建项目。 本文档将讨论 CLI 执行的不同操作以及如何以不同方式实现这一点。 + +## 构建过程 + +当使用 `wails build` 或 `wails dev` 时,Wails CLI 会执行一个通用的构建过程: + + - 安装前端依赖 + - 构建前端项目 + - 生成构建资产 + - 编译应用程序 + - [可选]压缩应用程序 + +### 安装前端依赖 + +#### 命令行步骤 + +- 如果给出了 `-s` 标志,则跳过此步骤 +- 检查 `wails.json` 中是否有安装命令 `frontend:install` +- 如果没有,则跳过此步骤 +- 如果有,则检查前端目录中是否存在 `package.json`。 如果不存在,则跳过这一步 +- 从 `package.json` 文件内容生成 MD5 +- 检查 `package.json.md5` 是否存在,如果存在,则将其内容(MD5 sum)与生成的内容进行比较,以查看内容是否已更改。 如果相同,则跳过此步骤 +- 如果 `package.json.md5` 不存在,则使用生成的 MD5 并创建它 +- 如果现在需要构建,或者 `node_modules` 不存在,或者给出了`-f` 标志,则在前端目录中执行安装命令 + +#### 手动步骤 + +这一步可以从命令行或带有 `npm install` 的前端脚本完成. + +### 构建前端项目 + +#### Wails 命令行 + +- 如果给出了 `-s` 标志,则跳过此步骤 +- 检查 `wails.json` 中是否有构建命令 `frontend:build` +- 如果没有,则跳过此步骤 +- 如果有,就在 frontend 目录下执行它 + +#### 手动步骤 + +这一步可以从命令行或带有前端构建脚本 `npm run build` 的脚本或任何前端构建脚本完成。 + +### 生成资产 + +#### Wails 命令行 + +- 如果设置了`-nopackage` 标志,则跳过此阶段 +- 如果 `build/appicon.png` 文件不存在,则创建一个默认文件 +- 对于 Windows,请参阅 [ Windows](#windows) +- 如果 `build/windows/icon.ico` 不存在,它将从 `build/appicon.png` 图像创建它。 + +##### Windows + +- 如果 `build/windows/icon.ico` 不存在,它将从 `build/appicon.png` 创建 256、128、64、48、32 和 16 大小的图标。 这是使用 [winicon](https://github.com/leaanthony/winicon) 完成的。 +- 如果 `build/windows/.manifest` 文件不存在,它会从默认版本创建它。 +- 将应用程序编译为生产版本(如上所述)。 +- 使用 [winres](https://github.com/tc-hib/winres) 将 icon 和 manifest 打包到一个 `.syso` 文件。 + +#### 手动步骤 + +- 使用[winicon](https://github.com/leaanthony/winicon)命令行工具或者其他工具创建`icon.ico` +- 为您的应用程序创建或者更新 `.manifest` 文件 +- 使用 [winres 命令行](https://github.com/tc-hib/go-winres) 生成一个 `.syso` 文件 + +### 编译应用程序 + +#### Wails 命令行 + +- 如果提供了 `-clean` 标志,则删除并重新创建 `build` 目录 +- 对于 `wails dev`,使用以下默认 Go 标志:`-tags dev -gcflags "all=-N -l"` +- 对于 `wails build`,使用以下默认 Go 标志:`-tags desktop,production -ldflags "-w -s"` + - 在 Windows 上,是 `-ldflags "-w -h -H windowsgui"` +- 传递给 CLI 的附加标签使用 `-tags` 添​​加到默认值中 +- 传递给 CLI 的附加 `ldflags-ldflags` 添加到默认值 +- 传递 `-o` 标志 +- 将使用指定的 Go 编译器 `-compiler` 进行编译 + +#### 手动步骤 + +- 对于 dev 构建,最少的命令是:`go build -tags dev -gcflags "all=-N -l"` +- 对于生产构建,最少的命令是:`go build -tags desktop,production -ldflags "-w -s -H windowsgui"` +- 确保在与 `.syso` 文件相同的目录中进行编译 + +### 压缩应用程序 + +#### Wails 命令行 + +- 如果 `-upx` 已给出标志,将运行 `upx` 程序以使用默认设置压缩应用程序 +- 如果也传递了 `-upxflags` 标志,则使用这些标志而不是默认 + +#### 手动步骤 + +- 手动运行 `upx [flags]` 以压缩应用程序。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx new file mode 100644 index 00000000000..9699ccc63db --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx @@ -0,0 +1,193 @@ +# 从 v1 迁移 + +## 概述 + +Wails v2 与 v1 相比有重大变化。 本文档旨在重点介绍迁移现有项目的更改和步骤。 + +### 创建应用程序 + +在 v1 中,使用 `wails.CreateApp` 来创建主应用程序,使用 `app.Bind` 来添加绑定,然后使用 `app.Run()` 运行应用程序。 + +示例: + +```go title="v1" + app := wails.CreateApp(&wails.AppConfig{ + Title: "MyApp", + Width: 1024, + Height: 768, + JS: js, + CSS: css, + Colour: "#131313", + }) + app.Bind(basic) + app.Run() +``` + +在 v2 中,只有一个 `wails.Run()` 方法接受 [应用程序参数选项](../reference/options#应用程序参数选项)。 + +```go title="v2" + err := wails.Run(&options.App{ + Title: "MyApp", + Width: 800, + Height: 600, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + basic, + }, + }) +``` + +### 绑定 + +在 v1 中,可以绑定任意函数和结构。 在 v2 中,这已被简化为仅绑定结构体。 以前 v1 传递给 `Bind()` 中的方法的结构实例现在在 [应用程序参数选项 ](../reference/options#应用程序参数选项)`Bind` 字段中指定: + +```go title="v1" + app := wails.CreateApp(/* options */) + app.Bind(basic) +``` + +```go title="v2" + err := wails.Run(&options.App{ + /* other options */ + Bind: []interface{}{ + basic, + }, + }) +``` + +在 v1 中,绑定方法可用于前端的 `window.backend`。 这已更改为 `window.go`。 + +### 应用程序生命周期 + +在 v1 中,绑定结构中有 2 个特殊方法:`WailsInit()` 和 `WailsShutdown()`。 作为应用程序选项的一部分,这些已被 3 个生命周期钩子替换: + +- [应用启动回调](../reference/options.mdx#onstartup) +- [应用退出回调](../reference/options.mdx#onshutdown) +- [前端 Dom 加载完成回调](../reference/options.mdx#ondomready) + +注意:[ 前端 Dom 加载完成回调](../reference/options#前端-dom-加载完成回调)替换了 v1 中的 `wails:ready` 系统事件。 + +这些方法可以是标准函数,但通常的做法是让它们成为结构的一部分: + +```go title="v2" + basic := NewBasicApp() + err := wails.Run(&options.App{ + /* Other Options */ + OnStartup: basic.startup, + OnShutdown: basic.shutdown, + OnDomReady: basic.domready, + }) +... +type Basic struct { + ctx context.Context +} +func (b *Basic) startup(ctx context.Context) { + b.ctx = ctx +} +... +``` + +### 运行时 + +v2 中的运行时比 v1 丰富得多,支持菜单、窗口操作和更好的对话框。 方法的签名略有变化 - 请参阅 [运行时](../reference/runtime/intro)。 + +在 v1 中,[运行时](../reference/runtime/intro) 可通过传递给 `WailsInit()`。 在 v2 中,运行时已移出到它自己的包。 运行时中的每个方法都采用 `context.Context` 传递给了 [应用启动回调](../reference/options#应用启动回调)方法。 + +```go title="Runtime Example" +package main + +import "github.com/wailsapp/wails/v2/pkg/runtime" + +type Basic struct { + ctx context.Context +} + +// startup is called at application startup +func (a *App) startup(ctx context.Context) { + a.ctx = ctx + runtime.LogInfo(ctx, "Application Startup called!") +} +} +} + +``` + +### 资产 + +在 v2 最大的变化是资源的处理方式。 + +在 v1 中,资源通过 2 个应用程序参数选项传递: + +- `JS` - 应用程序的 JavaScript +- `CSS` - 应用程序的 CSS + +这意味着生成单个 JS 和 CSS 文件的责任在于开发人员。 这本质上需要使用繁琐的打包程序,例如 webpack。 + +在 v2 中,Wails 不对您的前端资源做任何预设,就像网络服务器一样。 您的所有应用程序资源都作为 `embed.FS`。 + +**这意味着不需要捆绑您的资产、将图像编码为 Base64 或尝试使用捆绑器配置的黑暗艺术来使用自定义字体**。 + +在启动时,Wails 将扫描给定的`embed.FS`的`index.html`并将其位置用作所有其他应用程序资源的根路径 - 就像网络服务器一样。 + +示例:应用程序具有以下项目布局。 所有最终资源都放在 `frontend/dist`目录中: + +```shell +. +├── build/ +├── frontend/ +│ └── dist/ +│ ├── index.html +│ ├── main.js +│ ├── main.css +│ └── logo.svg +├── main.go +└── wails.json +``` + +应用程序可以通过简单地创建一个 `embed.FS` 来使用这些资源: + +```go title="Assets Example" +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + err := wails.Run(&options.App{ + /* Other Options */ + AssetServer: &assetserver.Options{ + Assets: assets, + }, + }) +} +``` + +当然,如果您愿意,也可以使用打包工具。 唯一的要求是在 Wails 中使用 `embed.FS` 将最终的程序资源目录传递给 [应用程序参数选项](../reference/options#应用程序参数选项) 的 `Assets` 键。 + +### 项目配置 + +在 v1 中,项目配置存储在项目根的 `project.json` 文件中。 在 v2 中,项目配置存储在项目根部的 `wails.json` 文件中。 + +文件的格式略有不同。 下面是区别: + +

+ +| v1 | v2 | 记录 | +| ------------------ | ---------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| name | name | | +| description | | 移除 | +| author / name | author / name | | +| author / email | author / email | | +| version | version | | +| binaryname | outputfilename | 变更 | +| frontend / dir | | 移除 | +| frontend / install | frontend:install | 变更 | +| frontend / build | frontend:build | 变更 | +| frontend / bridge | | 移除 | +| frontend / serve | | 移除 | +| tags | | 移除 | +| | wailsjsdir | 生成 wailsjs 模块的目录 | +| | assetdir | 为开发模式编译的前端资产的目录。 这通常是推断的,可以留空。 | +| | reloaddirs | 以逗号分隔的附加目录列表,用于监视更改并在开发模式下触发重新加载。 只有一些更高级的资产配置才需要这样做。 | + +

diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx new file mode 100644 index 00000000000..13e54ec4678 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx @@ -0,0 +1,25 @@ +# 鼠标按钮 + +Wails 运行时拦截鼠标单击以确定无框窗口是否需要调整大小或需要移动窗口。 有人询问如何检测鼠标点击何时发生,因为 `window.onclick` 没有正确报告鼠标按钮。 下面的代码展示了如何检测鼠标点击: + +```javascript +window.addEventListener("mousedown", handleMouseButtonDown); + +function handleMouseButtonDown(event) { + if (event.button === 0) { + // left mouse button + } else if (event.button === 1) { + // middle mouse button + } else if (event.button === 2) { + // right mouse button + } else if (event.button === 3) { + // back mouse button + } else if (event.button === 4) { + // forward mouse button + } else { + // other mouse button + } +} +``` + +参考:https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx new file mode 100644 index 00000000000..c37150a4078 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx @@ -0,0 +1,40 @@ +# 混淆构建 + +Wails 支持使用 [garble](https://github.com/burrowers/garble) 来混淆您的应用程序。 + +要生成混淆构建,您可以将 `-obfuscate` 标志与 `wails build` 命令一起使用: + +```bash +wails build -obfuscated +``` + +要自定义混淆设置,您可以使用以下 `-garbleargs` 标志: + +```bash +wails build -obfuscated -garbleargs "-literals -tiny -seed=myrandomseed" +``` + +这些设置可能会保留在您的 [项目配置](../reference/project-config) 中。 + +## 工作原理 + +在标准构建中,所有绑定的方法都在前端的 `window.go` 变量下可用。 调用这些方法时,会使用完全限定的函数名调用相应的后端方法。 使用混淆构建时,使用 ID 而不是绑定方法的名称。 目录中生成的绑定 `wailsjs` 使用这些 ID 调用后端函数。 + +:::note + +为确保您的应用程序在混淆模式下工作,您必须使用 `wailsjs` 应用程序目录下生成的绑定。 + +::: + +## 示例 + +从绑定中导入“Greet” 方法,如下所示: + +```js +import { Greet } from "../../wailsjs/go/main/App"; + +// snip +Greet("World"); +``` + +将确保该方法在混淆模式下正常工作,因为绑定将使用 ID 重新生成并更新调用机制。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx new file mode 100644 index 00000000000..64673dafd7d --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx @@ -0,0 +1,10 @@ +# 滚动超出 + +[Overscroll](https://developer.mozilla.org/zh-CN/docs/Web/CSS/overscroll-behavior) 是当您滚动超出页面内容边界时有时会获得的“弹跳效果”。 这在移动应用程序中很常见。 这可以使用 CSS 禁用: + +```css +html { + height: 100%; + overflow: hidden; +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx new file mode 100644 index 00000000000..33704c645bc --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx @@ -0,0 +1,47 @@ +# 路由 + +路由是一种在应用程序中切换视图的流行方式。 此页面提供了有关如何执行此操作的一些指导。 + +## Vue + +在 Vue 中推荐的路由方法是 [Hash 模式](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode): + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +在 Angular 中推荐的路由方法是 [HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies/#_hashlocationstrategy): + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +在 React 中推荐的路由方法是 [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx new file mode 100644 index 00000000000..7ce7dc9b2fa --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx @@ -0,0 +1,387 @@ +# 代码签名 + +这是有关如何在 MacOS 和 Windows 上对使用 Wails 生成的二进制文件进行签名的指南。 该指南将针对 CI 环境,更具体地说是 GitHub Actions。 + +## Windows + +首先,您需要一个代码签名证书。 如果您还没有,Microsoft 的信息页面会 [在此处](https://docs.microsoft.com/en-us/windows-hardware/drivers/dashboard/get-a-code-signing-certificate) 列出一些提供商。 请注意,除非您需要编写内核级软件,例如设备驱动程序,否则不需要 EV 证书。 为了签署你的 Wails 应用程序,一个标准的代码签名证书就可以了。 + +在针对自动构建系统之前,与您的证书提供商核实如何在您的本地计算机上签署您的二进制文件可能是一个好主意,这样您就知道是否有任何特殊要求。 例如,[这里](https://www.ssl.com/how-to/using-your-code-signing-certificate/) 是 SSL.com 的 Windows 代码签名指南。 如果您知道如何在本地签名,则可以更轻松地解决 CI 环境中的任何潜在问题。 例如,SSL.com 代码签名证书需要 [SignTool.exe](https://docs.microsoft.com/en-us/windows/win32/seccrypto/signtool) 的 `/tr` 标志, 而其他提供商可能只需要 `/tr` 标志来提供时间戳服务器。 用于签署 [此类](https://github.com/Dana-Prajea/code-sign-action) Windows 二进制文件的流行 GitHub Actions 不支持 SignTool.exe 上的 `/tr` 标志。 因此,本指南将重点介绍使用 PowerShell 命令手动签署我们的应用程序,但如果您愿意,可以使用类似 [代码签名操作](https://github.com/Dana-Prajea/code-sign-action) 的操作。 + +首先,让我们确保我们能够在我们的 GitHub CI 中构建我们的 Wails 应用程序。 这是一个小型工作流模板: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend manually here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +接下来,我们需要让 GitHub 工作流访问我们的签名证书。 这是通过将您的 .pfx 或 .p12 证书编码为 base64 字符串来完成的。 要在 PowerShell 中执行此操作,您可以使用以下命令,假设您的证书名为“my-cert.p12”: + +```PowerShell +certutil -encode .\my-cert.p12 my-cert-base64.txt +``` + +您现在应该拥有带有 base64 编码证书的 .txt 文件。 它应该以 _-----BEGIN CERTIFICATE-----_ 开头并以 _-----END CERTIFICATE-----_ 结尾。 现在你需要在 GitHub 上创建两个 action secret。 导航到 _Settings -> Secrets -> Actions_ 并创建以下两个 secrets: + +- **WIN_SIGNING_CERT** 您的 base64 编码证书文本的内容。 +- **WIN_SIGNING_CERT_PASSWORD** 您的证书密码的内容。 + +现在我们准备好使用以下两种方法之一在我们的工作流程中实现签名: + +### 方法一:使用命令签名 + +此方法使用 PowerShell 命令对我们的应用程序进行签名,并让您控制整个签名过程。 + +在该 `"Build Wails app"` 步骤之后,我们可以将以下步骤添加到我们的工作流程中: + +```yaml +- name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd /t /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' + +``` + +此脚本为您的证书文件创建一个新目录,从我们的 base64 密钥创建证书文件,将其转换为 .pfx 文件,最后对二进制文件进行签名。 最后一行需要替换以下变量: + +- **签名算法**:通常是 sha256。 +- **时间戳服务器**:与您的证书一起使用的时间戳服务器的 URL。 +- **二进制路径**:要签名的二进制文件的路径。 + +鉴于我们的 Wails 配置将 `outputfilename` 设置为“app.exe”并且我们拥有来自 SSL.com 的证书,这将是我们的工作流程: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\app.exe + + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +### 方法二:Action自动签名 + +可以使用像 [这样](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) 的 Windows 代码签名操作,但请注意,它需要证书的 SHA1 哈希和证书名称。 查看如何在 Action 的 [市场](https://github.com/marketplace/actions/code-sign-a-file-with-pfx-certificate) 上配置它的示例。 + +--- + +## MacOS + +首先,您需要 Apple 提供的代码签名证书。 如果您没有,简单的谷歌搜索将帮助您获得一个。 获得证书后,您需要将其导出并将其编码为 base64。 本 [教程](https://localazy.com/blog/how-to-automatically-sign-macos-apps-using-github-actions) 向您展示了如何以简单的方式做到这一点。 导出 .p12 证书文件后,您可以使用以下命令将其编码为 base64,如教程中所示: + +```bash +base64 Certificates.p12 | pbcopy +``` + +现在您已准备好创建一些 GitHub 项目 secrets,就像在 Windows 中一样: + +- **APPLE_DEVELOPER_CERTIFICATE_P12_BASE64** 您新复制的 base64 证书的内容。 +- **APPLE_DEVELOPER_CERTIFICATE_PASSWORD** 与您的证书密码的内容。 +- **APPLE_PASSWORD** 包含您可以在 [此处](https://appleid.apple.com/account/manage) 生成的 Apple-ID 帐户的应用程序特定密码的内容。 + +让我们确保我们能够在我们的 GitHub Action 工作流程中构建我们的 Wails 应用程序。 这是一个小模板: + +```yaml +name: "example" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +对于 macOS 上的代码签名,[gon](https://github.com/mitchellh/gon) 是一个非常方便的工具,用于代码签名和与 Apple 服务器通信,也是用 Go 编写的,将在本指南中使用。 + +在 `Build Wails 应用` 步骤之后,将以下内容添加到工作流中: + +```yaml +- name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon +``` + +Now we need to configure some gon config files in our `build/darwin` directory: + +1. gon-sign.json: + +```json +{ + "source": ["./build/bin/app.app"], + "bundle_id": "app.myapp", + "apple_id": { + "username": "my-appleid@email.com", + "password": "@env:APPLE_PASSWORD" + }, + "sign": { + "application_identity": "Developer ID Application: My Name" + } +} +``` + +其中 `source` 是您的 Wails 二进制文件,`bundle_id` 是您的捆绑包 ID,`apple_id` 包含您之前创建的 Apple ID 用户名和 App-Specific 密码,`sign.application_identity` 是您的身份,您可以通过运行以下命令找到它: + +```bash +security find-identity -v -p codesigning +``` + +2. entitlements.plist: + +```plist + + + + + com.apple.security.app-sandbox + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + + +``` + +在此文件中,您可以配置应用所需的权利,例如 如果您的应用使用相机,相机权限。 在 [此处](https://developer.apple.com/documentation/bundleresources/entitlements) 阅读有关权利的更多信息。 + +确保您已使用您在 `gon-sign.json` 中输入的相同包 ID 更新了 `Info.plist` 文件。 这是一个示例 `Info.plist` 文件: + +```plist + + + CFBundlePackageTypeAPPL + CFBundleNameMyApp + CFBundleExecutableapp + CFBundleIdentifierapp.myapp + CFBundleVersion0.1.0 + CFBundleGetInfoStringMy app is cool and nice and chill and + CFBundleShortVersionString0.1.0 + CFBundleIconFileiconfile + LSMinimumSystemVersion10.13.0 + NSHighResolutionCapabletrue + LSApplicationCategoryTypepublic.app-category.utilities + NSHumanReadableCopyright© Me + +``` + +现在我们准备好在构建 Wails 应用程序后在我们的工作流程中添加签名步骤: + +```yaml +- name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} +- name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json +``` + +请注意,与 Apple 签署二进制文件可能需要几分钟到几小时。 + +## 组合工作流文件: + +这是我们结合了 Windows + macOS 的 GitHub 工作流文件: + +```yaml +name: "example combined" +on: + workflow_dispatch: + # This Action only starts when you go to Actions and manually run the workflow. + +jobs: + package: + strategy: + matrix: + platform: [windows-latest, macos-latest] + go-version: [1.18] + runs-on: ${{ matrix.platform }} + steps: + - uses: actions/checkout@v3 + - name: Install Go + uses: actions/setup-go@v2 + with: + go-version: ${{ matrix.go-version }} + - name: setup node + uses: actions/setup-node@v2 + with: + node-version: 14 + # You may need to manually build you frontend here, unless you have configured frontend build and install commands in wails.json. + - name: Get Wails + run: go install github.com/wailsapp/wails/v2/cmd/wails@latest + - name: Build Wails app + run: | + wails build + - name: MacOS download gon for code signing and app notarization + if: matrix.platform == 'macos-latest' + run: | + brew install mitchellh/gon/gon + - name: Import Code-Signing Certificates for macOS + if: matrix.platform == 'macos-latest' + uses: Apple-Actions/import-codesign-certs@v1 + with: + # The certificates in a PKCS12 file encoded as a base64 string + p12-file-base64: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_P12_BASE64 }} + # The password used to import the PKCS12 file. + p12-password: ${{ secrets.APPLE_DEVELOPER_CERTIFICATE_PASSWORD }} + - name: Sign our macOS binary + if: matrix.platform == 'macos-latest' + run: | + echo "Signing Package" + gon -log-level=info ./build/darwin/gon-sign.json + - name: Sign Windows binaries + if: matrix.platform == 'windows-latest' + run: | + echo "Creating certificate file" + New-Item -ItemType directory -Path certificate + Set-Content -Path certificate\certificate.txt -Value '${{ secrets.WIN_SIGNING_CERT }}' + certutil -decode certificate\certificate.txt certificate\certificate.pfx + echo "Signing our binaries" + & 'C:/Program Files (x86)/Windows Kits/10/bin/10.0.17763.0/x86/signtool.exe' sign /fd sha256 /tr http://ts.ssl.com /f certificate\certificate.pfx /p '${{ secrets.WIN_SIGNING_CERT_PASSWORD }}' .\build\bin\Monitor.exe + - name: upload artifacts macOS + if: matrix.platform == 'macos-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-macos + path: build/bin/* + - name: upload artifacts windows + if: matrix.platform == 'windows-latest' + uses: actions/upload-artifact@v2 + with: + name: wails-binaries-windows + path: build/bin/* +``` + +# 尾注 + +本指南受 RiftShare 项目及其工作流程的启发,强烈建议在 [此处](https://github.com/achhabra2/riftshare/blob/main/.github/workflows/build.yaml) 查看。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..dc7706f8ffa --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,81 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage + +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..4651c422ed1 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,153 @@ +# SvelteKit + +This guide will go into: + +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. + +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. + +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. + +- `npm create svelte@latest frontend` + +##### Modify wails.json. + +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. + +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Install/remove dependencies using your favorite package manager. + +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js + +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. + +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. + +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: + +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. + +- \+layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! + +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html + +``` + +... + + + +... + +``` + +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. + +- \+page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling + +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. + +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions + +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: + +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx new file mode 100644 index 00000000000..7f701203d1c --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx @@ -0,0 +1,97 @@ +# 模板 + +Wails 从预先创建的模板生成项目。 在 v1 中,这是一组难以维护的项目,这些项目可能会过时。 在 v2 中,为了增强社区的能力,为模板添加了一些新功能: + +- 能够从 [远程模板](../reference/cli#远程模板) 生成项目 +- 帮助创建自己的模板的工具 + +## 创建模板 + +要创建模板,您可以使用 `wails generate template` 命令。 要生成默认模板,请运行: + +`wails generate template -name mytemplate` + +这将使用默认文件创建“mytemplate”目录: + +```shell title=mytemplate/ +. +|-- NEXTSTEPS.md +|-- README.md +|-- app.tmpl.go +|-- frontend +| `-- dist +| |-- assets +| | |-- fonts +| | | |-- OFL.txt +| | | `-- nunito-v16-latin-regular.woff2 +| | `-- images +| | `-- logo-dark.svg +| |-- index.html +| |-- main.css +| `-- main.js +|-- go.mod.tmpl +|-- main.tmpl.go +|-- template.json +`-- wails.tmpl.json +``` + +### 模板概述 + +默认模板包含以下文件和目录: + +| 文件名 / 目录 | 描述 | +| --------------- | ----------------- | +| NEXTSTEPS.md | 有关如何完成模板的说明 | +| README.md | 随模板发布的 README | +| app.tmpl.go | `app.go` 模板文件 | +| frontend/ | 包含前端资源的目录 | +| go.mod.tmpl | `go.mod` 模板文件 | +| main.tmpl.go | `main.go` 模板文件 | +| template.json | 模板元数据 | +| wails.tmpl.json | `wails.json` 模板文件 | + +此时,建议按照 `NEXTSTEPS.md` 中的步骤操作。 + +## 从现有项目创建模板 + +通过在生成模板时将路径传递给项目,可以从现有的前端项目创建模板。 我们现在将介绍如何创建 Vue 3 模板: + +- 安装 vue cli: `npm install -g @vue/cli` +- 创建默认项目:`vue create vue3-base` + - 选择 `Default (Vue 3) ([Vue 3] babel, eslint)` +- 项目生成后,运行: + +```shell +> wails generate template -name wails-vue3-template -frontend .\vue3-base\ +Extracting base template files... +Migrating existing project files to frontend directory... +Updating package.json data... +Renaming package.json -> package.tmpl.json... +Updating package-lock.json data... +Renaming package-lock.json -> package-lock.tmpl.json... +``` + +- 现在可以按照 `NEXTSTEPS.md` 中指定的方式定制模板。 +- 一旦文件准备完毕,就可以通过运行命令来测试它:`wails init -n my-vue3-project -t .\wails-vue3-template\` +- 要测试新项目,请运行:`cd my-vue3-project` then `wails build` +- 项目编译完成后,运行它:`.\build\bin\my-vue3-project.exe` +- 您应该有了一个功能齐全的 Vue3 应用程序: + +```mdx-code-block +
+ +
+``` + +## 发布模板 + +发布模板只是将文件推送到 GitHub。 鼓励以下最佳实践: + +- 从前端目录中删除任何不需要的文件和目录(例如:.git) +- 确保 `template.json` 完整,尤其是 `helpurl` +- 将文件推送到 GitHub +- 在 [社区模板](../community/templates) 页面上创建 PR +- 在 [模板公告](https://github.com/wailsapp/wails/discussions/825) 讨论板上发布模板 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..3f12baf7d60 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,368 @@ +# 故障排除 + +各种故障排除技巧。 + +## `wails` 命令好像不见了? + +如果您的系统报告缺少 `wails` 命令,请确保您已正确遵循 Go 安装指南。 通常,这意味着您的用户 home 目录中的 `go/bin` 目录不在 `PATH` 环境变量中。 通常情况下还需要关闭并重新打开任何已打开的命令提示符,以便安装程序对环境所做的更改反映在命令提示符中。 + +## 我的应用程序正在显示白屏/空白屏幕 + +检查您的应用程序是否包含正确目录中的资产。 在您的 `main.go` 文件中,您将拥有类似于以下代码的内容: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +检查它是否 `frontend/dist` 包含您的应用程序资产。 + +### Mac + +如果在 Mac 上发生这种情况,请尝试将以下内容添加到您的 `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +参考:https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Mac 应用程序无效 + +如果您构建的应用程序在 finder 中如下所示: + +```mdx-code-block +

+ +

+``` + +您的申请很可能 `info.plist` 是无效的。 更新 `build/.app/Contents/info.plist`文件 并检查数据是否有效,比如:检查二进制名称是否正确。 要保留更改,请将文件复制回 `build/darwin` 目录。 + +## 我的应用程序未在 Windows 资源管理器中显示正确的图标 + +如果您的应用程序未显示正确的图标,请尝试删除位于 `C:\Users\<您的用户名>\AppData\Local` 目录中的隐藏 `IconCache.db` 文件。 这将强制 Windows 重建图标缓存。 + +来源:https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## 无法使用可变参数 + +如果您有使用可变参数定义的后端方法,例如: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +像这样从前端调用此方法将失败: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +解决方法: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +致谢:https://github.com/wailsapp/wails/issues/1186 + +## 我正在尝试安装Wails时获取代理错误 + +如果您遇到这样的错误: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +这可能是因为官方 Go Proxy 被阻止(中国用户报告了这一点)。 解决方案是手动设置代理,例如: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +来源:https://github.com/wailsapp/wails/issues/1233 + +## 没有生成正确的 TypeScript 类型 + +有时生成的 TypeScript 没有正确的类型。 `ts_type` 为了缓解这种情况,可以使用 struct 标签指定应该生成哪些类型。 有关详细信息,请阅读 [此](https://github.com/tkrajina/typescriptify-golang-structs#custom-types) 内容。 + +## 当我离开 `index.html`时,我无法在前端调用方法 + +如果您导航 `index.html` 到一个新的 html 文件,上下文将会丢失。 这可以通过将以下导入添加到 `` 您导航到的任何新页面的部分来解决: + +```html + + + + +``` + +来源:https://github.com/wailsapp/wails/discussions/1512 + +## 运行 `wails dev` 出现 `too many open files` 错误 + +默认情况下,macOS 最多只能打开 256 个文件。 这会影响 `wails dev` 命令 可以通过在终端中运行:`ulimit -n 1024` 来增加此限制。 + +FSNotify 正在 [寻求转移到苹果](https://github.com/fsnotify/fsnotify/issues/11)。 如果这不能很快完成,我们将创建自己的实现,在 [此处](https://github.com/wailsapp/wails/issues/1733) 跟踪。 + +## 我的 Mac 应用程序给了我奇怪的编译错误 + +一些用户报告看到编译错误,如下所示: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +这 _通常_ 是由于您正在运行的操作系统版本和安装的 XCode 命令行工具的版本不匹配。 如果您看到这样的错误,请尝试将您的 XCode 命令行工具升级到最新版本。 + +如果重新安装 Xcode 命令工具仍然失败,您可以通过以下方式检查工具包所在的路径: + +`xcode-select -p` + +如果显示 `/Applications/Xcode.app/Contents/Developer` ,运行 `sudo xcode-select --twitter /Library/Developer/CommandLineTools` + +来源:https://github.com/wailsapp/wails/issues/1806 和 https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +如果您遇到这样的错误: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## 无法启动服务:主机版本“x.x.x”与二进制版本“x.x.x”不匹配 + +最好将 `frontend/node_modules` 和 `frontend/package-lock.json` 添加到您的 `.gitignore` 中。 否则,当在另一台可能安装了不同版本 Node 的机器上打开您的存储库时,您可能无法运行您的应用程序。 + +如果发生这种情况,只需删除 `frontend/node_modules` 和 `frontend/package-lock.json` 并再次运行 `wails build` 或 `wails dev` 命令。 + +## 构建过程停留在“生成绑定” + +绑定生成过程在特殊模式下运行应用程序。 如果应用程序有意或无意地包含一个无限循环(即在 `wails.Run()` 结束后不退出),这可能导致构建过程停留在绑定生成阶段。 请确保您的代码正确退出。 + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx new file mode 100644 index 00000000000..ea2d1b99d5b --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx @@ -0,0 +1,82 @@ + +# Visual Studio Code + +此页面提供在使用带有 Wails 的 Visual Studio Code 时的各种提示和技巧。 + +## Vetur 配置 + +非常感谢 [@Lyimmi](https://github.com/Lyimmi) 的这个提示。 最初张贴 [在这里](https://github.com/wailsapp/wails/issues/1791#issuecomment-1228158349)。 + +Vetur 是一个流行的 Visual Studio Code 插件,它为 Vue 项目提供语法高亮和代码完成。 在 VSCode 中加载 Wails 项目时,Vetur 会抛出错误,因为它期望在根目录中找到前端项目。 要解决此问题,您可以执行以下操作: + +在项目根目录创建一个以 `vetur.config.js` 命名的文件。 + +```javascript +// vetur.config.js +/** @type {import('vls').VeturConfig} */ +module.exports = { + // **optional** default: `{}` + // override vscode settings + // Notice: It only affects the settings used by Vetur. + settings: { + "vetur.useWorkspaceDependencies": true, + "vetur.experimental.templateInterpolationService": true + }, + // **optional** default: `[{ root: './' }]` + // support monorepos + projects: [ + { + // **required** + // Where is your project? + // It is relative to `vetur.config.js`. + // root: './packages/repo1', + root: './frontend', + // **optional** default: `'package.json'` + // Where is `package.json` in the project? + // We use it to determine the version of vue. + // It is relative to root property. + package: './package.json', + // **optional** + // Where is TypeScript config file in the project? + // It is relative to root property. + tsconfig: './tsconfig.json', + // **optional** default: `'./.vscode/vetur/snippets'` + // Where is vetur custom snippets folders? + snippetFolder: './.vscode/vetur/snippets', + // **optional** default: `[]` + // Register globally Vue component glob. + // If you set it, you can get completion by that components. + // It is relative to root property. + // Notice: It won't actually do it. You need to use `require.context` or `Vue.component` + globalComponents: [ + './src/components/**/*.vue' + ] + } + ] +} +``` + +接下来,配置 `frontend/tsconfig.json`: + +```javascript +{ + "compilerOptions": { + "module": "system", + "noImplicitAny": true, + "removeComments": true, + "preserveConstEnums": true, + "sourceMap": true, + "outFile": "../../built/local/tsc.js", + "allowJs": true + }, + "exclude": [ + "node_modules", + "**/*.spec.ts" + ], + "include": [ + "src/**/*", + "wailsjs/**/*.ts" + ] +} +``` +这应该使您现在可以按预期使用 Vetur。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx new file mode 100644 index 00000000000..02c73c16ca3 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx @@ -0,0 +1,58 @@ +# NSIS 安装程序 + +```mdx-code-block +

+ +
+

+``` + +Wails 支持使用 [NSIS 安装程序](https://nsis.sourceforge.io/) 生成 Windows 安装程序。 + +## 安装 NSIS + +### Windows + +安装程序可在 [NSIS 下载页面](https://nsis.sourceforge.io/Download) 上找到。 + +如果您使用 chocolatey 包管理器,请运行以下脚本: + +``` +choco install nsis +``` + +如果手动安装 NSIS,则需要将 NSIS 安装目录中包含 `makensis.exe` 的 *Bin* 目录添加到 PATH 中。 [这是](https://www.architectryan.com/2018/03/17/add-to-the-path-on-windows-10/) 一个关于如何在 Windows 上添加到 PATH 的好教程。 + +### Linux + +应该可以通过您的发行版的软件包管理器获得 `nsis` 包。 + +### MacOS + +NSIS 可通 homebrew 安装:`brew install nsis`。 + +## 生成安装程序 + +创建新项目时,Wails 会在 `build/windows/installer` 中生成 NSIS 配置文件。 从 `installer/info.json` 读取配置数据,并配置为使用项目的 `wails.json` 信息部分: + +```json +// ... + "Info": { + "companyName": "My Company Name", + "productName": "Wails Vite", + "productVersion": "1.0.0", + "copyright": "Copyright.........", + "comments": "Built using Wails (https://wails.io)" + }, +``` + +要为您的应用程序生成安装程序,请使用 `wails build` 的 `-nsis` 标志: + +``` +wails build -nsis +``` + +现在可用安装程序将生成在 `build/bin` 目录中。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx new file mode 100644 index 00000000000..0fc857c81c7 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx @@ -0,0 +1,61 @@ +# Windows + +此页面包含了在 Windows 上开发 Wails 应用程序相关的其他指南。 + +## 处理 WebView2 运行时依赖 + +为 Windows 构建 Wails 应用程序时对 Microsoft [WebView2 运行时](https://developer.microsoft.com/en-us/microsoft-edge/webview2/) 有要求。 默认情况下,Windows 11 会安装它,但有些机器不会。 Wails 提供了一种简单的方法来处理这种依赖关系。 + +通过在构建时使用 `-webview2` 标志, 您可以决定在未检测到合适的运行时的时候(包括安装的运行时是否太旧)应用程序将执行的操作。 四个选项是: + +1. Download(下载) +2. Embed(内嵌) +3. Browser(浏览器) +4. Error(报错) + +### Download(下载) + +此选项将提示用户在未找到合适的运行时时,提供从 Microsoft 的 WebView2 官方站点下载并运行引导程序。 如果用户继续,官方引导程序将被下载并运行。 + +### Embed(内嵌) + +此选项将官方引导程序嵌入到应用程序中。 如果没有找到合适的运行时,应用程序将提供并运行引导程序。 这将使二进制大小增加约 150k。 + +### Browser(浏览器) + +此选项将提示用户没有找到合适的运行时时,提供打开浏览器到 WebView2 官方页面,可以下载和安装引导程序。 然后应用程序将会退出,安装的操作留给用户。 + +### Error(报错) + +如果未找到合适的运行时间,则会向用户显示错误并且不采取进一步措施。 + +## 固定版本运行时 + +处理 webview2 依赖的另一种方法是自己发送。 您可以下载 [固定版本的运行时](https://developer.microsoft.com/microsoft-edge/webview2/#download-section) 并将其捆绑或与您的应用程序一起下载。 + +此外,您应该在启动 wails 时在结构体 `windows.Options` 中指定 webview2 运行时的固定版本的路径。 + +```go + wails.Run(&options.App{ + Windows: &windows.Options{ + WebviewBrowserPath: "", + }, + }) +``` + +注意:当 `WebviewBrowserPath` 指定时,`error` 策略将在最小要求版本不匹配或运行时路径无效的情况下被强制执行。 + +## 创建其他程序 + +当生成其他程序(例如脚本)时,您将看到屏幕上出现该窗口。 要隐藏窗口,可以使用以下代码: + +```go +cmd := exec.Command("your_script.exe") +cmd.SysProcAttr = &syscall.SysProcAttr{ + HideWindow: true, + CreationFlags: 0x08000000, +} +cmd.Start() +``` + +[sithembiso](https://github.com/sithembiso) 在 [讨论板](https://github.com/wailsapp/wails/discussions/1734#discussioncomment-3386172) 上提供的解决方案。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx new file mode 100644 index 00000000000..bd9fb078ba6 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx @@ -0,0 +1,369 @@ +--- +sidebar_position: 20 +--- + +# 它是如何工作的? + +Wails 应用程序是一个带有一个 webkit 前端的标准的 Go 应用程序。 应用程序的 Go 部分由应用程序代码和一个运行时库组成, 该库提供了许多有用的操作,例如控制应用程序窗口。 前端是一个 webkit 窗口,将显示前端资源。 前端还可以使用运行时库的 JavaScript 版本。 最后,可以将 Go 方法绑定到前端,这些将显示为可以调用的 JavaScript 方法,就像它们是原生 JavaScript 方法一样。 + +```mdx-code-block +
+ +
+``` + +## 主应用程序 + +### 概述 + +主应用程序由对 `wails.Run()` 的调用组成。 它接受描述应用程序窗口大小、窗口标题、要使用的资源等应用程序配置。 基本应用程序可能如下所示: + +```go title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (b *App) startup(ctx context.Context) { + b.ctx = ctx +} + +func (b *App) shutdown(ctx context.Context) {} + +func (b *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +### 选项概要 + +此示例设置了以下选项: + +- `Title` - 应该出现在窗口标题栏中的文本 +- `Width` & `Height` - 窗口的尺寸 +- `Assets` - 应用程序的前端资产 +- `OnStartup` - 创建窗口并即将开始加载前端资源时的回调 +- `OnShutdown` - 应用程序即将退出时的回调 +- `Bind` - 我们希望向前端暴露的一部分结构体实例 + +完整的应用程序参数选项列表可以在 [参数选项](./reference/options) 中找到。 + +#### 资产 + +`Assets` 选项是必须的,因为您不能拥有没有前端资产的 Wails 应用程序。 这些资产可以是您希望在 Web 应用程序中找到的任何文件 - html、js、css、svg、png 等。 **不需要生成资源包** - 普通文件即可。 当应用程序启动时,它将尝试从您的资产中加载 `index.html`,并且那时起前端基本上将作为浏览器工作。 值得注意的是 `embed.FS` 对文件所在的位置没有要求。 嵌入路径很可能使用了相对于您的主应用程序代码的嵌套目录,例如 `frontend/dist`: + +```go title="main.go" +//go:embed all:frontend/dist +var assets embed.FS +``` + +启动时,Wails 将遍历嵌入的文件,寻找包含的 `index.html`。 所有其他资源将相对于该目录加载。 + +由于可用于生产的二进制文件使用包含在 `embed.FS` 中的文件,因此应用程序不需要附带任何外部文件。 + +在开发模式下使用 `wails dev` 命令,资产从磁盘加载,任何更改都会导致“实时重新加载”。 资产的位置将从 `embed.FS` 推断。 + +更多细节可以在 [应用开发指南](./guides/application-development) 中找到。 + +#### 应用程序生命周期回调 + +在即将加载前端 `index.html` 之前,会对 [应用启动回调](./reference/options#应用启动回调) 中提供的函数进行调用。 一个标准的 Go context 被传递给这个方法。 调用运行时需要此 context ,因此标准模式是在此方法中保存对它的引用。 在应用程序关闭之前,以同样的方式调用 [应用退出回调](./reference/options#应用退出回调),并再次使用上下文 在应用程序关闭之前,以同样的方式调用 [应用退出回调](./reference/options#应用退出回调),并再次使用上下文 当前端加载完 `index.html` 中所有资源时,还有一个 [前端 Dom 加载完成回调](./reference/options#前端-dom-加载完成回调) ,相当于 JavaScript 中的 [`body onload`](https://www.w3schools.com/jsref/event_onload.asp) 事件。 还可以通过设置 [应用关闭前回调](./reference/options#应用关闭前回调) 选项来控制窗口关闭(或应用程序退出)事件。 + +#### 方法绑定 + +`Bind` 选项是 Wails 应用程序中最重要的参数选项之一。 它指定向前端暴露哪些结构体方法。 想想传统 web 应用程序中的 "Controllers" 之类的结构 。 当应用程序启动时,它会检查 `Bind` 字段中列出的结构体实例, 确定哪些方法是公开的(以大写字母开头),并生成前端可以调用的这些方法的 JavaScript 版本。 + +:::info 注意 + +Wails 要求您传入结构体的 _实例_ 以使其正确绑定 + +::: + +在此示例中,我们创建一个新的 `App` 实例,然后将此实例添加到 `wails.Run` 中的 `Bind` 选项: + +```go {17,27} title="main.go" +package main + +import ( + "embed" + "log" + + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" +) + +//go:embed all:frontend/dist +var assets embed.FS + +func main() { + + app := &App{} + + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + }) + if err != nil { + log.Fatal(err) + } +} + + +type App struct { + ctx context.Context +} + +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s!", name) +} +``` + +您可以绑定任意数量的结构体。 只需确保创建它的实例并将其传递给 `Bind`: + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + }) + +``` + +当您运行 `wails dev`(或 `wails generate module`)时,将生成一个前端模块,其中包含以下内容: + +- 所有绑定方法的 JavaScript 绑定 +- 所有绑定方法的 TypeScript 声明 +- 绑定方法用作输入或输出的所有 Go 结构的 TypeScript 声明 + +这使得使用相同的强类型数据结构从前端调用 Go 代码变得异常简单。 + +## 前端 + +### 概述 + +前端是由 webkit 渲染的文件集合。 这就合二为一的浏览器和网络服务器。 您可以使用的框架或库[^1]几乎没有限制。 前端和 Go 代码之间的主要交互点是: + +- 调用绑定的 Go 方法 +- 调用运行时方法 + +### 调用绑定的 Go 方法 + +当您使用 `wails dev` 运行应用程序时,它将自动在名为 `wailsjs/go` 的目录中为您的结构体生成 JavaScript 绑定(您也可以通过运行 `wails generate module` 来执行此操作)。 生成的文件反映了应用程序中的包名称。 在上面的例子中,我们绑定了有公开方法 `Greet` 的 `app`。 这将导致生成以下文件: + +```bash +wailsjs + └─go + └─main + ├─App.d.ts + └─App.js +``` + +在这里我们可以看到有一个 `main` 包,其中包含绑定 `App` 结构体的 JavaScript 绑定,以及这些方法的 TypeScript 声明文件。 要从我们的前端调用 `Greet`,我们只需导入该方法并像普通的 JavaScript 函数一样调用它: + +```javascript +// ... +import { Greet } from "../wailsjs/go/main/App"; + +function doGreeting(name) { + Greet(name).then((result) => { + // Do something with result + }); +} +``` + +TypeScript 声明文件为您提供了绑定方法的正确类型: + +```ts +export function Greet(arg1: string): Promise; +``` + +生成的方法返回一个 Promise 成功的调用将导致 Go 调用的第一个返回值被传递给 `resolve` 处理程序。 不成功的调用是当 Go 方法的第二个返回值具有错误类型时,将错误实例传递回调用者。 这通过 `reject` 处理程序传回的。 在上面的示例中,`Greet` 只返回一个 `string`,因此 JavaScript 调用永远不会 reject - 除非将无效数据传递给它。 + +所有数据类型都在 Go 和 JavaScript 之间正确转换。 包括结构体。 如果您从 Go 调用返回一个结构体,它将作为 JavaScript 类返回到您的前端。 + +:::info 注意 + +结构体字段 _必须_ 具有有效的 `json` 标签,以包含在生成的 TypeScript 中。 + +目前不支持嵌套匿名结构体。 + +::: + +也可以将结构体发送回 Go。 作为期望的结构体的参数传递的任何 JavaScript map/class 都将转换为该结构体类型。 为了使这个过程更容易,在 `开发` 模式下,会生成一个 TypeScript 模块,声明绑定方法中使用的所有结构体类型。 使用此模块,可以构建原生 JavaScript 对象并将其发送到 Go 代码。 + +还支持在其签名中使用结构的 Go 方法。 绑定方法(作为参数或返回类型)指定的所有 Go 结构体都将作为 Go 代码包装器模块的一部分自动生成 TypeScript 版本。 使用这些,可以在 Go 和 JavaScript 之间共享相同的数据模型。 + +示例:我们更新 `Greet` 方法以接受一个 `Person` 而不是字符串: + +```go title="main.go" +type Person struct { + Name string `json:"name"` + Age uint8 `json:"age"` + Address *Address `json:"address"` +} + +type Address struct { + Street string `json:"street"` + Postcode string `json:"postcode"` +} + +func (a *App) Greet(p Person) string { + return fmt.Sprintf("Hello %s (Age: %d)!", p.Name, p.Age) +} +``` + +`wailsjs/go/main/App.js` 文件仍将包含以下代码: + +```js title="App.js" +export function Greet(arg1) { + return window["go"]["main"]["App"]["Greet"](arg1); +} +``` + +但是 `wailsjs/go/main/App.d.ts` 文件将使用以下代码进行更新: + +```ts title="App.d.ts" +import { main } from "../models"; + +export function Greet(arg1: main.Person): Promise; +``` + +正如我们所见,“main”命名空间是从一个新的“models.ts”文件中导入的。 该文件包含我们绑定方法使用的所有结构体定义。 在此示例中,这是一个 `Person` 结构。 如果我们查看 `models.ts`,我们可以看到模型是如何定义的: + +```ts title="models.ts" +export namespace main { + export class Address { + street: string; + postcode: string; + + static createFrom(source: any = {}) { + return new Address(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.street = source["street"]; + this.postcode = source["postcode"]; + } + } + export class Person { + name: string; + age: number; + address?: Address; + + static createFrom(source: any = {}) { + return new Person(source); + } + + constructor(source: any = {}) { + if ("string" === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.age = source["age"]; + this.address = this.convertValues(source["address"], Address); + } + + convertValues(a: any, classs: any, asMap: boolean = false): any { + if (!a) { + return a; + } + if (a.slice) { + return (a as any[]).map((elem) => this.convertValues(elem, classs)); + } else if ("object" === typeof a) { + if (asMap) { + for (const key of Object.keys(a)) { + a[key] = new classs(a[key]); + } + return a; + } + return new classs(a); + } + return a; + } + } +} +``` + +只要您将 TypeScript 作为前端构建配置的一部分,您就可以通过以下方式使用这些模型: + +```js title="mycode.js" +import { Greet } from "../wailsjs/go/main/App"; +import { main } from "../wailsjs/go/models"; + +function generate() { + let person = new main.Person(); + person.name = "Peter"; + person.age = 27; + Greet(person).then((result) => { + console.log(result); + }); +} +``` + +生成的绑定和 TypeScript 模型的结合构成了一个强大的开发环境。 + +有关绑定的更多信息,请参见 [应用开发指南](guides/application-development.mdx) 的 [绑定方法](guides/application-development.mdx#绑定方法) 部分。 + +### 调用运行时方法 + +JavaScript 运行时位于`window.runtime`并包含许多方法来执行各种任务,例如发出事件或执行日志记录操作: + +```js title="mycode.js" +window.runtime.EventsEmit("my-event", 1); +``` + +更多关于 JS 运行时的细节可以在 [运行时参考](reference/runtime/intro) 中找到。 + +[^1]: 有一小部分库使用了 WebView 中不支持的功能。 对于这种情况,通常有替代方案和解决方法。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx new file mode 100644 index 00000000000..6637fd6641c --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx @@ -0,0 +1,75 @@ +--- +sidebar_position: 1 +--- + +# 简介 + +Wails 是一个可让您使用 Go 和 Web 技术编写桌面应用的项目。 + +将它看作为 Go 的快并且轻量的 Electron 替代品。 您可以使用 Go 的灵活性和强大功能,结合丰富的现代前端,轻松的构建应用程序。 + +### 功能 + +- 原生菜单、对话框、主题和半透明 +- Windows、macOS 和 linux 支持 +- 内置 Svelte、React 、Preact 、Vue、Lit 和 Vanilla JS 的模板 +- 从 JavaScript 轻松调用 Go 方法 +- 自动将 Go 结构体转换为 TypeScript 模块 +- Windows 上不需要 CGO 或外部 DLL +- 使用 [Vite ](https://vitejs.dev/) 的实时开发模式 +- 可以轻松创建、构建和打包应用的强大命令行工具 +- 丰富的 [运行时库](/docs/reference/runtime/intro) +- 使用 Wails 构建的应用程序兼容 Apple & Microsoft 商店 + +这是 [varly](https://varly.app) - 一个使用 Wails 编写的 MacOS 和 Windows 桌面应用。 它不仅看起来很强,它使用原生菜单和半透明 - 你希望从现代原生应用中得到的一切 + +```mdx-code-block +

+ + + +

+``` + +### 快速启动模板 + +Wails 带有许多预配置的模板,可让您快速启动和运行应用程序。 有以下框架的模板:Svelte、React、Vue、Preact、Lit 和 Vanilla。 每个模板都有 JavaScript 和 TypeScript 版本。 + +### 原生元素 + +Wails 使用专门构建的库来处理窗口、菜单、对话框等原生元素,因此您可以构建美观、功能丰富的桌面应用程序。 + +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. 在 Windows 上,是基于 Chromium 构建的新 Microsoft Webview2 库。 + +### Go 和 JavaScript 互操作 + +Wails 自动使您的 Go 方法可用于 JavaScript,因此您可以从前端按名称调用它们! 它甚至会生成 Go 方法使用的结构体的 TypeScript 版本,因此您可以在 Go 和 JavaScript 之间传递相同的数据结构。 + +### 运行时库 + +Wails 为 Go 和 JavaScript 提供了一个运行时库,它可以处理现代应用程序需要的很多东西,比如事件、日志记录、对话框等。 + +### 实时开发体验 + +#### 自动重新构建 + +当您在“开发”模式下运行您的应用程序时,Wails 会将您的应用程序构建为原生桌面应用程序,但会从磁盘读取您的资源。 它将检测您的 Go 代码的任何更改并自动重新构建和重新启动您的应用程序。 + +#### 自动重新加载 + +当检测到对您的应用程序资产的更改时,您正在运行的应用程序将“重新加载”,几乎立即反映您的更改 + +#### 在浏览器中开发您的应用程序 + +如果您更喜欢在浏览器中调试和开发,那么 Wails 可以满足您的需求。 正在运行的应用程序还有一个网络服务器,它将在连接到它的任何浏览器中运行您的应用程序。 当您的资源在磁盘上发生变化时,它会刷新。 + +### 可用于生产的原生二进制文件 + +当您准备好完成应用程序的最终构建时,CLI 会将其编译为单个可执行文件,并将所有资源打包到其中。 在 Windows 和 MacOS 上,可以创建用于分发的原生包。 使用打包工具后生成的资源(图标、info.plist、清单文件等)是您项目的一部分,可以自定义,让您完全控制应用程序的构建方式。 + +### 工具 + +Wails CLI 提供了一种简单的方法来生成、构建和打包您的应用程序。 它将完成创建图标的繁重工作,使用最佳设置编译您的应用程序,并提供可分发的、可用于生产的二进制文件。 可以从许多入门模板中进行选择,以快速启动和运行! diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/_category_.json new file mode 100644 index 00000000000..ebb337b8360 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Reference", + "position": 40 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx new file mode 100644 index 00000000000..d28d2732c30 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx @@ -0,0 +1,240 @@ +--- +sidebar_position: 2 +--- + +# 命令行 + +Wails CLI 有许多用于管理项目的命令。 所有命令都以此方式运行: + +`wails <命令> <标志>` + +## 初始化 + +`wails init` 用于生成项目。 + +| 标志 | 描述 | 默认 | +|:--------- |:---------------------------------------------- |:-------:| +| -n "项目名称" | 项目名称。 **强制必填** | | +| -d "项目目录" | 要创建的项目目录 | 项目名 | +| -g | 初始化 git 存储库 | | +| -l | 可用项目模板列表 | | +| -q | 禁止输出到控制台 | | +| -t "模板名称" | 要使用的项目模板。 这可能是默认模板的名称或托管在 github 上的远程模板的 URL 。 | vanilla | +| -ide | 生成 IDE 项目文件 | | +| -f | 强制构建应用 | false | + +示例: `wails init -n test -d mytestproject -g -ide vscode -q` + +这将在 "mytestproject" 目录生成一个名为 "test" 的项目,初始化 git,生成 vscode 项目文件并静默执行。 + +可以在 [此处](../guides/ides) 找到有关在 Wails 中使用 IDE 的更多信息。 + +### 远程模板 + +支持远程模板(托管在 GitHub )并且可以使用模板项目的 URL 进行安装。 + +示例: `wails init -n test -t https://github.com/leaanthony/testtemplate[@v1.0.0]` + +可以在 [此处](../community/templates.mdx) 找到社区维护的模板列表 + +:::warning 注意 + +**Wails 项目不维护也不对第 3 方模板负责** + +如果您不确定某个模板,请检查 `package.json` 和 `wails.json` 中安装的模块和运行的脚本。 + +::: + +## 构建 + +`wails build` 用于将您的项目编译为生产可用的二进制文件。 + +| 标志 | 描述 | 默认 | +|:--------------- |:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |:---------------------------------------------------------------------------------------------------------- | +| -clean | 清理 `build/bin` 目录 | | +| -compiler "编译器" | 使用不同的 go 编译器来构建,例如 go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. 允许在应用程序窗口中使用 devtools | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | 打印构建命令但不执行它 | | +| -f | 强制构建应用 | | +| -garbleargs | 传递给 garble 的参数 | `-literals -tiny -seed=random` | +| -ldflags "标志" | 传递给编译器的额外 ldflags | | +| -m | 编译前跳过 mod tidy | | +| -nopackage | 不打包应用程序 | | +| -nocolour | 在输出中禁用颜色 | | +| -nosyncgomod | 不同步 go.mod 中的 Wails 版本 | | +| -nsis | 为 Windows 生成 NSIS 安装程序 | | +| -o 文件名 | 输出文件名 | | +| -obfuscated | 使用 [garble](https://github.com/burrowers/garble) 混淆应用程序 | | +| -platform | 为指定的 [平台](../reference/cli#平台)(逗号分割)构建,例如: `windows/arm64`。 `windows/arm64`。 注意,如果不给出架构,则使用 `runtime.GOARCH`。 | 如果给定环境变量 platform = `GOOS` 否则等于 `runtime.GOOS`。
如果给定环境变量 arch = `GOARCH` 否则等于 `runtime.GOARCH`. | +| -race | 使用 Go 的竞态检测器构建 | | +| -s | 跳过前端构建 | | +| -skipbindings | 跳过 bindings 生成 | | +| -tags "额外标签" | 构建标签以传递给 Go 编译器。 必须引用。 空格或逗号(但不能同时使用)分隔 | | +| -trimpath | 从生成的可执行文件中删除所有文件系统路径。 | | +| -u | 更新项目的 `go.mod` 以使用与 CLI 相同版本的 Wails | | +| -upx | 使用 “upx” 压缩最终二进制文件 | | +| -upxflags | 传递给 upx 的标志 | | +| -v int | 详细级别 (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 安装策略:download,embed,browser,error. | download | +| -windowsconsole | 保留Windows构建控制台窗口 | | + +有关 `webview2` 标志的详细描述,请参阅 [Windows 系统指南](../guides/windows)。 + +如果您更喜欢使用标准 Go 工具进行构建,请参阅 [手动构建指南](../guides/manual-builds)。 + +示例: + +`wails build -clean -o myproject.exe` + +:::info + +在 Mac 上,应用程序将被绑定到 `Info.plist`,而不是 `Info.dev.plist`。 + +::: + +:::info 苹果芯片上的 UPX + +在苹果芯片上使用 UPX 相关的 [问题](https://github.com/upx/upx/issues/446)。 + +::: + +:::info Windows 上的 UPX + +一些防病毒软件供应商误将 `upx` 压缩的二进制文件标记为病毒,请查看相关 [问题](https://github.com/upx/upx/issues/437)。 + +::: + +### 平台 + +支持的平台有: + +| 平台 | 描述 | +|:---------------- |:--------------------------------------------- | +| darwin | MacOS + architecture of build machine | +| darwin/amd64 | MacOS 10.13+ AMD64 | +| darwin/arm64 | MacOS 11.0+ ARM64 | +| darwin/universal | MacOS AMD64+ARM64 universal application | +| windows | Windows 10/11 + architecture of build machine | +| windows/amd64 | Windows 10/11 AMD64 | +| windows/arm64 | Windows 10/11 ARM64 | +| linux | Linux + architecture of build machine | +| linux/amd64 | Linux AMD64 | +| linux/arm64 | Linux ARM64 | + +## 诊断检查 + +`wails doctor` 将运行诊断程序以确保您的系统已准备好进行开发。 + +示例: + +``` +Wails CLI v2.0.0-beta + +Scanning system - Please wait (this may take a long time)...Done. + +System +------ +OS: Windows 10 Pro +Version: 2009 (Build: 19043) +ID: 21H1 +Go Version: go1.18 +Platform: windows +Architecture: amd64 + +Dependency Package Name Status Version +---------- ------------ ------ ------- +WebView2 N/A Installed 93.0.961.52 +npm N/A Installed 6.14.15 +*upx N/A Installed upx 3.96 + +* - Optional Dependency + +Diagnosis +--------- +Your system is ready for Wails development! + +``` + +## 开发 + +`wails dev` 用于以 "实时开发" 模式运行您的应用。 这意味着: + +- 应用程序的 `go.mod` 将被更新为与 Wails CLI 相同的版本 +- 应用程序被编译并自动运行 +- 一个观察者被启动,如果它检测到您的 go 文件的变化,它将触发您的开发应用程序的重新构建 +- 启动一个网络服务器 `http://localhost:34115`,通过 http 为您的应用程序(不仅仅是前端)提供服务。 这允许您使用您喜欢的浏览器开发扩展 +- 所有应用程序资源都从磁盘加载。 如果它们被更改,应用程序将自动重新加载(而不是重新构建)。 所有连接的浏览器也将重新加载 +- 生成的 JS 模块提供以下内容: +- 带有自动生成的 JSDoc 的 Go 方法的 JavaScript 包装器,提供代码提示 +- 您的 Go 结构的 TypeScript 版本,可以构建并传递给您的 go 方法 +- 生成第二个 JS 模块,为运行时提供包装器 + TS 声明 +- 在 macOS 上,它将应用程序捆绑到一个 `.app` 文件中并运行它。 开发模式将使用 `build/darwin/Info.dev.plist` 。 + +| 标志 | 描述 | 默认 | +|:---------------------------- |:-------------------------------------------------------------------------------------------------------------------------------- |:----------------- | +| -appargs "参数" | 以 shell 样式传递给应用程序的参数 | | +| -assetdir "./path/to/assets" | 从给定目录提供资产,而不是使用提供的资产 FS | `wails.json` 中的值 | +| -browser | 在启动时打开浏览器到 `http://localhost:34115` | | +| -compiler "编译器" | 使用不同的 go 编译器来构建,例如 go1.15beta1 | go | +| -debounce | 检测到资产更改后等待重新加载的时间 | 100 (毫秒) | +| -devserver "host:port" | 将 wails 开发服务器绑定到的地址 | "localhost:34115" | +| -extensions | 触发重新构建的扩展(逗号分隔) | go | +| -forcebuild | 强制构建应用程序 | | +| -frontenddevserverurl "url" | 使用 3rd 方开发服务器 url 提供资产,例如:Vite | "" | +| -ldflags "标志" | 传递给编译器的额外 ldflags | | +| -loglevel "日志级别" | 要使用的日志级别 - Trace, Debug, Info, Warning, Error | Debug(调试) | +| -nocolour | 关闭彩色命令行输出 | false | +| -noreload | 资产更改时禁用自动重新加载 | | +| -nosyncgomod | 不同步 go.mod 中的 Wails 版本 | false | +| -race | 使用 Go 的竞态检测器构建 | false | +| -reloaddirs | 触发重新加载的附加目录(逗号分隔) | `wails.json` 中的值 | +| -s | 跳过前端构建 | false | +| -save | 将指定的 `assetdir`、 `reloaddirs`、 `wailsjsdir`、 `debounce` 、 `devserver` 和 `frontenddevserverurl` 标志的值保存到 `wails.json` 以成为后续调用的默认值。 | | +| -skipbindings | 跳过 bindings 生成 | | +| -tags "额外标签" | 传递给编译器的构建标签(引号和空格分隔) | | +| -v | 详细级别 (0 - silent, 1 - standard, 2 - verbose) | 1 | +| -wailsjsdir | 生成生成的Wails JS模块的目录 | `wails.json` 中的值 | + +示例: + +`wails dev -assetdir ./frontend/dist -wailsjsdir ./frontend/src -browser` + +此命令将执行以下操作: + +- 构建应用程序并运行它(更多细节在 [这里](../guides/manual-builds)) +- 在 `./frontend/src` 中生成 Wails JS 模块 +- 监听 `./frontend/dist` 中文件的更新并在更改时重新加载 +- 打开浏览器并连接到应用程序 + +有更多关于在现有框架脚本中使用此功能的 [信息](../guides/application-development.mdx#live-reloading)。 + +## 生成 + +### 模板 + +Wails 使用模板来生成项目。 `wails generate template` 命令有助于构建模板,以使它可以用于生成项目。 + +| 标志 | 描述 | +|:-------------- |:--------------- | +| -name | 模板名称(必填) | +| -frontend "路径" | 要在模板中使用的前端项目的路径 | + +有关创建模板的更多详细信息,请参阅 [模板指南](../guides/templates)。 + +### 模块 + +`wails generate module` 命令允许您为应用程序手动生成 `wailsjs` 目录。 + +## 更新 + +`wails update` 将更新 Wails CLI 的版本。 + +| 标志 | 描述 | +|:------------- |:----------- | +| -pre | 更新到最新的预发布版本 | +| -version "版本" | 安装指定版本的 CLI | + +## 版本 + +`wails version` 仅输出当前的 CLI 版本。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx new file mode 100644 index 00000000000..4e87c3f48f8 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx @@ -0,0 +1,230 @@ +--- +sidebar_position: 4 +--- + +# 菜单 + +可以将应用程序菜单添加到 Wails 项目。 可以通过定义一个 [菜单结构体](#菜单结构体) 并设置 [`菜单选项`](../reference/options#菜单) 或者通过调用运行时方法 [设置应用程序菜单](../reference/runtime/menu#设置应用程序菜单) 来将应用程序菜单添加到 Wails 项目。 + +如何创建菜单的示例: + +```go + + app := NewApp() + + AppMenu := menu.NewMenu() + FileMenu := AppMenu.AddSubmenu("File") + FileMenu.AddText("&Open", keys.CmdOrCtrl("o"), openFile) + FileMenu.AddSeparator() + FileMenu.AddText("Quit", keys.CmdOrCtrl("q"), func(_ *menu.CallbackData) { + runtime.Quit(app.ctx) + }) + + if runtime.GOOS == "darwin" { + AppMenu.Append(menu.EditMenu()) // on macos platform, we should append EditMenu to enable Cmd+C,Cmd+V,Cmd+Z... shortcut + } + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + Menu: AppMenu, // reference the menu above + Bind: []interface{}{ + app, + }, + ) + // ... +``` + +也可以通过更新菜单结构体并调用 [更新应用程序菜单](../reference/runtime/menu#更新应用程序菜单) 来动态更新菜单 。 + +上面的示例使用辅助方法,但是可以手动构建菜单结构。 + +## 菜单结构体 + +Menu 是 MenuItem 的集合: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Menu struct { + Items []*MenuItem +} +``` + +对于应用程序菜单,每个 MenuItem 代表一个菜单,例如“编辑”。 + +提供了一个简单的辅助方法来构建菜单: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func NewMenuFromItems(first *MenuItem, rest ...*MenuItem) *Menu +``` + +这使得代码的布局更像菜单的布局,而无需在创建菜单项后手动添加它们。 或者,您可以只创建菜单项并将它们手动添加到菜单中。 + +## 菜单项结构体 + +MenuItem 表示菜单中的一个项目。 + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +// MenuItem represents a menu item contained in a menu +type MenuItem struct { + Label string + Role Role + Accelerator *keys.Accelerator + Type Type + Disabled bool + Hidden bool + Checked bool + SubMenu *Menu + Click Callback +} +``` + +| 字段 | 类型 | 注解 | +| ----------- | ------------------------------------ | ---------------------------- | +| Label | string | 菜单文字 | +| Accelerator | [\*keys.Accelerator](#accelerator) | 此菜单项的键绑定 | +| 类型 | [类型](#type) | 菜单项的类型 | +| Disabled | bool | 禁用菜单项 | +| Hidden | bool | 隐藏此菜单项 | +| Checked | bool | 添加检查项目 (复选框和单选类型) | +| SubMenu | [\*Menu](#menu) | 设置子菜单 | +| Click | [Callback](#callback) | 单击菜单时的回调函数 | +| Role | string | 定义此菜单项的 [角色](#角色)。 暂时只支持 Mac | + +### 快捷键 + +加速器(有时称为键盘快捷键)定义了按键和菜单项之间的绑定。 Wails 将加速器定义为一个组合或键 + [修饰符](#修饰符)。 它们在 `"github.com/wailsapp/wails/v2/pkg/menu/keys"` 包中提供。 + +示例: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut := keys.CmdOrCtrl("o") +``` + +键是键盘上除了`+`的任何字符,它被定义为`加号`。 有些键不能表示为字符,因此可以使用一组命名字符: + +| | | | | +|:-----------:|:-----:|:---------:|:---------:| +| `backspace` | `f1` | `f16` | `f31` | +| `tab` | `f2` | `f17` | `f32` | +| `return` | `f3` | `f18` | `f33` | +| `enter` | `f4` | `f19` | `f34` | +| `escape` | `f5` | `f20` | `f35` | +| `left` | `f6` | `f21` | `numlock` | +| `right` | `f7` | `f22` | | +| `up` | `f8` | `f23` | | +| `down` | `f9` | `f24` | | +| `space` | `f10` | `f25` | | +| `delete` | `f11` | `f36` | | +| `home` | `f12` | `f37` | | +| `end` | `f13` | `f38` | | +| `page up` | `f14` | `page up` | | +| `page down` | `f15` | `f30` | | + +Wails 还支持使用与 Electron 相同的语法来解析加速器。 这对于将加速器存储在配置文件中很有用。 + +示例: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines cmd+o on Mac and ctrl-o on Window/Linux + myShortcut, err := keys.Parse("Ctrl+Option+A") +``` + +#### 修饰符 + +以下修饰符是可以与加速键结合使用的键: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +const ( + // CmdOrCtrlKey represents Command on Mac and Control on other platforms + CmdOrCtrlKey Modifier = "cmdorctrl" + // OptionOrAltKey represents Option on Mac and Alt on other platforms + OptionOrAltKey Modifier = "optionoralt" + // ShiftKey represents the shift key on all systems + ShiftKey Modifier = "shift" + // ControlKey represents the control key on all systems + ControlKey Modifier = "ctrl" +) +``` + +许多辅助方法可用于使用修饰符创建加速器: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" +func CmdOrCtrl(key string) *Accelerator +func OptionOrAlt(key string) *Accelerator +func Shift(key string) *Accelerator +func Control(key string) *Accelerator +``` + +可以使用 `keys.Combo(key string, modifier1 Modifier, modifier2 Modifier, rest ...Modifier)` 用以下方式组合修饰符 : + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu/keys" + // Defines "Ctrl+Option+A" on Mac and "Ctrl+Alt+A" on Window/Linux + myShortcut := keys.Combo("a", ControlKey, OptionOrAltKey) +``` + +### 类型 + +每个菜单项必须有一个类型,有 5 种类型可用: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +const ( + TextType Type = "Text" + SeparatorType Type = "Separator" + SubmenuType Type = "Submenu" + CheckboxType Type = "Checkbox" + RadioType Type = "Radio" +) +``` + +为方便起见,提供了帮助方法来快速创建菜单项: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func Text(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func Separator() *MenuItem +func Radio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func Checkbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func SubMenu(label string, menu *Menu) *Menu +``` + +您还可以使用“添加”助手直接在菜单上创建菜单项: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +func (m *Menu) AddText(label string, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSeparator() *MenuItem +func (m *Menu) AddRadio(label string, selected bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddCheckbox(label string, checked bool, accelerator *keys.Accelerator, click Callback) *MenuItem +func (m *Menu) AddSubMenu(label string, menu *Menu) *MenuI +``` + +关于单选组的说明:单选组被定义为在菜单中彼此相邻的多个单选菜单项。 这意味着您不需要将项目组合在一起,因为它是自动的。 但是,这也意味着您不能有 2 个彼此相邻的无线电组 - 它们之间必须有一个非无线电项目。 + +### 回调 + +每个菜单项都可能有一个回调,在单击该项时执行: + +```go title="Package: github.com/wailsapp/wails/v2/pkg/menu" +type Callback func(*CallbackData) + +type CallbackData struct { + MenuItem *MenuItem +} +``` + +该函数被赋予一个 `CallbackData` 结构,该结构指示哪个菜单项触发了回调。 这在使用可能共享回调的单选组时很有用。 + +### 角色 + +:::info 角色 + +目前仅 Mac 支持角色。 + +::: + +一个菜单项可能有一个角色,它本质上是一个预定义的菜单项。 我们目前支持以下角色: + +| 角色 | 描述 | +| ------------ | ---------------------------------------- | +| AppMenuRole | 标准的 Mac 应用程序菜单。 可以使用 `menu.AppMenu()` 创建 | +| EditMenuRole | 标准的 Mac 编辑菜单。 可以使用 `menu.EditMenu()` 创建 | diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx new file mode 100644 index 00000000000..d6c559d8d69 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx @@ -0,0 +1,869 @@ +--- +sidebar_position: 3 +--- + +# 参数选项 + +## 应用程序参数选项 + +该 `Options.App` 结构包含应用程序配置。 它被传递给 `wails.Run()` 方法: + +```go title="Example" +import ( + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/linux" + "github.com/wailsapp/wails/v2/pkg/options/mac" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +func main() { + + err := wails.Run(&options.App{ + Title: "Menus Demo", + Width: 800, + Height: 600, + DisableResize: false, + Fullscreen: false, + WindowStartState: options.Maximised, + Frameless: true, + MinWidth: 400, + MinHeight: 400, + MaxWidth: 1280, + MaxHeight: 1024, + StartHidden: false, + HideWindowOnClose: false, + BackgroundColour: &options.RGBA{R: 0, G: 0, B: 0, A: 255}, + AlwaysOnTop: false, + AssetServer: &assetserver.Options{ + Assets: assets, + Handler: assetsHandler, + Middleware: assetsMidldeware, + }, + Menu: app.applicationMenu(), + Logger: nil, + LogLevel: logger.DEBUG, + LogLevelProduction: logger.ERROR, + OnStartup: app.startup, + OnDomReady: app.domready, + OnShutdown: app.shutdown, + OnBeforeClose: app.beforeClose, + CSSDragProperty: "--wails-draggable", + CSSDragValue: "drag", + EnableDefaultContextMenu: false, + EnableFraudulentWebsiteDetection: false, + ZoomFactor: 1.0, + IsZoomControlEnabled: false, + Bind: []interface{}{ + app, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + Windows: &windows.Options{ + WebviewIsTransparent: false, + WindowIsTranslucent: false, + BackdropType: windows.Mica, + DisableWindowIcon: false, + DisableFramelessWindowDecorations: false, + WebviewUserDataPath: "", + WebviewBrowserPath: "", + Theme: windows.SystemDefault, + CustomTheme: &windows.ThemeSettings{ + DarkModeTitleBar: windows.RGB(20, 20, 20), + DarkModeTitleText: windows.RGB(200, 200, 200), + DarkModeBorder: windows.RGB(20, 0, 20), + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + }, + // User messages that can be customised + Messages *windows.Messages + // OnSuspend is called when Windows enters low power mode + OnSuspend func() + // OnResume is called when Windows resumes from low power mode + OnResume func(), + WebviewGpuDisabled: false, + }, + Mac: &mac.Options{ + TitleBar: &mac.TitleBar{ + TitlebarAppearsTransparent: true, + HideTitle: false, + HideTitleBar: false, + FullSizeContent: false, + UseToolbar: false, + HideToolbarSeparator: true, + }, + Appearance: mac.NSAppearanceNameDarkAqua, + WebviewIsTransparent: true, + WindowIsTranslucent: false, + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + Linux: &linux.Options{ + Icon: icon, + WindowIsTranslucent: false, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" + }, + Debug: options.Debug{ + OpenInspectorOnStartup: false, + }, + }) + + if err != nil { + log.Fatal(err) + } +} + +``` + +### 标题 + +窗口标题栏中显示的文本。 + +名称:Title
类型:`string` + +### 宽度 + +窗口的初始宽度。 + +名称:Width
类型:`int`
默认值:1024. + +### 高度 + +窗口的初始高度。 + +名称:Height
类型:`int`
默认值:768 + +### 禁用调整窗口尺寸 + +默认情况下,主窗口可调整大小。 将此设置为 `true` 将使其保持固定大小。 + +名称:DisableResize
类型:`bool` + +### 全屏 + +已弃用:请使用 [窗口启动状态](#窗口启动状态). + +### 窗口启动状态 + +定义窗口在启动时应如何呈现。 + +| 值 | Win | Mac | Lin | +| -------------- | --- | --- | --- | +| Fullscreen(全屏) | ✅ | ✅ | ✅ | +| Maximised(最大化) | ✅ | ✅ | ✅ | +| Minimised(最小化) | ✅ | ❌ | ✅ | + +名称:WindowStartState
类型:`options.WindowStartState` + +### 无边框 + +设置为`true`时,窗口将没有边框或标题栏。 另请参阅 [无边框窗口](../guides/frameless)。 + +名称:Frameless
类型:`bool` + +### 最小宽度 + +这将设置窗口的最小宽度。 如果给出的值 `Width` 小于这个值,窗口将被设置为 `MinWidth` 默认值。 + +名称:MinWidth
类型:`int` + +### 最小高度 + +这将设置窗口的最小高度。 如果给出的值 `Height` 小于这个值,窗口将被设置为 `MinHeight` 默认值。 + +名称:MinHeight
类型:`int` + +### 最大宽度 + +这将设置窗口的最大宽度。 如果给出的值 `Width` 大于这个值,窗口将被设置为 `MaxWidth` 默认值。 + +名称:MaxWidth
类型:`int` + +### 最大高度 + +这将设置窗口的最大高度。 如果给出的值 `Height` 大于这个值,窗口将被设置为 `MaxHeight` 默认值。 + +名称:MaxHeight
类型:`int` + +### 启动时隐藏窗口 + +设置为 `true` 时,应用程序将被隐藏,直到调用 [显示窗口](../reference/runtime/window#显示窗口)。 + +名称:StartHidden
类型:`bool` + +### 关闭时隐藏窗口 + +默认情况下,关闭窗口将关闭应用程序。 将此设置为 `true` 意味着关闭窗口将隐藏窗口。 + +隐藏窗口。 + +名称:HideWindowOnClose
类型:`bool` + +### 背景颜色 + +此值是窗口的默认背景颜色。 示例:options.NewRGBA(255,0,0,128) - 红色,透明度为 50% + +名称:BackgroundColour
类型:`*options.RGBA`
默认值:white + +### 窗口固定在最顶层 + +窗口在失去焦点时应保持在其他窗口之上。 + +名称:AlwaysOnTop
类型:`bool` + +### 资产 + +已弃用:请在 [AssetServer 特定选项](#资产服务器) 上使用资产。 + +### 资产处理程序 + +已弃用:请在 [AssetServer 特定选项](#资产服务器) 上使用资产处理程序。 + +### 资产服务器 + +这定义了资产服务器特定的选项。 它允许使用静态资产自定义资产服务器,使用 `http.Handler` 动态地提供资产或使用 `assetsserver.Middleware` 钩到请求链。 + +并非当前支持 `http.Request` 的所有功能,请参阅以下功能矩阵: + +| 功能 | Win | Mac | Lin | +| ----------------------- | --- | --- | ------ | +| GET | ✅ | ✅ | ✅ | +| POST | ✅ | ✅ | ✅ [^1] | +| PUT | ✅ | ✅ | ✅ [^1] | +| PATCH | ✅ | ✅ | ✅ [^1] | +| DELETE | ✅ | ✅ | ✅ [^1] | +| Request Headers | ✅ | ✅ | ✅ [^1] | +| Request Body | ✅ | ✅ | ✅ [^2] | +| Request Body Streaming | ✅ | ✅ | ✅ [^2] | +| Response StatusCodes | ✅ | ✅ | ✅ [^1] | +| Response Headers | ✅ | ✅ | ✅ [^1] | +| Response Body | ✅ | ✅ | ✅ | +| Response Body Streaming | ❌ | ✅ | ✅ | +| WebSockets | ❌ | ❌ | ❌ | +| HTTP Redirects 30x | ✅ | ❌ | ❌ | + +名称: AssetServer
类型: `*assetserver.Options` + +#### 资产 + +应用程序要使用的静态前端资产。 + +首先尝试从 `fs.FS` 提供 GET 请求。 如果 `fs.FS` 为该文件返回 `os.ErrNotExist`,则请求处理将回退到 [处理程序](#处理程序) 并尝试服务来自它的 GET 请求。 + +如果设置为 nil,则所有 GET 请求都将转发给 [处理程序](#处理程序)。 + +名称: Assets
类型: `fs.FS` + +#### 处理程序 + +资产处理程序是一个通用的 `http.Handler`,用于对无法找到的资产进行后备处理。 + +由于 `os.ErrNotExist`,对于每个无法从 [资产](#资产) 提供服务的 GET 请求,都会调用该处理程序。 此外,所有非 GET 请求将始终从此处理程序提供服务。 如果未定义,则调用处理程序的结果如下: + +- GET 请求: `http.StatusNotFound` +- 其他请求: `http.StatusMethodNotAllowed` + +注意:当与前端 DevServer 结合使用时,可能会有一些限制,例如。 Vite 在不包含文件扩展名的每个路径上提供 index.html。 + +名称:AssetsHandler
类型:`http.Handler` + +#### 中间件 + +中间件是一个 HTTP 中间件,它允许挂钩到资产服务器请求链。 它允许动态跳过默认请求处理程序,例如实现专门的路由等。 调用中间件来构建资产服务器使用的新 `http.Handler`,它还接收资产服务器使用的默认处理程序作为参数。 + +如果未定义,则执行默认的资产服务器请求链。 + +名称: Middleware
类型: `assetserver.Middleware` + +### 菜单 + +应用程序要使用的菜单。 [菜单参考](../reference/runtime/menu) 中有关菜单的更多详细信息。 + +:::note + +在 Mac 上,如果未指定菜单,将创建一个默认菜单。 + +::: + +名称:Menu
类型:`*menu.Menu` + +### 日志 + +应用程序要使用的记录器。 有关日志记录的更多详细信息,请参阅 [日志参考](../reference/runtime/log)。 + +名称:Logger
类型:`logger.Logger`
默认值:Logs to Stdout + +### 日志级别 + +默认日志级别。 有关日志记录的更多详细信息,请参阅 [日志参考](../reference/runtime/log)。 + +名称:LogLevel
类型:`logger.LogLevel`
默认值:开发模式为 `Info`, 生产模式为 `Error` + +### 生产日志级别 + +生产构建的默认日志级别。 有关日志记录的更多详细信息,请参阅 [日志参考](../reference/runtime/log)。 + +名称:LogLevelProduction
类型:`logger.LogLevel`
默认值:`Error` + +### 应用启动回调 + +此回调在前端创建之后调用,但在 `index.html` 加载之前调用。 它提供了应用程序上下文。 + +名称:OnStartup
类型:`func(ctx context.Context)` + +### 前端 Dom 加载完成回调 + +在前端加载完毕 `index.html` 及其资源后调用此回调。 它提供了应用程序上下文。 + +名称:OnDomReady
类型:`func(ctx context.Context)` + +### 应用退出回调 + +在前端被销毁之后,应用程序终止之前,调用此回调。 它提供了应用程序上下文。 + +名称:OnShutdown
类型:`func(ctx context.Context)` + +### 应用关闭前回调 + +如果设置了此回调,它将在通过单击窗口关闭按钮或调用`runtime.Quit`即将退出应用程序时被调用. 返回 `true` 将导致应用程序继续,`false` 将继续正常关闭。 返回 true 将导致应用程序继续,false 将继续正常关闭。 这有助于与用户确认他们希望退出程序。 + +示例: + +```go title=windowsapp.go +func (b *App) beforeClose(ctx context.Context) (prevent bool) { + dialog, err := runtime.MessageDialog(ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Quit?", + Message: "Are you sure you want to quit?", + }) + + if err != nil { + return false + } + return dialog != "Yes" +} +``` + +名称:OnBeforeClose
类型:`func(ctx context.Context) bool` + +### CSS 拖动属性 + +指示用于标识哪些元素可用于拖动窗口的 CSS 属性。 默认值:`--wails-draggable` + +名称:CSSDragProperty
类型:`string` + +### CSS 拖动值 + +指示 `CSSDragProperty` 样式应该具有什么值才能拖动窗口。 默认值:`drag` + +名称:CSSDragValue
类型:`string` + +### EnableDefaultContextMenu + +EnableDefaultContextMenu enables the browser's default context-menu in production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : + +| CSS Style | Behavior | +| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `--default-contextmenu: auto;` | (**default**) will show the default context menu only if :
contentEditable is true OR text has been selected OR element is input or textarea | +| `--default-contextmenu: show;` | will always show the default context menu | +| `--default-contextmenu: hide;` | will always hide the default context menu | + +This rule is inherited like any normal CSS rule, so nesting works as expected. + +:::note +This filtering functionality is only enabled in production, so in development and in debug build, the full context-menu is always available everywhere. +::: + +:::warning +This filtering functionality is NOT a security measure, the developer should expect that the full context-menu could be leaked anytime which could contain commands like (Download image, Reload, Save webpage), if this is a concern, the developer SHOULD NOT enable the default context-menu. +::: + + +Name: EnableDefaultContextMenu
Type: `bool` + +### 启用欺诈网站检测 + +EnableFraudulentWebsiteDetection 启用针对欺诈内容(例如恶意软件或网络钓鱼尝试)的扫描服务。 这些服务可能会从你的应用中发送信息,比如导航到苹果和微软的云服务的url和其他内容。 + +名称:EnableFraudulentWebsiteDetection
类型:`bool` + +### 缩放比例 + +名称:ZoomFactor
类型:`float64` + +这定义了 WebView2 的缩放比例。 这是匹配 Edge 用户激活放大或缩小的选项 + +### 启用缩放比例 + +名称:IsZoomControlEnabled
类型:`bool` + +这将允许用户更改缩放比例。 请注意,可以在选项中设置缩放比例,但不允许在运行时更改它。 适用于屏幕固定的或类似的应用程序。 + +### 绑定 + +定义需要绑定到前端的方法的结构实例切片。 + +名称:Bind
类型:`[]interface{}` + +### ErrorFormatter + +A function that determines how errors are formatted when returned by a JS-to-Go method call. The returned value will be marshalled as JSON. + +Name: ErrorFormatter
Type: `func (error) any` + +### Windows + +这定义了 [Windows 特定的选项](#windows)。 + +名称:Windows
类型:`*windows.Options` + +#### Webview 透明 + +当使用 `alpha` 值 `0` 时,将此设置为 true 将使 webview 背景透明。 这意味着如果您在 CSS 中使用 `rgba(0,0,0,0)` 作为 `background-color`,则主机窗口将显示出来。 通常与 [窗口半透明](#窗口半透明) 结合使用以制作看起来冷冰冰的应用程序。 + +名称:WebviewIsTransparent
类型:`bool` + +#### 窗口半透明 + +将此设置为 `true` 将使窗口半透明。 通常与 [Webview 透明](#webview-透明) 结合使用。 + +对于 build 22621 之前的 Windows 11 版本,将使用 [BlurBehind](https://learn.microsoft.com/en-us/windows/win32/dwm/blur-ovw) 方法来实现半透明,这可能会很慢。 对于构建 build 22621 之后的 Windows 11 版本,这将启用速度更快的新半透明类型。 默认情况下,使用的半透明类型将由 Windows 确定。 要对此进行配置,请使用 [背景类型](#背景类型) 选项。 + +名称:WindowIsTranslucent
类型:`bool` + +#### 背景类型 + +:::note + +需要 Windows 11 build 22621 或更高版本。 + +::: + +设置窗口的半透明类型。 这仅在 [窗口半透明](#窗口半透明) 设置为 `true` 时适用。 + +名称:BackdropType
类型:`windows.BackdropType` + +值可以是以下之一: + +| 值 | 描述 | +| ------- | -------------------------------------------------------------------------------- | +| Auto | 让 Windows 决定使用哪个背景 | +| None | 不要使用半透明 | +| Acrylic | 使用 [亚克力](https://learn.microsoft.com/en-us/windows/apps/design/style/acrylic) 效果 | +| Mica | 使用 [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) 效果 | +| Tabbed | 使用 Tabbed。 这是一个类似于 Mica 的背景。 | + +#### 禁用窗口图标 + +将此设置为 `true` 将删除标题栏左上角的图标。 + +名称:DisableWindowIcon
类型:`bool` + +#### 禁用无边框窗口装饰 + +将此设置为 `true` 将移除 [无边框](#无边框) 模式下的窗口装饰。 这意味着将不会有`Aero 阴影` 和 `圆角`显示在窗口上。 请注意,'圆角' 只在 Windows 11 上支持。 + +名称:DisableFramelessWindowDecorations
类型:`bool` + +#### Webview 用户数据路径 + +这定义了 WebView2 存储用户数据的路径。 如果为空将使用 `%APPDATA%\[BinaryName.exe]`。 + +名称:WebviewUserDataPath
类型:`string` + +#### Webview 浏览器路径 + +这定义了带有 WebView2 可执行文件和库的目录的路径 如果为空,则使用系统中安装的 webview2 + +有关固定版本运行时分发的重要信息: + +- [如何获取和提取运行时](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#details-about-the-fixed-version-runtime-distribution-mode) +- [固定版本的已知问题](https://docs.microsoft.com/en-us/microsoft-edge/webview2/concepts/distribution#known-issues-for-fixed-version) +- [WebView2 Runtime 固定版本的路径不应包含 \Edge\Application](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/webview2-idl?view=webview2-1.0.1245.22#createcorewebview2environmentwithoptions) + +名称:WebviewBrowserPath
类型:`string` + +#### 主题 + +最低 Windows 版本:Windows 10 2004/20H1 + +这定义了应用程序应该使用的主题: + +| 值 | 描述 | +| ------------- | -------------------------------------------- | +| SystemDefault | _默认_。 主题将基于系统默认值。 如果用户更改了他们的主题,应用程序将更新以使用新设置 | +| Dark | 该应用程序将只使用深色主题 | +| Light | 该应用程序将专门使用浅色主题 | + +名称:Theme
类型:`windows.Theme` + +#### 自定义主题 + +:::note + +最低 Windows 版本:Windows 10/11 2009/21H2 Build 22000 + +::: + +允许您为浅色和深色模式以及窗口处于活动或非活动状态的 TitleBar、TitleText 和 Border 指定自定义颜色。 + +名称:CustomTheme
类型:`windows.CustomTheme` + +##### 自定义主题类型 + +CustomTheme 结构体使用 `int32` 指定颜色值。 它们采用标准(!)Windows 格式:`0x00BBGGAA`。 These are in the standard(!) Windows format of: `0x00BBGGAA`. 提供了一个辅助函数来将 RGB 转换为这种格式:`windows.RGB(r,g,b uint8)`。 + +注意:任何未提供的值都将默认为黑色。 + +```go +type ThemeSettings struct { + DarkModeTitleBar int32 + DarkModeTitleBarInactive int32 + DarkModeTitleText int32 + DarkModeTitleTextInactive int32 + DarkModeBorder int32 + DarkModeBorderInactive int32 + LightModeTitleBar int32 + LightModeTitleBarInactive int32 + LightModeTitleText int32 + LightModeTitleTextInactive int32 + LightModeBorder int32 + LightModeBorderInactive int32 +} +``` + +示例: + +```go + CustomTheme: &windows.ThemeSettings{ + // Theme to use when window is active + DarkModeTitleBar: windows.RGB(255, 0, 0), // Red + DarkModeTitleText: windows.RGB(0, 255, 0), // Green + DarkModeBorder: windows.RGB(0, 0, 255), // Blue + LightModeTitleBar: windows.RGB(200, 200, 200), + LightModeTitleText: windows.RGB(20, 20, 20), + LightModeBorder: windows.RGB(200, 200, 200), + // Theme to use when window is inactive + DarkModeTitleBarInactive: windows.RGB(128, 0, 0), + DarkModeTitleTextInactive: windows.RGB(0, 128, 0), + DarkModeBorderInactive: windows.RGB(0, 0, 128), + LightModeTitleBarInactive: windows.RGB(100, 100, 100), + LightModeTitleTextInactive: windows.RGB(10, 10, 10), + LightModeBorderInactive: windows.RGB(100, 100, 100), + }, +``` + +#### 消息 + +一个如果找不到有效的 webview2 运行时,webview2 安装程序所使用的字符串结构。 + +名称:Messages
类型:`*windows.Messages` + +您可以选择支持的任意语言定制此选项。 + +#### 重置尺寸防抖间隔 + +ResizeDebounceMS 是调整窗口大小时去抖动 webview2 重绘的时间量。 默认值 (0) 将尽可能快地执行重绘。 + +名称:ResizeDebounceMS
类型:`uint16` + +#### 待机回调 + +如果设置,当 Windows 启动切换到低功耗模式(挂起/休眠)时将调用此函数 + +名称:OnSuspend
类型:`func()` + +#### 恢复回调 + +如果设置,当 Windows 从低功耗模式(挂起/休眠)恢复时将调用此函数 + +名称:OnResume
类型:`func()` + +#### 禁用 Webview GPU 硬件加速 + +设置为 `true` 将禁用 webview 的 GPU 硬件加速。 + +名称:WebviewGpuIsDisabled
类型:`bool` + +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
Type: `bool` + +### Mac + +这定义了 [Mac 特定的选项](#mac)。 + +名称:Mac
类型:`*mac.Options` + +#### 标题栏 + +TitleBar 结构提供了配置标题栏外观的能力。 + +名称:TitleBar
类型:[`*mac.TitleBar`](#标题栏结构体) + +##### 标题栏结构体 + +可以使用 TitleBar 选项自定义应用程序的标题栏: + +```go +type TitleBar struct { + TitlebarAppearsTransparent bool + HideTitle bool + HideTitleBar bool + FullSizeContent bool + UseToolbar bool + HideToolbarSeparator bool +} +``` + +| 设置 | 描述 | +| -------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------- | +| TitlebarAppearsTransparent | 使标题栏透明。 这具有隐藏标题栏和内容填充窗口的效果。 [苹果文档](https://developer.apple.com/documentation/appkit/nswindow/1419167-titlebarappearstransparent?language=objc) | +| HideTitle | 隐藏窗口的标题。 [苹果文档](https://developer.apple.com/documentation/appkit/nswindowtitlevisibility?language=objc) | +| HideTitleBar | 从 style mask 中删除 [NSWindowStyleMaskTitled](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemasktitled/) | +| FullSizeContent | 使 webview 填满整个窗口。 [苹果文档](https://developer.apple.com/documentation/appkit/nswindowstylemask/nswindowstylemaskfullsizecontentview) | +| UseToolbar | 向窗口添加默认工具栏。 [苹果文档](https://developer.apple.com/documentation/appkit/nstoolbar?language=objc) | +| HideToolbarSeparator | 删除工具栏下方的线条。 [苹果文档](https://developer.apple.com/documentation/appkit/nstoolbar/1516954-showsbaselineseparator?language=objc) | + +预配置的标题栏设置可用: + +| 设置 | 示例 | +| --------------------------- | ---------------------------------------------- | +| `mac.TitleBarDefault()` | ![](/img/reference/titlebar-default.webp) | +| `mac.TitleBarHidden()` | ![](/img/reference/titlebar-hidden.webp) | +| `mac.TitleBarHiddenInset()` | ![](/img/reference/titlebar-hidden-inset.webp) | + +示例: + +```go +Mac: &mac.Options{ + TitleBar: mac.TitleBarHiddenInset(), +} +``` + +单击 [此处](https://github.com/lukakerr/NSWindowStyles) 获取有关自定义标题栏的一些灵感。 + +#### 外观 + +Appearance 用于根据 Apple 的 [NSAppearance](https://developer.apple.com/documentation/appkit/nsappearancename?language=objc) 名称设置您的应用程序的样式。 + +名称:Appearance
类型:[`mac.AppearanceType`](#外观类型) + +##### 外观类型 + +您可以指定应用程序的 [外观](https://developer.apple.com/documentation/appkit/nsappearance?language=objc)。 + +| 值 | 描述 | +| ----------------------------------------------------- | --------------- | +| DefaultAppearance | 使用默认系统值 | +| NSAppearanceNameAqua | 标准日间系统外观 | +| NSAppearanceNameDarkAqua | 标准黑夜系统外观 | +| NSAppearanceNameVibrantLight | 轻盈灵动的外观 | +| NSAppearanceNameAccessibilityHighContrastAqua | 标准白天系统外观的高对比度版本 | +| NSAppearanceNameAccessibilityHighContrastDarkAqua | 标准黑夜系统外观的高对比度版本 | +| NSAppearanceNameAccessibilityHighContrastVibrantLight | 轻盈灵动外观的高对比度版本 | +| NSAppearanceNameAccessibilityHighContrastVibrantDark | 深色活力外观的高对比度版本 | + +示例: + +```go +Mac: &mac.Options{ + Appearance: mac.NSAppearanceNameDarkAqua, +} +``` + +#### Webview 透明 + +当使用 `alpha` 值 `0` 时,将此设置为 true 将使 webview 背景透明。 这意味着如果您在 CSS 中使用 `rgba(0,0,0,0)` 作为 `background-color`,则主机窗口将显示出来。 通常与 [窗口半透明](#窗口半透明-1) 结合使用以制作看起来冷冰冰的应用程序。 + +名称:WebviewIsTransparent
类型:`bool` + +#### 窗口半透明 + +将此设置为 `true` 将使窗口半透明。 通常与[Webview 透明](#webview-透明) 结合使用以制作冰霜效果的应用程序。 + +名称:WindowIsTranslucent
类型:`bool` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| 设置 | 描述 | +| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +示例: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + +#### 关于 + +此配置允许您在“AppMenu”角色创建的应用程序菜单中设置“关于”菜单项的标题、消息和图标。 + +名称:About
类型:[`*mac.AboutInfo`](#关于结构体) + +##### 关于结构体 + +```go + +type AboutInfo struct { + Title string + Message string + Icon []byte +} +``` + +如果提供了这些设置,“关于”菜单项将出现在应用程序菜单中(使用`AppMenu` role 时)。 建议这样配置: + +```go +//go:embed build/appicon.png +var icon []byte + +func main() { + err := wails.Run(&options.App{ + ... + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) + Mac: &mac.Options{ + About: &mac.AboutInfo{ + Title: "My Application", + Message: "© 2021 Me", + Icon: icon, + }, + }, + }) +``` + +“关于”菜单项将出现在应用程序菜单中: + +```mdx-code-block +
+ +
+
+``` + +单击后,将打开一个关于消息框: + +```mdx-code-block +
+ +
+
+``` + +### Linux + +这定义了 [Linux 特定的选项](#linux)。 + +名称:Linux
类型:`*linux.Options` + +#### 图标 + +设置代表窗口的图标。 当窗口最小化(也称为图标化)时使用此图标。 + +名称:Icon
类型:`[]byte` + +一些窗口管理器或桌面环境也可能将其放置在窗口框架中,或在其他上下文中显示。 在其他情况下,根本不使用该图标,因此您的预计情况可能会有所不同。 + +注意:Wayland 上的 Gnome 至少不显示此图标。 要在那里有一个应用程序图标,必须使用一个`.desktop`文件。 在 KDE 上它应该可以工作。 + +图标应该以自然绘制的任何尺寸提供;也就是说,在传递图像之前不要缩放图像。 缩放将延迟到当所需的最终尺寸已知的最后一刻,以获得最佳质量。 + +#### 窗口半透明 + +将此设置为 `true` 将使窗口半透明。 某些窗口管理员可能忽略它,或导致黑窗口。 + +名称:WindowIsTranslucent
类型:`bool` + +#### Webview GPU 策略 + +该选项用于决定 webview 的硬件加速策略。 + +名称:WebviewGpuPolicy
类型:[`options.WebviewGpuPolicy`](#webviewgpupolicy-type)
默认:`WebviewGpuPolicyAlways` + +##### Webview GPU 策略类型 + +| 值 | 描述 | +| ------------------------ | --------------------- | +| WebviewGpuPolicyAlways | 始终启用硬件加速 | +| WebviewGpuPolicyOnDemand | 根据 Web 内容的请求启用/禁用硬件加速 | +| WebviewGpuPolicyNever | 硬件加速始终处于禁用状态 | + +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` property differs form the executable's filename. + +Name: ProgramName
Type: string
+ +### 调试 + +这定义了用于调试构建的 [调试特定选项](#调试)。 + +名称: Debug
类型: `options.Debug` + +#### 启动时打开检查器 + +设置为 `true` 将在应用程序启动时打开 Web 检查器。 + +名称: OpenInspectorOnStartup
类型: `bool` + +[^1]: 这需要 WebKit2GTK 2.36+ 支持,并且您的应用程序需要使用构建标签 `webkit2_36` 来构建以激活对此功能的支持。 这也会将您应用程序的 WebKit2GTK 最低要求提高到 2.36。 +[^2]: 这需要 WebKit2GTK 2.40+ 支持,并且您的应用程序需要使用构建标签 `webkit2_40` 来构建以激活对此功能的支持。 这也将您的应用程序的 WebKit2GTK 最低要求提高到 2.40。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx new file mode 100644 index 00000000000..dc2a96b8a19 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx @@ -0,0 +1,104 @@ +--- +sidebar_position: 5 +--- + +# 项目配置 + +项目配置在项目目录中的 `wails.json` 文件中。 配置的结构是: + +```json5 +{ + // 项目配置版本 + "version": "", + // 项目名称 + "name": "", + // 包含编译资产的目录的相对路径,这通常是推断的并且可以留空 + "assetdir": "", + // 触发重新加载的附加目录(逗号分隔),这仅用于某些高级资产配置 + "reloaddirs": "", + // 构建文件所在的目录。默认为“build” + "build:dir": "", + // 前端目录的相对路径。默认为“frontend” + "frontend:dir": "", + // 安装 Node 依赖的命令,在前端目录运行 - 通常是`npm install` + "frontend:install": "", + // 构建资产的命令,在前端目录中运行 - 通常是 `npm run build` + "frontend:build": "", + // 此命令已被 frontend:dev:build 取代。如果未指定 frontend:dev:build 将回退到此命令。 + // 如果此命令也未指定,将回退到 frontend:build + "frontend:dev": "", + // 此命令是 frontend:build 的 dev 等价物。 + // 如果未指定回退到 frontend:dev + "frontend:dev:build": "", + // 此命令是 frontend:install 的 dev 等价物。如果未指定回退到 frontend:install + "frontend:dev:install": "", + // 此命令在 `wails dev`上的单独进程中运行。用于第 3 方观察者或启动 3d 方开发服务器 + "frontend:dev:watcher": "", + // 用于服务资产的第 3 方开发服务器的 URL,比如 Vite。 + // 如果设置为 'auto' 那么 devServerUrl 将从 Vite 输出中推断出来 + "frontend:dev:serverUrl": "", + // 创建自动生成的 JS 模块的目录的相对路径 + "wailsjsdir": "", + // 二进制文件的名称 + "outputfilename": "", + // 开发服务器在检测到资产更改时等待重新加载的默认时间 + "debounceMS": 100, + // 将 wails 开发服务器绑定到的地址。默认为:localhost:34115 + "devServer": "", + // 在开发模式下以 shell 样式传递给应用程序的参数 + "appargs": "", + // 定义是否应该运行构建 Hooks,尽管它们是为主机操作系统以外的操作系统定义的。 + "runNonNativeBuildHooks": false, + // 构建前 Hooks + "preBuildHooks": { + // 在构建指定的 GOOS/GOARCH 之前将执行的命令:${platform} 被替换为“GOOS/GOARCH”。 + // “GOOS/GOARCH” hook 在“GOOS/*”和“*/*” hook 之前执行。 + "GOOS/GOARCH": "", + // 在指定 GOOS 的构建之前将执行的命令:${platform} 被替换为“GOOS/GOARCH”。 + // “GOOS/*” hook 在“*/*” hook 之前执行。 + "GOOS/*": "", + // 将在每次构建之前执行的命令:${platform} 替换为“GOOS/GOARCH”。 + "*/*": "" + }, + // 构建后 Hooks + "postBuildHooks": { + // 在构建指定的 GOOS/GOARCH 之后将执行的命令:${platform} 替换为“GOOS/GOARCH”, + // ${bin} 替换为已编译二进制文件的路径。 “GOOS/GOARCH” hook 在“GOOS/*”和“*/*” hook 之前执行。 + "GOOS/GOARCH": "", + // 在构建指定的 GOOS 之后将执行的命令:${platform} 替换为“GOOS/GOARCH”, + // ${bin} 替换为已编译二进制文件的路径。 “GOOS/*” hook 在“*/*” hook 之前执行。 + "GOOS/*": "", + // 每次构建后将执行的命令:${platform} 替换为“GOOS/GOARCH”, + // ${bin} 替换为已编译二进制文件的路径。 + "*/*": "" + }, + // 用于填充 manifest 和版本信息的数据。 + "info": { + // 公司名称。 默认值:[项目名] + "companyName": "", + // 产品名称。 默认值:[项目名] + "productName": "", + // 产品版本。默认值:'1.0.0' + "productVersion": "", + // 产品的版权。默认值:'Copyright.........' + "copyright": "", + // 该应用程序的简短评论。默认值:'Built using Wails (https://wails.app)' + "comments": "" + }, + // 'multiple': 每个架构一个安装程序。 + // 'single': 适用于正在构建的所有体系结构的单一通用安装程序。 + // 默认值:'multiple' + "nsisType": "", + // 应用程序是否应该被混淆。默认值:false + "obfuscated": "", + // 使用 obfuscated 标志时传递给乱码命令的参数 + "garbleargs": "" +} + +``` + +该文件将在运行 `wails build` 或 `wails dev` 时,由 Wails CLI 读取。 + +`wails build/dev` 命令中的 `assetdir`、`reloaddirs`、`wailsjsdir`、`debounceMS`、`devserver` 和 `frontenddevserverurl` 标志将覆盖项目配置并作为后续运行的默认值。 + +此文件的 `JSON Schema` 位于 [此处](https://wails.io/schemas/config.v2.json)。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/_category_.json new file mode 100644 index 00000000000..ac6d5548816 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Runtime", + "position": 1 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx new file mode 100644 index 00000000000..0829be25f70 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx @@ -0,0 +1,13 @@ +--- +sidebar_position: 7 +--- + +# Browser 浏览器 + +这些方法与系统浏览器相关。 + +### BrowserOpenURL 浏览器打开 URL + +使用系统默认浏览器打开给定的 URL。 + +Go: `BrowserOpenURL(ctx context.Context, url string)`
JS: `BrowserOpenURL(url string)` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx new file mode 100644 index 00000000000..845dbc9f612 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx @@ -0,0 +1,23 @@ +--- +sidebar_position: 8 +--- + +# Clipboard 剪贴板 + +运行时的这一部分提供了对操作系统剪贴板的访问。
当前实现仅处理文本。 + +### ClipboardGetText 剪贴板获取文本 + +从剪切板读取当前存储的文本。 + +Go: `ClipboardGetText(ctx context.Context) (string, error)`
返回: 一个字符串(如果剪贴板为空,将返回一个空字符串)或一个错误。 + +JS: `ClipboardGetText(): Promise`
返回: 带有字符串结果的 Promise(如果剪贴板为空,将返回空字符串)。 + +### ClipboardSetText 剪贴板设置文本 + +将文本写入剪切板。 + +Go: `ClipboardSetText(ctx context.Context, text string) error`
返回: 如果存在错误,则会出现错误。 + +JS: `ClipboardSetText(text: string): Promise`
返回值: 一个Promise,如果文本成功地设置在剪贴板上,结果为 true,否则为 false。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx new file mode 100644 index 00000000000..7fd7f9640ae --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx @@ -0,0 +1,302 @@ +--- +sidebar_position: 5 +--- + +# Dialog 对话框 + +运行时的这一部分提供对原生对话框的调用,例如文件选择器和消息框。 + +:::info JavaScript + +JS 运行时当前不支持对话框。 + +::: + +### OpenDirectoryDialog 打开选择目录对话框 + +打开一个对话框,提示用户选择目录。 可以使用 [ 打开选择文件对话框参数选项](#打开选择文件对话框参数选项) 进行自定义。 + +Go: `OpenDirectoryDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +返回值: 所选目录(如果用户取消则为空白)或错误 + +### OpenFileDialog 打开选择文件对话框 + +打开一个对话框,提示用户选择文件。 可以使用 [ 打开选择文件对话框参数选项](#打开选择文件对话框参数选项) 进行自定义。 + +Go: `OpenFileDialog(ctx context.Context, dialogOptions OpenDialogOptions) (string, error)` + +返回值: 所选文件(如果用户取消则为空白)或错误 + +### OpenMultipleFilesDialog 打开选择多文件对话框 + +打开一个对话框,提示用户选择多个文件。 可以使用 [ 打开选择文件对话框参数选项](#打开选择文件对话框参数选项) 进行自定义。 + +Go: `OpenMultipleFilesDialog(ctx context.Context, dialogOptions OpenDialogOptions) ([]string, error)` + +返回值: 选定的文件(如果用户取消则为 nil)或错误 + +### SaveFileDialog 保存文件对话框 + +打开一个对话框,提示用户选择文件名以进行保存。 可以使用 [保存文件对话框参数选项](#保存文件对话框参数选项) 自定义。 + +Go: `SaveFileDialog(ctx context.Context, dialogOptions SaveDialogOptions) (string, error)` + +返回值: 所选文件(如果用户取消则为空白)或错误 + +### MessageDialog 消息对话框 + +使用消息对话框显示消息。 可以使用 [消息对话框参数选项](#消息对话框参数选项) 进行自定义。 + +Go: `MessageDialog(ctx context.Context, dialogOptions MessageDialogOptions) (string, error)` + +返回值: 所选按钮的文本或错误 + +## 参数选项 + +### 打开选择文件对话框参数选项 + +```go +type OpenDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + ResolvesAliases bool + TreatPackagesAsDirectories bool +} +``` + +| 字段 | 描述 | Win | Mac | Lin | +| -------------------------- | ------------------- | --- | --- | --- | +| DefaultDirectory | 对话框打开时显示的目录 | ✅ | ✅ | ✅ | +| DefaultFilename | 默认文件名 | ✅ | ✅ | ✅ | +| Title | 对话框的标题 | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | 文件过滤器列表 | ✅ | ✅ | ✅ | +| ShowHiddenFiles | 显示系统隐藏的文件 | | ✅ | ✅ | +| CanCreateDirectories | 允许用户创建目录 | | ✅ | | +| ResolvesAliases | 如果为 true,则返回文件而不是别名 | | ✅ | | +| TreatPackagesAsDirectories | 允许导航到包 | | ✅ | | + +### 保存文件对话框参数选项 + +```go +type SaveDialogOptions struct { + DefaultDirectory string + DefaultFilename string + Title string + Filters []FileFilter + ShowHiddenFiles bool + CanCreateDirectories bool + TreatPackagesAsDirectories bool +} +``` + +| 字段 | 描述 | Win | Mac | Lin | +| -------------------------- | ----------- | --- | --- | --- | +| DefaultDirectory | 对话框打开时显示的目录 | ✅ | ✅ | ✅ | +| DefaultFilename | 默认文件名 | ✅ | ✅ | ✅ | +| Title | 对话框的标题 | ✅ | ✅ | ✅ | +| [Filters](#filefilter) | 文件过滤器列表 | ✅ | ✅ | ✅ | +| ShowHiddenFiles | 显示系统隐藏的文件 | | ✅ | ✅ | +| CanCreateDirectories | 允许用户创建目录 | | ✅ | | +| TreatPackagesAsDirectories | 允许导航到包 | | ✅ | | + +### 消息对话框参数选项 + +```go +type MessageDialogOptions struct { + Type DialogType + Title string + Message string + Buttons []string + DefaultButton string + CancelButton string +} +``` + +| 字段 | 描述 | Win | Mac | Lin | +| ------------- | ------------------------------ | -------------- | --- | --- | +| 类型 | 消息对话框的类型,例如问题、信息... | ✅ | ✅ | ✅ | +| Title | 对话框的标题 | ✅ | ✅ | ✅ | +| Message | 向用户显示的消息 | ✅ | ✅ | ✅ | +| Buttons | 按钮标题列表 | | ✅ | | +| DefaultButton | 带有此文本的按钮应被视为默认按钮。 必定 `return`。 | ✅[*](#windows) | ✅ | | +| CancelButton | 带有此文本的按钮应被视为取消。 必定 `escape` | | ✅ | | + +#### Windows + +Windows 具有标准对话框类型,其中的按钮不可自定义。 返回的值将是以下之一:"Ok"、"Cancel"、"Abort"、"Retry"、"Ignore"、"Yes"、"No"、"Try Again"或"Continue"。 + +对于问题对话框,默认按钮是 “是”,取消按钮是 “否”。 可以通过将 `默认按钮` 值设置为 `"否"` 来改变这一点。 + +示例: +```go + result, err := runtime.MessageDialog(a.ctx, runtime.MessageDialogOptions{ + Type: runtime.QuestionDialog, + Title: "Question", + Message: "Do you want to continue?", + DefaultButton: "No", + }) +``` + +#### Linux + +Linux 有标准的对话框类型,其中的按钮是不可定制的。 返回的值将是以下之一:“Ok”、“Cancel”、“Yes”、“No” + +#### Mac + +Mac 上的消息对话框最多可以指定 4 个按钮。 如果没有 `DefaultButton` 或 `CancelButton` 给出,第一个按钮被认为是默认的并绑定到 `return` 键。 + +对于以下代码: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, +}) +``` + +第一个按钮显示为默认值: + +```mdx-code-block +
+ +
+
+``` + +如果我们指定 `DefaultButton` 为“two”: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", +}) +``` + +第二个按钮显示为默认值。 当 `return` 被按下时,则返回数值“two”。 + +```mdx-code-block +
+ +
+
+``` + +如果我们现在指定`CancelButton`为“three”: + +```go +selection, err := runtime.MessageDialog(b.ctx, runtime.MessageDialogOptions{ + Title: "It's your turn!", + Message: "Select a number", + Buttons: []string{"one", "two", "three", "four"}, + DefaultButton: "two", + CancelButton: "three", +}) +``` + +带有“three”的按钮显示在对话框的底部。 当 `escape` 被按下时,则返回值“three”: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### 对话框类型 + +```go +const ( + InfoDialog DialogType = "info" + WarningDialog DialogType = "warning" + ErrorDialog DialogType = "error" + QuestionDialog DialogType = "question" + ) +``` + +### 文件过滤 + +```go +type FileFilter struct { + DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)" + Pattern string // semi-colon separated list of extensions, EG: "*.jpg;*.png" +} +``` + +#### Windows + +Windows 允许您在对话框中使用多个文件过滤器。 每个 FileFilter 将在对话框中显示为一个单独的条目: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Linux + +Linux 允许您在对话框中使用多个文件过滤器。 每个 FileFilter 将在对话框中显示为一个单独的条目: + +```mdx-code-block +
+ +
+
+
+
+``` + +#### Mac + +Mac 对话框只有一组模式来过滤文件的概念。 如果提供了多个 FileFilters,Wails 将使用所有定义的模式。 + +示例: + +```go + selection, err := runtime.OpenFileDialog(b.ctx, runtime.OpenDialogOptions{ + Title: "Select File", + Filters: []runtime.FileFilter{ + { + DisplayName: "Images (*.png;*.jpg)", + Pattern: "*.png;*.jpg", + }, { + DisplayName: "Videos (*.mov;*.mp4)", + Pattern: "*.mov;*.mp4", + }, + }, + }) +``` + +这将导致使用 `*.png,*.jpg,*.mov,*.mp4` 作为过滤器打开文件对话框。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx new file mode 100644 index 00000000000..7788eed7b4c --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx @@ -0,0 +1,37 @@ +--- +sidebar_position: 2 +--- + +# Events 事件 + +Wails 运行时提供了一个统一的事件系统,其中事件可以由 Go 或 JavaScript 发出或接收。 可选地,数据可以与事件一起传递。 侦听器将接收本地数据类型中的数据。 + +### EventsOn 添加事件侦听器 + +此方法为给定的事件名称设置一个侦听器。 当 [触发指定事件](#触发指定事件) 名为 `eventName` 类型的事件时,将触发回调。 与触发事件一起发送的任何其他数据都将传递给回调。 它返回 一个函数来取消侦听器。 + +Go: `EventsOn(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOn(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOff 移除事件侦听器 + +此方法取消注册给定事件名称的侦听器,可选地,可以通过 `additionalEventNames` 取消注册多个侦听器。 + +Go: `EventsOff(ctx context.Context, eventName string, additionalEventNames ...string)`
JS: `EventsOff(eventName string, ...additionalEventNames)` + +### EventsOnce 添加只触发一次的事件侦听器 + +此方法为给定的事件名称设置一个侦听器,但只会触发一次。 它返回 一个函数来取消侦听器。 + +Go: `EventsOnce(ctx context.Context, eventName string, callback func(optionalData ...interface{})) func()`
JS: `EventsOnce(eventName string, callback function(optionalData?: any)): () => void` + +### EventsOnMultiple 添加指定对多触发次数的事件侦听器 + +此方法为给定的事件名称设置一个侦听器,但最多只能触发 `counter` 次。 它返回 一个函数来取消侦听器。 + +Go: `EventsOnMultiple(ctx context.Context, eventName string, callback func(optionalData ...interface{}), counter int) func()`
JS: `EventsOnMultiple(eventName string, callback function(optionalData?: any), counter int): () => void` + +### EventsEmit 触发指定事件 + +此方法触发指定的事件。 可选数据可以与事件一起传递。 这将触发任意事件侦听器。 + +Go: `EventsEmit(ctx context.Context, eventName string, optionalData ...interface{})`
JS: `EventsEmit(eventName: string, ...optionalData: any)` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx new file mode 100644 index 00000000000..107e2def8e2 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx @@ -0,0 +1,85 @@ +--- +sidebar_position: 1 +--- + +# 简介 + +运行时是一个为应用程序提供实用方法的库。 有 Go 和 JavaScript 运行时,目的是在可能的情况下尝试使它们保持一致。 + +它具有以下实用方法: + +- [窗口](window.mdx) +- [菜单](menu.mdx) +- [对话框](dialog.mdx) +- [事件](events.mdx) +- [浏览器](browser.mdx) +- [日志](log.mdx) +- [剪切板](clipboard.mdx) + +Go 运行时可通过导入 `github.com/wailsapp/wails/v2/pkg/runtime` 获取。 此包中的所有方法都将 context 作为第一个参数。 此 context 应该从 [应用启动回调](../options.mdx#onstartup) 或 [前端 Dom 加载完成回调](../options.mdx#ondomready) 回调方法中获取。 + +:::info 注意 + +虽然上下文将提供给 [应用启动回调](../../reference/options#应用启动回调) 方法,但不能保证运行时将在此方法中工作,因为窗口正在不同的线程中初始化。 如果您希望在启动时调用运行时方法,请使用 [前端 Dom 加载完成回调](../../reference/options#前端-dom-加载完成回调) 方法。 + +::: + +JavaScript 库可通过 `window.runtime` 提供给前端。 使用 `开发` 模式时会生成一个运行时包,该包为运行时提供 TypeScript 声明。 这应该位于您的前端目录的`wailsjs`目录中。 + +### 隐藏 + +Go: `Hide(ctx context.Context)`
JS: `Hide()` + +隐藏应用程序。 + +:::info 注意 + +`Hide` 在 Mac 上,这将以与标准 Mac 应用程序中的菜单项相同的方式隐藏应用程序。 这与隐藏窗口不同,但应用程序仍处于前台。 对于 Windows 和 Linux,这与 `WindowHide` 相同。 + +::: + +### 显示 + +显示应用程序。 + +:::info 注意 + +在 Mac 上,这会将应用程序带回前台。 对于 Windows 和 Linux,这目前与 `WindowShow` 相同。 + +::: + +Go: `Show(ctx context.Context)`
JS: `Show()` + +### 退出 + +退出应用程序。 + +Go: `Quit(ctx context.Context)`
JS: `Quit()` + +### 环境 + +返回当前环境的详细信息。 + +Go: `Environment(ctx context.Context) EnvironmentInfo`
JS: `Environment(): Promise` + +#### 环境信息 + +Go: + +```go +type EnvironmentInfo struct { + BuildType string + Platform string + Arch string +} +``` + +JS: + +```ts +interface EnvironmentInfo { + buildType: string; + platform: string; + arch: string; +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx new file mode 100644 index 00000000000..d109faab2ee --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 3 +--- + +# Log 日志 + +Wails 运行时提供了一种可以从 Go 或 JavaScript 调用日志记录的机制。 像大多数记录器一样,有许多日志级别: + +- Trace(追踪) +- Debug(调试) +- Info(信息) +- Warning(警告) +- Error(错误) +- Fatal(致命) + +记录器将输出当前或更高日志级别的任何日志消息。 示例:`Debug`日志级别将输出除`Trace`消息之外的所有消息。 + +### LogPrint Print 日志 + +将给定的消息记录为原始消息。 + +Go: `LogPrint(ctx context.Context, message string)`
JS: `LogPrint(message: string)` + +### LogPrintf 格式化 Print 日志 + +将给定的消息记录为原始消息。 + +Go: `LogPrintf(ctx context.Context, format string, args ...interface{})`
+ +### LogTrace Trace 日志 + +在 `Trace` 日志级别记录给定的消息。 + +Go: `LogTrace(ctx context.Context, message string)`
JS: `LogTrace(message: string)` + +### LogTracef 格式化 Trace 日志 + +在 `Trace` 日志级别记录给定的消息。 + +Go: `LogTracef(ctx context.Context, format string, args ...interface{})`
+ +### LogDebug Debug 日志 + +在 `Debug` 日志级别记录给定的消息。 + +Go: `LogDebug(ctx context.Context, message string)`
JS: `LogDebug(message: string)` + +### LogDebugf 格式化 Debug 日志 + +在 `Debug` 日志级别记录给定的消息。 + +Go: `LogDebugf(ctx context.Context, format string, args ...interface{})`
+ +### LogInfo Info 日志 + +在`Info`日志级别记录给定的消息。 + +Go: `LogInfo(ctx context.Context, message string)`
JS: `LogInfo(message: string)` + +### LogInfof 格式化 Info 日志 + +在`Info`日志级别记录给定的消息。 + +Go: `LogInfof(ctx context.Context, format string, args ...interface{})`
+ +### LogWarning Warning 日志 + +在 `Warning` 日志级别记录给定的消息。 + +Go: `LogWarning(ctx context.Context, message string)`
JS: `LogWarning(message: string)` + +### LogWarningf 格式化 Warning 日志 + +在 `Warning` 日志级别记录给定的消息。 + +Go: `LogWarningf(ctx context.Context, format string, args ...interface{})`
+ +### LogError Error 日志 + +在 `Error` 日志级别记录给定的消息。 + +Go: `LogError(ctx context.Context, message string)`
JS: `LogError(message: string)` + +### LogErrorf 格式化 Error 日志 + +在 `Error` 日志级别记录给定的消息。 + +Go: `LogErrorf(ctx context.Context, format string, args ...interface{})`
+ +### LogFatal Fatal 日志 + +在 `Fatal` 日志级别记录给定的消息。 + +Go: `LogFatal(ctx context.Context, message string)`
JS: `LogFatal(message: string)` + +### LogFatalf 格式化 Fatal 日志 + +在 `Fatal` 日志级别记录给定的消息。 + +Go: `LogFatalf(ctx context.Context, format string, args ...interface{})`
+ +### LogSetLogLevel 设置日志级别 + +设置日志级别。 在 JavaScript 中,该数字与以下日志级别有关: + +| 值 | 日志等级 | +| - | ----------- | +| 1 | Trace(追踪) | +| 2 | Debug(调试) | +| 3 | Info(信息) | +| 4 | Warning(警告) | +| 5 | Error(错误) | + +Go: `LogSetLogLevel(ctx context.Context, level logger.LogLevel)`
JS: `LogSetLogLevel(level: number)` + +## 使用自定义日志 + +可以通过使用应用程序参数选项 [日志](../../reference/options#日志) 提供自定义记录器来使用它。 唯一的要求是记录器实现在 `github.com/wailsapp/wails/v2/pkg/logger` 里 `logger.Logger` 定义的接口: + +```go title="logger.go" +type Logger interface { + Print(message string) + Trace(message string) + Debug(message string) + Info(message string) + Warning(message string) + Error(message string) + Fatal(message string) +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx new file mode 100644 index 00000000000..9d176d06a61 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx @@ -0,0 +1,25 @@ +--- +sidebar_position: 6 +--- + +# Menu 菜单 + +这些方法与应用程序菜单相关。 + +:::info JavaScript + +JS 运行时当前不支持菜单。 + +::: + +### MenuSetApplicationMenu 设置应用程序菜单 + +将应用程序菜单设置为给定的 [菜单](../menus.mdx)。 + +Go: `MenuSetApplicationMenu(ctx context.Context, menu *menu.Menu)` + +### MenuUpdateApplicationMenu 更新应用程序菜单 + +获取传递给 `MenuSetApplicationMenu` 的菜单的任意更改更新应用程序菜单。 + +Go: `MenuUpdateApplicationMenu(ctx context.Context)` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx new file mode 100644 index 00000000000..457c92ebff9 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx @@ -0,0 +1,38 @@ +--- +sidebar_position: 9 +--- + +# Screen + +These methods provide information about the currently connected screens. + +### ScreenGetAll + +Returns a list of currently connected screens. + +Go: `ScreenGetAll(ctx context.Context) []screen`
+JS: `ScreenGetAll()` + +#### Screen + +Go struct: + +```go +type Screen struct { + IsCurrent bool + IsPrimary bool + Width int + Height int +} +``` + +Typescript interface: + +```ts +interface Screen { + isCurrent: boolean; + isPrimary: boolean; + width : number + height : number +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx new file mode 100644 index 00000000000..f5397d8472d --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -0,0 +1,227 @@ +--- +sidebar_position: 4 +--- + +# Window 窗口 + +这些方法可以控制应用程序窗口。 + +### WindowSetTitle 窗口标题 + +设置窗口标题栏中的文本。 + +Go: `WindowSetTitle(ctx context.Context, title string)`
JS: `WindowSetTitle(title: string)` + +### WindowFullscreen 窗口全屏 + +使窗口全屏。 + +Go: `WindowFullscreen(ctx context.Context)`
JS: `WindowFullscreen()` + +### WindowUnfullscreen 窗口取消全屏 + +恢复全屏之前的先前窗口尺寸和位置。 + +Go: `WindowUnfullscreen(ctx context.Context)`
JS: `WindowUnfullscreen()` + +### WindowIsFullscreen 窗口是否全屏 + +如果窗口是全屏的,则返回 true。 + +Go: `WindowIsFullscreen(ctx context.Context) bool`
JS: `WindowIsFullscreen() bool` + +### WindowCenter 窗口居中 + +使窗口在当前窗口所在的监视器上居中。 + +Go: `WindowCenter(ctx context.Context)`
JS: `WindowCenter()` + +### WindowExecJS 窗口执行JS代码 + +在窗口中执行任意 JS 代码。 + +此方法在浏览器中异步运行代码并立即返回。 如果脚本导致任何错误,它们将只在浏览器控制台中可用。 + +Go: `WindowExecJS(ctx context.Context, js string)` + +### WindowReload 窗口重新加载 + +执行“重新加载”(重新加载当前页面)。 + +Go: `WindowReload(ctx context.Context)`
JS: `WindowReload()` + +### WindowReloadApp 重新加载应用程序前端。 + +重新加载应用程序前端。 + +Go: `WindowReloadApp(ctx context.Context)`
JS: `WindowReloadApp()` + +### WindowSetSystemDefaultTheme 窗口设置系统默认主题 + +仅限 Windows。 + +Go: `WindowSetSystemDefaultTheme(ctx context.Context)`
JS: `WindowSetSystemDefaultTheme()` + +将窗口主题设置为系统默认值(暗/亮)。 + +### WindowSetLightTheme 窗口设置浅色主题 + +仅限 Windows。 + +Go: `WindowSetLightTheme(ctx context.Context)`
JS: `WindowSetLightTheme()` + +将窗口主题设置为浅色。 + +### WindowSetDarkTheme 窗口设置深色主题 + +仅限 Windows。 + +Go: `WindowSetDarkTheme(ctx context.Context)`
JS: `WindowSetDarkTheme()` + +将窗口主题设置为深色。 + +### WindowShow 显示窗口 + +显示窗口,如果它当前是隐藏的。 + +Go: `WindowShow(ctx context.Context)`
JS: `WindowShow()` + +### WindowShow 隐藏窗口 + +如果当前可见,则隐藏窗口。 + +Go: `WindowHide(ctx context.Context)`
JS: `WindowHide()` + +### WindowIsNormal 窗口是否为正常 + +如果窗口未最小化、最大化或全屏,则返回 true。 + +Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` + +### WindowSetSize 设置窗口尺寸 + +设置窗口的宽度和高度。 + +Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` + +### WindowSetSize 获取窗口尺寸 + +获取窗口的宽度和高度。 + +Go: `WindowGetSize(ctx context.Context) (width int, height int)`
JS: `WindowGetSize() : Size` + +### WindowSetMinSize 设置窗口最小尺寸 + +设置窗口最小尺寸。 如果窗口当前小于给定尺寸,将调整窗口大小。 + +设置大小 `0,0` 将禁用此约束。 + +Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
JS: `WindowSetMinSize(width: number, height: number)` + +### WindowSetMaxSize 设置窗口最大尺寸 + +设置窗口最大尺寸。 如果窗口当前大于给定尺寸,将调整窗口大小。 + +设置大小 `0,0` 将禁用此约束。 + +Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
JS: `WindowSetMaxSize(width: number, height: number)` + +### WindowSetAlwaysOnTop 设置窗口置顶 + +设置窗口置顶或取消置顶。 + +Go: `WindowSetAlwaysOnTop(ctx context.Context, b bool)`
JS: `WindowSetAlwaysOnTop(b: Boolen)` + +### WindowSetPosition 设置窗口位置 + +设置相对于窗口当前所在监视器的窗口位置。 + +Go: `WindowSetPosition(ctx context.Context, x int, y int)`
JS: `WindowSetPosition(x: number, y: number)` + +### WindowGetPosition 获取窗口位置 + +获取相对于窗口当前所在监视器的窗口位置。 + +Go: `WindowGetPosition(ctx context.Context) (x int, y int)`
JS: `WindowGetPosition() : Position` + +### WindowMaximise 窗口最大化 + +最大化窗口以填满屏幕。 + +Go: `WindowMaximise(ctx context.Context)`
JS: `WindowMaximise()` + +### WindowUnmaximise 窗口取消最大化 + +将窗口恢复到最大化之前的尺寸和位置。 + +Go: `WindowUnmaximise(ctx context.Context)`
JS: `WindowUnmaximise()` + +### WindowIsMaximised 窗口是否最大化 + +如果窗口最大化,则返回 true。 + +Go: `WindowIsMaximised(ctx context.Context) bool`
JS: `WindowIsMaximised() bool` + +### WindowToggleMaximise 窗口最大化切换 + +在最大化和未最大化之间切换。 + +Go: `WindowToggleMaximise(ctx context.Context)`
JS: `WindowToggleMaximise()` + +### WindowMinimise 窗口最小化。 + +最小化窗口。 + +Go: `WindowMinimise(ctx context.Context)`
JS: `WindowMinimise()` + +### WindowUnminimise 窗口取消最小化 + +将窗口恢复到最小化之前的尺寸和位置。 + +Go: `WindowUnminimise(ctx context.Context)`
JS: `WindowUnminimise()` + +### WindowIsMinimised 窗口是否最小化 + +如果窗口最小化,则返回 true。 + +Go: `WindowIsMinimised(ctx context.Context) bool`
JS: `WindowIsMinimised() bool` + +### WindowSetBackgroundColour 窗口设置背景色 + +将窗口的背景颜色设置为给定的 RGBA 颜色定义。 这种颜色将显示所有透明像素。 + +R、G、B 和 A 的有效值为 0-255。 + +:::info Windows + +在 Windows 上,仅支持 0 或 255 的 alpha 值。 任何非 0 的值都将被视为 255。 + +::: + +Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` + +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
JS: `WindowPrint()` + +## TypeScript 对象定义 + +### Position(位置) + +```ts +interface Position { + x: number; + y: number; +} +``` + +### Size(尺寸) + +```ts +interface Size { + w: number; + h: number; +} +``` diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/_category_.json b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/_category_.json new file mode 100644 index 00000000000..dfac1d175e6 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Tutorials", + "position": 70 +} diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx new file mode 100644 index 00000000000..cb5d4d79ce6 --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx @@ -0,0 +1,245 @@ +--- +sidebar_position: 20 +--- + +# 狗狗 API + +```mdx-code-block +
+ +
+
+``` + +:::note + +本教程由 [@tatadan](https://twitter.com/tatadan) 友情提供,是他们的 [Wails Examples Repository](https://github.com/tataDan/wails-v2-examples) 的一部分。 + +::: + +在本教程中,我们将开发一个应用程序,从网络上检索狗的照片,然后显示它们。 + +### 创建项目 + +让我们创建应用程序。 从终端输入:`wails init -n dogs-api -t svelte` + +注意:如果您想添加 IDE 支持,我们可以选择在此命令的末尾添加 `-ide vscode` 或 `-ide goland`。 + +现在让我们 `cd dogs-api` 开始编辑项目文件。 + +### 删除未使用的代码 + +我们将从删除一些我们知道不会使用的元素开始: + +- 打开 `app.go` 并删除以下行: + +```go +// Greet returns a greeting for the given name +func (a *App) Greet(name string) string { + return fmt.Sprintf("Hello %s, It's show time!", name) +} +``` + +- 打开 `frontend/src/App.svelte` 并删除所有行。 +- 删除 `frontend/src/assets/images/logo-universal.png` 文件 + +### 创建应用程序 + +现在让我们添加新的 Go 代码。 + +在函数定义之前添加以下结构声明到 `app.go`: + +```go +type RandomImage struct { + Message string + Status string +} + +type AllBreeds struct { + Message map[string]map[string][]string + Status string +} + +type ImagesByBreed struct { + Message []string + Status string +} +``` + +将以下函数添加到 `app.go`(可能在现有函数定义之后): + +```go +func (a *App) GetRandomImageUrl() string { + response, err := http.Get("https://dog.ceo/api/breeds/image/random") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data RandomImage + json.Unmarshal(responseData, &data) + + return data.Message +} + +func (a *App) GetBreedList() []string { + var breeds []string + + response, err := http.Get("https://dog.ceo/api/breeds/list/all") + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data AllBreeds + json.Unmarshal(responseData, &data) + + for k := range data.Message { + breeds = append(breeds, k) + } + + sort.Strings(breeds) + + return breeds +} + +func (a *App) GetImageUrlsByBreed(breed string) []string { + + url := fmt.Sprintf("%s%s%s%s", "https://dog.ceo/api/", "breed/", breed, "/images") + response, err := http.Get(url) + if err != nil { + log.Fatal(err) + } + + responseData, err := ioutil.ReadAll(response.Body) + if err != nil { + log.Fatal(err) + } + + var data ImagesByBreed + json.Unmarshal(responseData, &data) + + return data.Message +} +``` + +将 `app.go` 的 `import` 部分修改为如下所示: + +```go +import ( + "context" + "fmt" + "encoding/json" + "io/ioutil" + "log" + "net/http" + "sort" +) +``` + +将以下行添加到 `frontend/src/App.svelte`: + + +```html + + +

Dogs API

+
+ + Click on down arrow to select a breed + + +
+
+{#if showRandomPhoto} + No dog found +{/if} +{#if showBreedPhotos} + {#each photos as photo} + No dog found + {/each} +{/if} + + +``` + + +### 创建应用程序 + +要生成绑定并测试应用程序,请运行 `wails dev`。 + +### 编译应用程序 + +要将应用程序编译为单个可用于生产的二进制文件,请运行 `wails build`。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx new file mode 100644 index 00000000000..863f7ac8f0a --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx @@ -0,0 +1,123 @@ +--- +sidebar_position: 10 +--- + +# 你好世界 + +本教程的目的是让您使用 Wails 启动并运行最基本的应用程序。 你将可以: + +- 创建一个新的 Wails 应用 +- 构建应用 +- 运行应用 + +:::note + +本教程使用 Windows 作为目标平台。 根据您的操作系统,输出会略有不同。 + +::: + +## 创建一个新的 Wails 应用 + +使用默认的 vanilla JS 模板创建新的 Wails 程序, 您需要运行这个指令: + +```bash +wails init -n helloworld +``` + +您应该会看到如下输出: + +``` +Wails CLI v2.0.0 + +Initialising Project 'helloworld' +--------------------------------- + +Project Name: helloworld +Project Directory: C:\Users\leaan\tutorial\helloworld +Project Template: vanilla +Template Support: https://wails.io + +Initialised project 'helloworld' in 232ms. +``` + +这将在当前目录中创建一个名为 `helloworld` 的新目录。 在此目录中,您将找到许多文件: + +``` +build/ - Contains the build files + compiled application +frontend/ - Contains the frontend files +app.go - Contains the application code +main.go - The main program with the application configuration +wails.json - The project configuration file +go.mod - The go module file +go.sum - The go module checksum file +``` + +## 构建应用 + +要构建应用程序,请切换到新的 `helloword` 项目目录并运行以下命令: + +```bash +wails build +``` + +您应该会看到如下输出: + +``` +Wails CLI v2.0.0 + +App Type: desktop +Platforms: windows/amd64 +Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe +Build Mode: Production +Devtools: false +Skip Frontend: false +Compress: false +Package: true +Clean Build Dir: false +LDFlags: "" +Tags: [] +Race Detector: false + +Building target: windows/amd64 +------------------------------ + - Installing frontend dependencies: Done. + - Compiling frontend: Done. + - Generating bundle assets: Done. + - Compiling application: Done. +Built 'C:\Users\leaan\tutorial\helloworld\build\bin\helloworld.exe' in 10.616s. +``` + +这已经编译了应用程序并将其保存在 `build/bin` 目录中。 + +## 运行应用 + +如果我们在 Windows 文件管理器中查看 `build/bin` 目录,我们应该看到我们的项目二进制文件: + +```mdx-code-block +
+ +
+
+``` + +我们可以通过简单的双击 `helloworld.exe` 文件来运行它。 + +在 Mac 上,Wails 生成一个 `helloworld.app` 文件,可以通过双击来运行。 + +在 Linux 上,您可以从 `build/bin` 目录使用 `./helloword` 运行应用程序。 + +您应该看到应用程序正常工作: + +```mdx-code-block +
+ +
+
+``` diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 3cbadd5b616..a1476656608 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -14,6 +14,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +## v2.7.0 - 2023-12-09 + ### Added - Update the description of `ZoomFactor` and `IsZoomControlEnabled` attributes in the document. Added by @biuaxia in [PR](https://github.com/wailsapp/wails/pull/3013) diff --git a/website/versioned_docs/version-v2.5.0/guides/troubleshooting.mdx b/website/versioned_docs/version-v2.5.0/guides/troubleshooting.mdx deleted file mode 100644 index 9a68610c39b..00000000000 --- a/website/versioned_docs/version-v2.5.0/guides/troubleshooting.mdx +++ /dev/null @@ -1,180 +0,0 @@ -# Troubleshooting - -An assortment of troubleshooting tips. - -## The `wails` command appears to be missing? - -If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide -correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment -variable. You will also normally need to close and reopen any open command prompts so that changes to the environment -made by the installer are reflected at the command prompt. - -## My application is displaying a white/blank screen - -Check that your application includes the assets from the correct directory. In your `main.go` file, you will have -something similar to the following code: - -```go -//go:embed all:frontend/dist -var assets embed.FS -``` - -Check that `frontend/dist` contains your application assets. - -### Mac - -If this happens on Mac, try adding the following to your `Info.plist`: - -```xml -NSAppTransportSecurity - - NSAllowsLocalNetworking - - -``` - -Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 - -## Mac application not valid - -If your built application looks like this in finder: - -```mdx-code-block -

- -

-``` - -it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` -and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to -the `build/darwin` directory. - -## Cannot call backend method from frontend with variadic arguments - -If you have a backend method defined with variadic parameters, eg: - -```go -func (a *App) TestFunc(msg string, args ...interface{}) error { - // Code -} -``` - -calling this method from the frontend like this will fail: - -```js -var msg = "Hello: "; -var args = ["Go", "JS"]; -window.go.main.App.TestFunc(msg, ...args) - .then((result) => { - //do things here - }) - .catch((error) => { - //handle error - }); -``` - -Workaround: - -```js -var msg = "Hello "; -var args = ["Go", "JS"]; -window.go.main.App.TestFunc(msg, args) - .then((result) => { - //without the 3 dots - //do things here - }) - .catch((error) => { - //handle error - }); -``` - -Credit: https://github.com/wailsapp/wails/issues/1186 - -## I'm having getting proxy errors when trying to install Wails - -If you are getting errors like this: - -``` -"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. -``` - -it's probably because the official Go Proxy is being blocked (Users in China have reported this). -The solution is to set up the proxy manually, eg: - -``` -go env -w GO111MODULE=on -go env -w GOPROXY=https://goproxy.cn,direct -``` - -Source: https://github.com/wailsapp/wails/issues/1233 - -## The generated TypeScript doesn't have the correct types - -Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, -it is possible to specify what types should be generated using the `ts_type` struct tag. For -more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). - -## When I navigate away from `index.html`, I am unable to call methods on the frontend - -If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding -the following imports to the `` section of any new page you navigate to: - -```html - - - - -``` - -Source: https://github.com/wailsapp/wails/discussions/1512 - -## I get `too many open files` errors on my Mac when I run `wails dev` - -By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. -This limit can be increased by running: `ulimit -n 1024` in the terminal. - -FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. -If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). - -## My Mac app gives me weird compilation errors - -A few users have reported seeing compilation errors such as the following: - -```shell -# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin -In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: -In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: -/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString -- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); - ~~~~~~~~~~~~~~ ^ ~ -/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' - #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) -``` - -This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools -installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. - -If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: - -`xcode-select -p` - -If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` - -Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 - --- - -## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" - -It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine -that may have different versions of Node installed, you may not be able to run your application. - -If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. - -## Build process stuck on "Generating bindings" - -Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. diff --git a/website/versioned_docs/version-v2.7.0/appendix/_category_.json b/website/versioned_docs/version-v2.7.0/appendix/_category_.json new file mode 100644 index 00000000000..83af4ca28fd --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/appendix/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Appendix", + "position": 70 +} diff --git a/website/versioned_docs/version-v2.7.0/community/_category_.json b/website/versioned_docs/version-v2.7.0/community/_category_.json new file mode 100644 index 00000000000..524986e1e49 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Community", + "position": 50 +} diff --git a/website/versioned_docs/version-v2.5.0/community/links.mdx b/website/versioned_docs/version-v2.7.0/community/links.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/links.mdx rename to website/versioned_docs/version-v2.7.0/community/links.mdx diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/_category_.json b/website/versioned_docs/version-v2.7.0/community/showcase/_category_.json new file mode 100644 index 00000000000..276e283b71f --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Showcase", + "position": 1 +} diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/bulletinboard.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..37be75135ed --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +

+ +
+

+``` + +The [BulletinBoard](https://github.com/raguay/BulletinBoard) application is a versital message board for static messages or dialogs to get information from the user for a script. It has a TUI for creating new dialogs that can latter be used to get information from the user. It's design is to stay running on your system and show the information as needed and then hide away. I have a process for watching a file on my system and sending the contents to BulletinBoard when changed. It works great with my workflows. There is also an [Alfred workflow](https://github.com/raguay/MyAlfred/blob/master/Alfred%205/EmailIt.alfredworkflow) for sending information to the program. The workflow is also for working with [EmailIt](https://github.com/raguay/EmailIt). diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/emailit.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/emailit.mdx new file mode 100644 index 00000000000..c1817b70fff --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/emailit.mdx @@ -0,0 +1,10 @@ +# EmailIt + +```mdx-code-block +

+ +
+

+``` + +[EmailIt](https://github.com/raguay/EmailIt/) is a Wails 2 program that is a markdown based email sender only with nine notepads, scripts to manipulate the text, and templates. It also has a scripts terminal to run scripts in EmailIt on files in your system. The scripts and templates can be used from the commandline itself or with the Alfred, Keyboard Maestro, Dropzone, or PopClip extensions. It also supports scripts and themes downloaded form GitHub. Documentation is not complete, but the programs works. It’s built using Wails2 and Svelte, and the download is a universal macOS application. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/encrypteasy.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/encrypteasy.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/encrypteasy.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/encrypteasy.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/filehound.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/filehound.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/filehound.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/filehound.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/hiposter.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/hiposter.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/hiposter.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/hiposter.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/minecraftupdater.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/minecraftupdater.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/minecraftupdater.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/minecraftupdater.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/modalfilemanager.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/modalfilemanager.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/modalfilemanager.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/modalfilemanager.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/mollywallet.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/mollywallet.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/mollywallet.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/mollywallet.mdx diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/october.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/october.mdx new file mode 100644 index 00000000000..66d634dc519 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/october.mdx @@ -0,0 +1,14 @@ +# October + +```mdx-code-block +

+ +
+

+``` + +[October](https://october.utf9k.net) is a small Wails application that makes it really easy to extract highlights from [Kobo eReaders](https://en.wikipedia.org/wiki/Kobo_eReader) and then forward them to [Readwise](https://readwise.io). + +It has a relatively small scope with all platform versions weighing in under 10MB, and that's without enabling [UPX compression](https://upx.github.io/)! + +In contrast, the author's previous attempts with Electron quickly bloated to several hundred megabytes. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/optimus.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/optimus.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/optimus.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/optimus.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/portfall.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/portfall.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/portfall.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/portfall.mdx diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/restic-browser.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/restic-browser.mdx new file mode 100644 index 00000000000..3646384ecad --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/restic-browser.mdx @@ -0,0 +1,12 @@ +# Restic Browser + +```mdx-code-block +

+ +
+

+``` + +[Restic-Browser](https://github.com/emuell/restic-browser) - A simple, cross-platform [restic](https://github.com/restic/restic) backup GUI for browsing and restoring restic repositories. diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/riftshare.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/riftshare.mdx new file mode 100644 index 00000000000..9928b478576 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/riftshare.mdx @@ -0,0 +1,21 @@ +# RiftShare + +```mdx-code-block +

+ +
+

+``` + +Easy, Secure, and Free file sharing for everyone. Learn more at [Riftshare.app](https://riftshare.app) + +## Features + +- Easy secure file sharing between computers both in the local network and through the internet +- Supports sending files or directories securely through the [magic wormhole protocol](https://magic-wormhole.readthedocs.io/en/latest/) +- Compatible with all other apps using magic wormhole (magic-wormhole or wormhole-william CLI, wormhole-gui, etc.) +- Automatic zipping of multiple selected files to send at once +- Full animations, progress bar, and cancellation support for sending and receiving +- Native OS File Selection +- Open files in one click once received +- Auto Update - don't worry about having the latest release! diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/scriptbar.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/scriptbar.mdx new file mode 100644 index 00000000000..3e41eb32abf --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/scriptbar.mdx @@ -0,0 +1,10 @@ +# ScriptBar + +```mdx-code-block +

+ +
+

+``` + +[ScriptBar](https://GitHub.com/raguay/ScriptBarApp) is a program to show the output of scripts or [Node-Red](https://nodered.org) server. It runs scripts defined in EmailIt program and shows the output. Scripts from xBar or TextBar can be used, but currently on the TextBar scripts work well. It also displays the output of scripts on your system. ScriptBar doesn't put them in the menubar, but has them all in a convient window for easy viewing. You can have multiple tabs to have many different things show. You can also keep the links to your most visited web sites. diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/snippetexpander.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/snippetexpander.mdx new file mode 100644 index 00000000000..1f9fb615796 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/snippetexpander.mdx @@ -0,0 +1,27 @@ +# Snippet Expander + +```mdx-code-block +

+ +
+ Screenshot of Snippet Expander's Select Snippet window +

+

+ +
+ Screenshot of Snippet Expander's Add Snippet screen +

+

+ +
+ Screenshot of Snippet Expander's Search & Paste window +

+``` + +[Snippet Expander](https://snippetexpander.org) is "Your little expandable text snippets helper", for Linux. + +Snippet Expander comprises of a GUI application built with Wails for managing snippets and settings, with a Search & Paste window mode for quickly selecting and pasting a snippet. + +The Wails based GUI, go-lang CLI and vala-lang auto expander daemon all communicate with a go-lang daemon via D-Bus. The daemon does the majority of the work, managing the database of snippets and common settings, and providing services for expanding and pasting snippets etc. + +Check out the [source code](https://git.sr.ht/~ianmjones/snippetexpander/tree/trunk/item/cmd/snippetexpandergui/app.go#L38) to see how the Wails app sends messages from the UI to the backend that are then sent to the daemon, and subscribes to a D-Bus event to monitor changes to snippets via another instance of the app or CLI and show them instantly in the UI via a Wails event. diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/surge.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/surge.mdx new file mode 100644 index 00000000000..c3b3fb4c06d --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/surge.mdx @@ -0,0 +1,10 @@ +# Surge + +```mdx-code-block +

+ +
+

+``` + +[Surge](https://getsurge.io/) is a p2p filesharing app designed to utilize blockchain technologies to enable 100% anonymous file transfers. Surge is end-to-end encrypted, decentralized and open source. diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/tinyrdm.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/tinyrdm.mdx new file mode 100644 index 00000000000..cd9cec8b309 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/tinyrdm.mdx @@ -0,0 +1,10 @@ +# Tiny RDM + +```mdx-code-block +

+ +
+

+``` + +The [Tiny RDM](https://redis.tinycraft.cc/) application is an open-source, modern lightweight Redis GUI. It has a beautful UI, intuitive Redis database management, and compatible with Windows, Mac, and Linux. It provides visual key-value data operations, supports various data decoding and viewing options, built-in console for executing commands, slow log queries and more. diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/wally.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/wally.mdx new file mode 100644 index 00000000000..7408aa5854f --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/wally.mdx @@ -0,0 +1,10 @@ +# Wally + +```mdx-code-block +

+ +
+

+``` + +[Wally](https://ergodox-ez.com/pages/wally) is the official firmware flasher for [Ergodox](https://ergodox-ez.com/) keyboards. It looks great and is a fantastic example of what you can achieve with Wails: the ability to combine the power of Go and the rich graphical tools of the web development world. diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/warmine.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/warmine.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/warmine.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/warmine.mdx diff --git a/website/versioned_docs/version-v2.5.0/community/showcase/wombat.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/wombat.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/community/showcase/wombat.mdx rename to website/versioned_docs/version-v2.7.0/community/showcase/wombat.mdx diff --git a/website/versioned_docs/version-v2.7.0/community/showcase/ytd.mdx b/website/versioned_docs/version-v2.7.0/community/showcase/ytd.mdx new file mode 100644 index 00000000000..5db428f722c --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/community/showcase/ytd.mdx @@ -0,0 +1,10 @@ +# Ytd + +```mdx-code-block +

+ +
+

+``` + +[Ytd](https://github.com/marcio199226/ytd/tree/v2-wails) is an app for downloading tracks from youtube, creating offline playlists and share them with your friends, your friends will be able to playback your playlists or download them for offline listening, has an built-in player. diff --git a/website/versioned_docs/version-v2.5.0/community/templates.mdx b/website/versioned_docs/version-v2.7.0/community/templates.mdx similarity index 84% rename from website/versioned_docs/version-v2.5.0/community/templates.mdx rename to website/versioned_docs/version-v2.7.0/community/templates.mdx index 1ac9ef5262c..98726c60aff 100644 --- a/website/versioned_docs/version-v2.5.0/community/templates.mdx +++ b/website/versioned_docs/version-v2.7.0/community/templates.mdx @@ -40,18 +40,17 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for - [wails-react-template](https://github.com/AlienRecall/wails-react-template) - A template using reactjs - [wails-react-template](https://github.com/flin7/wails-react-template) - A minimal template for React that supports live development - [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - A template using Next.js and TypeScript +- [wails-template-nextjs-app-router](https://github.com/thisisvk-in/wails-template-nextjs-app-router) - A template using Next.js and TypeScript with App router - [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - A template for React + TypeScript + Vite + TailwindCSS +- [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui ## Svelte - [wails-svelte-template](https://github.com/raitonoberu/wails-svelte-template) - A template using Svelte - [wails-vite-svelte-template](https://github.com/BillBuilt/wails-vite-svelte-template) - A template using Svelte and Vite - [wails-vite-svelte-tailwind-template](https://github.com/BillBuilt/wails-vite-svelte-tailwind-template) - A template using Svelte and Vite with TailwindCSS v3 - -## SvelteKit - +- [wails-svelte-tailwind-vite-template](https://github.com/PylotLight/wails-vite-svelte-tailwind-template/tree/master) - An updated template using Svelte v4.2.0 and Vite with TailwindCSS v3.3.3 - [wails-sveltekit-template](https://github.com/h8gi/wails-sveltekit-template) - A template using SvelteKit -- [wails-sveltekit-ts](https://github.com/haukened/wails-sveltekit-ts) - A template using SvelteKit + Typescript ## Solid @@ -63,10 +62,10 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for - [wails-elm-template](https://github.com/benjamin-thomas/wails-elm-template) - Develop your GUI app with functional programming and a **snappy** hot-reload setup :tada: :rocket: - [wails-template-elm-tailwind](https://github.com/rnice01/wails-template-elm-tailwind) - Combine the powers :muscle: of Elm + Tailwind CSS + Wails! Hot reloading supported. -## Pure JavaScript (Vanilla) +## HTMX -- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS +- [wails-htmx-templ-chi-tailwind](https://github.com/PylotLight/wails-hmtx-templ-template) - Use a unique combination of pure htmx for interactivity plus templ for creating components and forms -## Qwik +## Pure JavaScript (Vanilla) -- [wails-qwik-template](https://github.com/LazyClicks/wails-qwik-template) - A template using Qwik +- [wails-pure-js-template](https://github.com/KiddoV/wails-pure-js-template) - A template with nothing but just basic JavaScript, HTML, and CSS diff --git a/website/versioned_docs/version-v2.7.0/gettingstarted/_category_.json b/website/versioned_docs/version-v2.7.0/gettingstarted/_category_.json new file mode 100644 index 00000000000..597b920dff9 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/gettingstarted/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Getting Started", + "position": 10 +} diff --git a/website/versioned_docs/version-v2.5.0/gettingstarted/building.mdx b/website/versioned_docs/version-v2.7.0/gettingstarted/building.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/gettingstarted/building.mdx rename to website/versioned_docs/version-v2.7.0/gettingstarted/building.mdx diff --git a/website/versioned_docs/version-v2.5.0/gettingstarted/development.mdx b/website/versioned_docs/version-v2.7.0/gettingstarted/development.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/gettingstarted/development.mdx rename to website/versioned_docs/version-v2.7.0/gettingstarted/development.mdx diff --git a/website/versioned_docs/version-v2.5.0/gettingstarted/firstproject.mdx b/website/versioned_docs/version-v2.7.0/gettingstarted/firstproject.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/gettingstarted/firstproject.mdx rename to website/versioned_docs/version-v2.7.0/gettingstarted/firstproject.mdx diff --git a/website/versioned_docs/version-v2.5.0/gettingstarted/installation.mdx b/website/versioned_docs/version-v2.7.0/gettingstarted/installation.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/gettingstarted/installation.mdx rename to website/versioned_docs/version-v2.7.0/gettingstarted/installation.mdx diff --git a/website/versioned_docs/version-v2.7.0/guides/_category_.json b/website/versioned_docs/version-v2.7.0/guides/_category_.json new file mode 100644 index 00000000000..5935dad936c --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Guides", + "position": 50 +} diff --git a/website/versioned_docs/version-v2.5.0/guides/angular.mdx b/website/versioned_docs/version-v2.7.0/guides/angular.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/angular.mdx rename to website/versioned_docs/version-v2.7.0/guides/angular.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/application-development.mdx b/website/versioned_docs/version-v2.7.0/guides/application-development.mdx similarity index 87% rename from website/versioned_docs/version-v2.5.0/guides/application-development.mdx rename to website/versioned_docs/version-v2.7.0/guides/application-development.mdx index 9d04fe9172d..78a6df3bcf1 100644 --- a/website/versioned_docs/version-v2.5.0/guides/application-development.mdx +++ b/website/versioned_docs/version-v2.7.0/guides/application-development.mdx @@ -144,6 +144,65 @@ func main() { } ``` +Also you might want to use Enums in your structs and have models for them on frontend. +In that case you should create array that will contain all possible enum values, instrument enum type and bind it to the app: + +```go {16-18} title="app.go" +type Weekday string + +const ( + Sunday Weekday = "Sunday" + Monday Weekday = "Monday" + Tuesday Weekday = "Tuesday" + Wednesday Weekday = "Wednesday" + Thursday Weekday = "Thursday" + Friday Weekday = "Friday" + Saturday Weekday = "Saturday" +) + +var AllWeekdays = []struct { + Value Weekday + TSName string +}{ + {Sunday, "SUNDAY"}, + {Monday, "MONDAY"}, + {Tuesday, "TUESDAY"}, + {Wednesday, "WEDNESDAY"}, + {Thursday, "THURSDAY"}, + {Friday, "FRIDAY"}, + {Saturday, "SATURDAY"}, +} +``` + +In the main application configuration, the `EnumBind` key is where we can tell Wails what we want to bind enums as well: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + EnumBind: []interface{}{ + AllWeekdays, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +This will add missing enums to your `model.ts` file. + More information on Binding can be found [here](../howdoesitwork.mdx#method-binding). ## Application Menu diff --git a/website/versioned_docs/version-v2.7.0/guides/crossplatform-build.mdx b/website/versioned_docs/version-v2.7.0/guides/crossplatform-build.mdx new file mode 100644 index 00000000000..a9afc616119 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/crossplatform-build.mdx @@ -0,0 +1,65 @@ +# Crossplatform build with Github Actions + +To build a Wails project for all the available platforms, you need to create an application build for each operating system. One effective method to achieve this is by utilizing GitHub Actions. + +An action that facilitates building a Wails app is available at: +https://github.com/dAppServer/wails-build-action + +In case the existing action doesn't fulfill your requirements, you can select only the necessary steps from the source: +https://github.com/dAppServer/wails-build-action/blob/main/action.yml + +Below is a comprehensive example that demonstrates building an app upon the creation of a new Git tag and subsequently uploading it to the Actions artifacts: + +```yaml +name: Wails build + +on: + push: + tags: + # Match any new tag + - '*' + +env: + # Necessary for most environments as build failure can occur due to OOM issues + NODE_OPTIONS: "--max-old-space-size=4096" + +jobs: + build: + strategy: + # Failure in one platform build won't impact the others + fail-fast: false + matrix: + build: + - name: 'App' + platform: 'linux/amd64' + os: 'ubuntu-latest' + - name: 'App' + platform: 'windows/amd64' + os: 'windows-latest' + - name: 'App' + platform: 'darwin/universal' + os: 'macos-latest' + + runs-on: ${{ matrix.build.os }} + steps: + - name: Checkout + uses: actions/checkout@v2 + with: + submodules: recursive + + - name: Build wails + uses: dAppServer/wails-build-action@v2.2 + id: build + with: + build-name: ${{ matrix.build.name }} + build-platform: ${{ matrix.build.platform }} + package: false + go-version: '1.20' +``` + +This example offers opportunities for various enhancements, including: +- Caching dependencies +- Code signing +- Uploading to platforms like S3, Supbase, etc. +- Injecting secrets as environment variables +- Utilizing environment variables as build variables (such as version variable extracted from the current Git tag) diff --git a/website/versioned_docs/version-v2.7.0/guides/custom-protocol-schemes.mdx b/website/versioned_docs/version-v2.7.0/guides/custom-protocol-schemes.mdx new file mode 100644 index 00000000000..c56634f0eb0 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/custom-protocol-schemes.mdx @@ -0,0 +1,189 @@ +# Custom Protocol Scheme association + +Custom Protocols feature allows you to associate specific custom protocol with your app so that when users open links with this protocol, +your app is launched to handle them. This can be particularly useful to connect your desktop app with your web app. +In this guide, we'll walk through the steps to implement custom protocols in Wails app. + + +## Set Up Custom Protocol Schemes Association: +To set up custom protocol, you need to modify your application's wails.json file. +In "info" section add a "protocols" section specifying the protocols your app should be associated with. + +For example: +```json +{ + "info": { + "protocols": [ + { + "scheme": "myapp", + "description": "My App Protocol", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +|:------------|:--------------------------------------------------------------------------------------| +| scheme | Custom Protocol scheme. e.g. myapp | +| description | Windows-only. The description. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS +When you open custom protocol with your app, the system will launch your app and call the `OnUrlOpen` function in your Wails app. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnUrlOpen: func(url string) { println(url) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + + +### Windows +On Windows Custom Protocol Schemes is supported only with NSIS installer. During installation, the installer will create a +registry entry for your schemes. When you open url with your app, new instance of app is launched and url is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there (note that `%u` is important in Exec). Example: +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=x-scheme-handler/myapp; +``` + +2. Prepare postInstall/postRemove scripts for your package. Example: +```sh +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +``` +3. Configure nfpm to use your scripts and files. Example: +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` +6. Build your .deb package using nfpm: +```sh +nfpm pkg --packager deb --target . +``` +7. Now when your package is installed, your app will be associated with custom protocol scheme. When you open url with your app, +new instance of app is launched and file path is passed as argument to your app. +To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open url with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/versioned_docs/version-v2.5.0/guides/dynamic-assets.mdx b/website/versioned_docs/version-v2.7.0/guides/dynamic-assets.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/dynamic-assets.mdx rename to website/versioned_docs/version-v2.7.0/guides/dynamic-assets.mdx diff --git a/website/versioned_docs/version-v2.7.0/guides/file-association.mdx b/website/versioned_docs/version-v2.7.0/guides/file-association.mdx new file mode 100644 index 00000000000..71bbff37ead --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/file-association.mdx @@ -0,0 +1,228 @@ +# File Association + +File association feature allows you to associate specific file types with your app so that when users open those files, +your app is launched to handle them. This can be particularly useful for text editors, image viewers, or any application +that works with specific file formats. In this guide, we'll walk through the steps to implement file association in Wails app. + + +## Set Up File Association: +To set up file association, you need to modify your application's wails.json file. +In "info" section add a "fileAssociations" section specifying the file types your app should be associated with. + +For example: +```json +{ + "info": { + "fileAssociations": [ + { + "ext": "wails", + "name": "Wails", + "description": "Wails Application File", + "iconName": "wailsFileIcon", + "role": "Editor" + }, + { + "ext": "jpg", + "name": "JPEG", + "description": "Image File", + "iconName": "jpegFileIcon", + "role": "Editor" + } + ] + } +} +``` + +| Property | Description | +|:------------|:---------------------------------------------------------------------------------------------------------------------------------------------------| +| ext | The extension (minus the leading period). e.g. png | +| name | The name. e.g. PNG File | +| iconName | The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows | +| description | Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. | +| role | macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. | + +## Platform Specifics: + +### macOS +When you open file (or files) with your app, the system will launch your app and call the `OnFileOpen` function in your Wails app. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + Mac: &mac.Options{ + OnFileOpen: func(filePaths []string) { println(filestring) }, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + + +### Windows +On Windows file association is supported only with NSIS installer. During installation, the installer will create a +registry entry for your file associations. When you open file with your app, new instance of app is launched and file path is passed +as argument to your app. To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` + +### Linux +Currently, Wails doesn't support bundling for Linux. So, you need to create file associations manually. +For example if you distribute your app as a .deb package, you can create file associations by adding required files in you bundle. +You can use [nfpm](https://nfpm.goreleaser.com/) to create .deb package for your app. + +1. Create a .desktop file for your app and specify file associations there. Example: +```ini +[Desktop Entry] +Categories=Office +Exec=/usr/bin/wails-open-file %u +Icon=wails-open-file.png +Name=wails-open-file +Terminal=false +Type=Application +MimeType=application/x-wails;application/x-test +``` + +2. Create mime types file. Example: +```xml + + + + Wails Application File + + + +``` + +3. Create icons for your file types. SVG icons are recommended. +4. Prepare postInstall/postRemove scripts for your package. Example: +```sh +# reload mime types to register file associations +update-mime-database /usr/share/mime +# reload desktop database to load app in list of available +update-desktop-database /usr/share/applications +# update icons +update-icon-caches /usr/share/icons/* +``` +5. Configure nfpm to use your scripts and files. Example: +```yaml +name: "wails-open-file" +arch: "arm64" +platform: "linux" +version: "1.0.0" +section: "default" +priority: "extra" +maintainer: "FooBarCorp " +description: "Sample Package" +vendor: "FooBarCorp" +homepage: "http://example.com" +license: "MIT" +contents: +- src: ../bin/wails-open-file + dst: /usr/bin/wails-open-file +- src: ./main.desktop + dst: /usr/share/applications/wails-open-file.desktop +- src: ./application-wails-mime.xml + dst: /usr/share/mime/packages/application-x-wails.xml +- src: ./application-test-mime.xml + dst: /usr/share/mime/packages/application-x-test.xml +- src: ../appicon.svg + dst: /usr/share/icons/hicolor/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/hicolor/scalable/mimetypes/application-x-test.svg +# copy icons to Yaru theme as well. For some reason Ubuntu didn't pick up fileicons from hicolor theme +- src: ../appicon.svg + dst: /usr/share/icons/Yaru/scalable/apps/wails-open-file.svg +- src: ../wailsFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-wails.svg +- src: ../testFileIcon.svg + dst: /usr/share/icons/Yaru/scalable/mimetypes/application-x-test.svg +scripts: + postinstall: ./postInstall.sh + postremove: ./postRemove.sh +``` +6. Build your .deb package using nfpm: +```sh +nfpm pkg --packager deb --target . +``` +7. Now when your package is installed, your app will be associated with specified file types. When you open file with your app, +new instance of app is launched and file path is passed as argument to your app. +To handle this you should parse command line arguments in your app. Example: +```go title="main.go" +func main() { + argsWithoutProg := os.Args[1:] + + if len(argsWithoutProg) != 0 { + println("launchArgs", argsWithoutProg) + } +} +``` + +You also can enable single instance lock for your app. In this case, when you open file with your app, new instance of app is not launched +and arguments are passed to already running instance. Check single instance lock guide for details. Example: +```go title="main.go" +func main() { + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) +} +``` diff --git a/website/versioned_docs/version-v2.5.0/guides/frameless.mdx b/website/versioned_docs/version-v2.7.0/guides/frameless.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/frameless.mdx rename to website/versioned_docs/version-v2.7.0/guides/frameless.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/frontend.mdx b/website/versioned_docs/version-v2.7.0/guides/frontend.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/frontend.mdx rename to website/versioned_docs/version-v2.7.0/guides/frontend.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/ides.mdx b/website/versioned_docs/version-v2.7.0/guides/ides.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/ides.mdx rename to website/versioned_docs/version-v2.7.0/guides/ides.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/linux-distro-support.mdx b/website/versioned_docs/version-v2.7.0/guides/linux-distro-support.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/linux-distro-support.mdx rename to website/versioned_docs/version-v2.7.0/guides/linux-distro-support.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/linux.mdx b/website/versioned_docs/version-v2.7.0/guides/linux.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/linux.mdx rename to website/versioned_docs/version-v2.7.0/guides/linux.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/local-development.mdx b/website/versioned_docs/version-v2.7.0/guides/local-development.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/local-development.mdx rename to website/versioned_docs/version-v2.7.0/guides/local-development.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/mac-appstore.mdx b/website/versioned_docs/version-v2.7.0/guides/mac-appstore.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/mac-appstore.mdx rename to website/versioned_docs/version-v2.7.0/guides/mac-appstore.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/manual-builds.mdx b/website/versioned_docs/version-v2.7.0/guides/manual-builds.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/manual-builds.mdx rename to website/versioned_docs/version-v2.7.0/guides/manual-builds.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/migrating.mdx b/website/versioned_docs/version-v2.7.0/guides/migrating.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/migrating.mdx rename to website/versioned_docs/version-v2.7.0/guides/migrating.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/mouse-buttons.mdx b/website/versioned_docs/version-v2.7.0/guides/mouse-buttons.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/mouse-buttons.mdx rename to website/versioned_docs/version-v2.7.0/guides/mouse-buttons.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/obfuscated.mdx b/website/versioned_docs/version-v2.7.0/guides/obfuscated.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/obfuscated.mdx rename to website/versioned_docs/version-v2.7.0/guides/obfuscated.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/overscroll.mdx b/website/versioned_docs/version-v2.7.0/guides/overscroll.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/overscroll.mdx rename to website/versioned_docs/version-v2.7.0/guides/overscroll.mdx diff --git a/website/versioned_docs/version-v2.7.0/guides/routing.mdx b/website/versioned_docs/version-v2.7.0/guides/routing.mdx new file mode 100644 index 00000000000..ce176943ea5 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/routing.mdx @@ -0,0 +1,47 @@ +# Routing + +Routing is a popular way to switch views in an application. This page offers some guidance around how to do that. + +## Vue + +The recommended approach for routing in Vue is [Hash Mode](https://next.router.vuejs.org/guide/essentials/history-mode.html#hash-mode): + +```js +import { createRouter, createWebHashHistory } from "vue-router"; + +const router = createRouter({ + history: createWebHashHistory(), + routes: [ + //... + ], +}); +``` + +## Angular + +The recommended approach for routing in Angular is [HashLocationStrategy](https://codecraft.tv/courses/angular/routing/routing-strategies#_hashlocationstrategy): + +```ts +RouterModule.forRoot(routes, { useHash: true }); +``` + +## React + +The recommended approach for routing in React is [HashRouter](https://reactrouter.com/en/main/router-components/hash-router): + +```jsx +import { HashRouter } from "react-router-dom"; + +ReactDOM.render( + + {/* The rest of your app goes here */} + + } exact /> + } /> + } /> + {/* more... */} + + , + root +); +``` diff --git a/website/versioned_docs/version-v2.5.0/guides/signing.mdx b/website/versioned_docs/version-v2.7.0/guides/signing.mdx similarity index 94% rename from website/versioned_docs/version-v2.5.0/guides/signing.mdx rename to website/versioned_docs/version-v2.7.0/guides/signing.mdx index b2a060c5f62..e57e99e76de 100644 --- a/website/versioned_docs/version-v2.5.0/guides/signing.mdx +++ b/website/versioned_docs/version-v2.7.0/guides/signing.mdx @@ -232,7 +232,7 @@ jobs: path: build/bin/* ``` -For code signing on macOS, [gon](https://github.com/mitchellh/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and +For code signing on macOS, [gon](https://github.com/Bearer/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and will be used in this guide. After the `Build Wails app` step, add the following to the workflow: @@ -241,7 +241,7 @@ After the `Build Wails app` step, add the following to the workflow: - name: MacOS download gon for code signing and app notarization if: matrix.platform == 'macos-latest' run: | - brew install mitchellh/gon/gon + brew install Bearer/tap/gon ``` Now we need to configure some gon config files in our `build/darwin` directory: @@ -254,19 +254,30 @@ Now we need to configure some gon config files in our `build/darwin` directory: "bundle_id": "app.myapp", "apple_id": { "username": "my-appleid@email.com", - "password": "@env:APPLE_PASSWORD" + "password": "@env:APPLE_PASSWORD", + "provider": "ABCDE12345" }, "sign": { - "application_identity": "Developer ID Application: My Name" + "application_identity": "Developer ID Application: Human User" } } ``` -Where `source` is your Wails binary, `bundle_id` is your bundle ID, `apple_id` contains your Apple ID username and App-Specific password -which you created earlier, and `sign.application_identity` is your identity which you can find by running the following command: +Here is a brief break down of the above fields: + +- `source`: The location of your wails binary to be signed +- `apple_id`: + - `username`: Your Apple ID email address + - `password`: Your app-specific password, referenced using Gon's environment variable syntax + - `provider`: Your team ID for your App Store Connect account +- `sign`: + - `application_identity`: Your Apple developer identity + +Your developer identity and team ID can both by found on macOS by running the following command: ```bash -security find-identity -v -p codesigning +$ security find-identity -v -p codesigning + 1) 00000000000000000000000000000000000000000 "Developer ID Application: Human User (ABCDE12345)" ``` 2. entitlements.plist: @@ -369,7 +380,7 @@ jobs: - name: MacOS download gon for code signing and app notarization if: matrix.platform == 'macos-latest' run: | - brew install mitchellh/gon/gon + brew install Bearer/tap/gon - name: Import Code-Signing Certificates for macOS if: matrix.platform == 'macos-latest' uses: Apple-Actions/import-codesign-certs@v1 diff --git a/website/versioned_docs/version-v2.7.0/guides/single-instance-lock.mdx b/website/versioned_docs/version-v2.7.0/guides/single-instance-lock.mdx new file mode 100644 index 00000000000..644e1e0d80c --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/single-instance-lock.mdx @@ -0,0 +1,80 @@ +# Single Instance Lock + +Single instance lock is a mechanism that allows you to prevent multiple instances of your app from running at the same time. +It is useful for apps that are designed to open files from the command line or from the OS file explorer. + +## Important + +Single Instance Lock does not implement a secure communications protocol between instances. When using single instance lock, +your app should treat any data passed to it from second instance callback as untrusted. +You should verify that args that you receive are valid and don't contain any malicious data. + +## How it works + +Windows: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via a shared window using [SendMessage](https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-sendmessage) +macOS: Single instance lock is implemented using a named mutex. The mutex name is generated from the unique id that you provide. Data is passed to the first instance via [NSDistributedNotificationCenter](https://developer.apple.com/documentation/foundation/nsdistributednotificationcenter) +Linux: Single instance lock is implemented using [dbus](https://www.freedesktop.org/wiki/Software/dbus/). The dbus name is generated from the unique id that you provide. Data is passed to the first instance via [dbus](https://www.freedesktop.org/wiki/Software/dbus/) + +## Usage +When creating your app, you can enable single instance lock by passing a `SingleInstanceLock` struct to the `App` struct. +Use the `UniqueId` field to specify a unique id for your app. +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. +The `OnSecondInstanceLaunch` field is used to specify a callback that is called when a second instance of your app is launched. +The callback receives a `SecondInstanceData` struct that contains the command line arguments passed to the second instance and the working directory of the second instance. + +Note that OnSecondInstanceLaunch don't trigger windows focus. +You need to call `runtime.WindowUnminimise` and `runtime.Show` to bring your app to the front. +Note that on linux systems window managers may prevent your app from being brought to the front to avoid stealing focus. + +```go title="main.go" +var wailsContext *context.Context + +// NewApp creates a new App application struct +func NewApp() *App { + return &App{} +} + +// startup is called when the app starts. The context is saved +// so we can call the runtime methods +func (a *App) startup(ctx context.Context) { + wailsContext = &ctx +} + +func (a *App) onSecondInstanceLaunch(secondInstanceData options.SecondInstanceData) { + secondInstanceArgs = secondInstanceData.Args + + println("user opened second instance", strings.Join(secondInstanceData.Args, ",")) + println("user opened second from", secondInstanceData.WorkingDirectory) + runtime.WindowUnminimise(*wailsContext) + runtime.Show(*wailsContext) + go runtime.EventsEmit(*wailsContext, "launchArgs", secondInstanceArgs) +} + +func main() { + // Create an instance of the app structure + app := NewApp() + + // Create application with options + err := wails.Run(&options.App{ + Title: "wails-open-file", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + BackgroundColour: &options.RGBA{R: 27, G: 38, B: 54, A: 1}, + OnStartup: app.startup, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "e3984e08-28dc-4e3d-b70a-45e961589cdc", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, + Bind: []interface{}{ + app, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` diff --git a/website/versioned_docs/version-v2.7.0/guides/sveltekit.mdx b/website/versioned_docs/version-v2.7.0/guides/sveltekit.mdx new file mode 100644 index 00000000000..e0357ca3c51 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/sveltekit.mdx @@ -0,0 +1,136 @@ +# SvelteKit + +This guide will go into: +1. Miminal Installation Steps - The steps needed to get a minimum Wails setup working for SvelteKit. +2. Install Script - Bash script for accomplishing the Minimal Installation Steps with optional Wails branding. +3. Important Notes - Issues that can be encountered when using SvelteKit + Wails and fixes. + +## 1. Minimal Installation Steps + +##### Install Wails for Svelte. +- `wails init -n myapp -t svelte` + +##### Delete the svelte frontend. +- Navigate into your newly created myapp folder. +- Delete the folder named "frontend" + +##### While in the Wails project root. Use your favorite package manager and install SvelteKit as the new frontend. Follow the prompts. +- `npm create svelte@latest frontend` + +##### Modify wails.json. +- Add `"wailsjsdir": "./frontend/src/lib",` Do note that this is where your Go and runtime functions will appear. +- Change your package manager frontend here if not using npm. + +##### Modify main.go. +- The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` + +##### Modify .gitignore +- The line `frontend/dist` needs to be replaced with `frontend/build` + +##### Install/remove dependencies using your favorite package manager. +- Navigate into your "frontend" folder. +- `npm i` +- `npm uninstall @sveltejs/adapter-auto` +- `npm i -D @sveltejs/adapter-static` + +##### Change adapter in svelte.config.js +- First line of file change `import adapter from '@sveltejs/adapter-auto';` to `import adapter from '@sveltejs/adapter-static';` + +##### Put SvelteKit into SPA mode with prerendering. +- Create a file under myapp/frontend/src/routes/ named +layout.ts/+layout.js. +- Add two lines into the newly created file `export const prerender = true` and `export const ssr = false` + +##### Test installation. +- Navigate back into the Wails project root (one directory up). +- run `wails dev` +- If the application doesn't run please check through the previous steps. + +## 2. Install Script + +##### This Bash Script does the steps listed above. Make sure to read over the script and understand what the script is doing on your computer. + +- Create a file sveltekit-wails.sh +- Copy the below code into the new file then save it. +- Make it executable with `chmod +x sveltekit-wails.sh` +- Brand is an optional param below that adds back in the wails branding. Leave third param blank to not insert the Wails branding. +- Example usage: `./sveltekit-wails.sh pnpm newapp brand` + +##### sveltekit-wails.sh: +``` +manager=$1 +project=$2 +brand=$3 +wails init -n $project -t svelte +cd $project +sed -i "s|npm|$manager|g" wails.json +sed -i 's|"auto",|"auto",\n "wailsjsdir": "./frontend/src/lib",|' wails.json +sed -i "s|all:frontend/dist|all:frontend/build|" main.go +if [[ -n $brand ]]; then + mv frontend/src/App.svelte +page.svelte + sed -i "s|'./assets|'\$lib/assets|" +page.svelte + sed -i "s|'../wails|'\$lib/wails|" +page.svelte + mv frontend/src/assets . +fi +rm -r frontend +$manager create svelte@latest frontend +if [[ -n $brand ]]; then + mv +page.svelte frontend/src/routes/+page.svelte + mkdir frontend/src/lib + mv assets frontend/src/lib/ +fi +cd frontend +$manager i +$manager uninstall @sveltejs/adapter-auto +$manager i -D @sveltejs/adapter-static +echo -e "export const prerender = true\nexport const ssr = false" > src/routes/+layout.ts +sed -i "s|-auto';|-static';|" svelte.config.js +cd .. +wails dev +``` + +## 3. Important Notes + +##### Server files will cause build failures. +- +layout.server.ts, +page.server.ts, +server.ts or any file with "server" in the name will fail to build as all routes are prerendered. + +##### The Wails runtime unloads with full page navigations! +- Anything that causes full page navigations: `window.location.href = '//'` or Context menu reload when using wails dev. What this means is that you can end up losing the ability to call any runtime breaking the app. There are two ways to work around this. +- Use `import { goto } from '$app/navigation'` then call `goto('//')` in your +page.svelte. This will prevent a full page navigation. +- If full page navigation can't be prevented the Wails runtime can be added to all pages by adding the below into the `` of myapp/frontend/src/app.html +``` + +... + + + +... + +``` +See https://wails.io/docs/guides/frontend for more information. + +##### Inital data can be loaded and refreshed from +page.ts/+page.js to +page.svelte. +- +page.ts/+page.js works well with load() https://kit.svelte.dev/docs/load#page-data +- invalidateAll() in +page.svelte will call load() from +page.ts/+page.js https://kit.svelte.dev/docs/load#rerunning-load-functions-manual-invalidation. + +##### Error Handling +- Expected errors using Throw error works in +page.ts/+page.js with a +error.svelte page. https://kit.svelte.dev/docs/errors#expected-errors +- Unexpected errors will cause the application to become unusable. Only recovery option (known so far) from unexpected errors is to reload the app. To do this create a file myapp/frontend/src/hooks.client.ts then add the below code to the file. +``` +import { WindowReloadApp } from '$lib/wailsjs/runtime/runtime' +export async function handleError() { + WindowReloadApp() +} +``` + +##### Using Forms and handling functions +- The simplest way is to call a function from the form is the standard, bind:value your variables and prevent submission `` +- The more advanced way is to use:enhance (progressive enhancement) which will allow for convenient access to formData, formElement, submitter. The important note is to always cancel() the form which prevents server side behavior. https://kit.svelte.dev/docs/form-actions#progressive-enhancement Example: +``` + { + cancel() + console.log(Object.fromEntries(formData)) + console.log(formElement) + console.log(submitter) + handle() +}}> +``` diff --git a/website/versioned_docs/version-v2.5.0/guides/templates.mdx b/website/versioned_docs/version-v2.7.0/guides/templates.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/templates.mdx rename to website/versioned_docs/version-v2.7.0/guides/templates.mdx diff --git a/website/versioned_docs/version-v2.7.0/guides/troubleshooting.mdx b/website/versioned_docs/version-v2.7.0/guides/troubleshooting.mdx new file mode 100644 index 00000000000..3bff4e44ffa --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/guides/troubleshooting.mdx @@ -0,0 +1,385 @@ +# Troubleshooting + +An assortment of troubleshooting tips. + +## The `wails` command appears to be missing? + +If your system is reporting that the `wails` command is missing, make sure you have followed the Go installation guide +correctly. Normally, it means that the `go/bin` directory in your User's home directory is not in the `PATH` environment +variable. You will also normally need to close and reopen any open command prompts so that changes to the environment +made by the installer are reflected at the command prompt. + +## My application is displaying a white/blank screen + +Check that your application includes the assets from the correct directory. In your `main.go` file, you will have +something similar to the following code: + +```go +//go:embed all:frontend/dist +var assets embed.FS +``` + +Check that `frontend/dist` contains your application assets. + +### Mac + +If this happens on Mac, try adding the following to your `Info.plist`: + +```xml +NSAppTransportSecurity + + NSAllowsLocalNetworking + + +``` + +Reference: https://github.com/wailsapp/wails/issues/1504#issuecomment-1174317433 + +## Mac application not valid + +If your built application looks like this in finder: + +```mdx-code-block +

+ +

+``` + +it's likely that your application's `info.plist` is invalid. Update the file in `build/.app/Contents/info.plist` +and check if the data is valid, EG check the binary name is correct. To persist the changes, copy the file back to +the `build/darwin` directory. + +## My application is not displaying the correct icon in Windows Explorer + +If your application is not displaying the correct icon, try deleting the hidden `IconCache.db` file located in the +`C:\Users\\AppData\Local` directory. This will force Windows to rebuild the icon cache. + +Source: https://github.com/wailsapp/wails/issues/2360#issuecomment-1556070036 + +## Cannot call backend method from frontend with variadic arguments + +If you have a backend method defined with variadic parameters, eg: + +```go +func (a *App) TestFunc(msg string, args ...interface{}) error { + // Code +} +``` + +calling this method from the frontend like this will fail: + +```js +var msg = "Hello: "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, ...args) + .then((result) => { + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Workaround: + +```js +var msg = "Hello "; +var args = ["Go", "JS"]; +window.go.main.App.TestFunc(msg, args) + .then((result) => { + //without the 3 dots + //do things here + }) + .catch((error) => { + //handle error + }); +``` + +Credit: https://github.com/wailsapp/wails/issues/1186 + +## I'm having getting proxy errors when trying to install Wails + +If you are getting errors like this: + +``` +"https://proxy.golang.org/github.com/wailsapp/wails/cmd/wails/@v/list": dial tcp 172.217.163.49:443: connectex: A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond. +``` + +it's probably because the official Go Proxy is being blocked (Users in China have reported this). +The solution is to set up the proxy manually, eg: + +``` +go env -w GO111MODULE=on +go env -w GOPROXY=https://goproxy.cn,direct +``` + +Source: https://github.com/wailsapp/wails/issues/1233 + +## The generated TypeScript doesn't have the correct types + +Sometimes the generated TypeScript doesn't have the correct types. To mitigate this, +it is possible to specify what types should be generated using the `ts_type` struct tag. For +more details, please read [this](https://github.com/tkrajina/typescriptify-golang-structs#custom-types). + +## When I navigate away from `index.html`, I am unable to call methods on the frontend + +If you navigate away from `index.html` to a new html file, the context will be lost. This can be fixed by adding +the following imports to the `` section of any new page you navigate to: + +```html + + + + +``` + +Source: https://github.com/wailsapp/wails/discussions/1512 + +## I get `too many open files` errors on my Mac when I run `wails dev` + +By default, macOS will only allow you to open a maximum of 256 files. This can affect the `wails dev` command. +This limit can be increased by running: `ulimit -n 1024` in the terminal. + +FSNotify is [looking to move to Apple's fsevents](https://github.com/fsnotify/fsnotify/issues/11) for Mac. +If this isn't completed soon, we will create our own implementation, tracked [here](https://github.com/wailsapp/wails/issues/1733). + +## My Mac app gives me weird compilation errors + +A few users have reported seeing compilation errors such as the following: + +```shell +# github.com/wailsapp/wails/v2/internal/frontend/desktop/darwin +In file included from ../../pkg/mod/github.com/wailsapp/wails/v2@v2.0.0-beta.44.2/internal/frontend/desktop/darwin/callbacks.go:9: +In file included from /Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/Foundation.h:12: +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSBundle.h:91:143: error: function does not return NSString +- (NSAttributedString *)localizedAttributedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1) NS_REFINED_FOR_SWIFT API_AVAILABLE(macos(12.0), ios(15.0), watchos(8.0), tvos(15.0)); + ~~~~~~~~~~~~~~ ^ ~ +/Library/Developer/CommandLineTools/SDKs/MacOSX12.1.sdk/System/Library/Frameworks/Foundation.framework/Headers/NSObjCRuntime.h:103:48: note: expanded from macro 'NS_FORMAT_ARGUMENT' + #define NS_FORMAT_ARGUMENT(A) __attribute__ ((format_arg(A))) +``` + +This is _normally_ due to a mismatch with the OS version you are running and the version of the XCode Command Line Tools +installed. If you see an error like this, try upgrading your XCode Command Line Tools to the latest version. + +If reinstalling Xcode Command Tools still fails, you can check the path where the toolkit is located using: + +`xcode-select -p` + +If `/Applications/Xcode.app/Contents/Developer` is displayed, run `sudo xcode-select --switch /Library/Developer/CommandLineTools` + +Sources: https://github.com/wailsapp/wails/issues/1806 and https://github.com/wailsapp/wails/issues/1140#issuecomment-1290446496 + +## My application won't compile on Mac + +If you are getting errors like this: + +```shell +l1@m2 GoEasyDesigner % go build -tags dev -gcflags "all=-N -l" +/Users/l1/sdk/go1.20.5/pkg/tool/darwin_arm64/link: running clang failed: exit status 1 +Undefined symbols for architecture arm64: + "_OBJC_CLASS_$_UTType", referenced from: + objc-class-ref in 000016.o +ld: symbol(s) not found for architecture arm64 +clang: error: linker command failed with exit code 1 (use -v to see invocation) +``` +Ensure you have the latest SDK installed. If so and you're still experiencing this issue, try the following: + +```shell +export CGO_LDFLAGS="-framework UniformTypeIdentifiers" && go build -tags dev -gcflags "all=-N -l" +``` + +Sources: https://github.com/wailsapp/wails/pull/2925#issuecomment-1726828562 + + +-- + +## Cannot start service: Host version "x.x.x does not match binary version "x.x.x" + +It's preferable to add `frontend/node_modules` and `frontend/package-lock.json` to your `.gitignore`. Otherwise when opening your repository on another machine +that may have different versions of Node installed, you may not be able to run your application. + +If this does happen, simply delete `frontend/node_modules` and `frontend/package-lock.json` and run your `wails build` or `wails dev` command again. + +## Build process stuck on "Generating bindings" + +Bindings generation process runs your application in a special mode. If application, intentionally or unintentionally, contains an endless loop (i.e. not exiting after `wails.Run()` finished), this can lead to build process stuck on the stage of bindings generation. Please make sure your code exits properly. + +## Mac application flashes white at startup + +This is due to the default background of the webview being white. If you want to use the window background colour instead, +you can make the webview background transparent using the following config: + +```go + err := wails.Run(&options.App{ + Title: "macflash", + Width: 1024, + Height: 768, + // Other settings + Mac: &mac.Options{ + WebviewIsTransparent: true, + }, + }) +``` + +## I get a "Microsoft Edge can't read or write to its data directory" error when running my program as admin on Windows + +You set your program to require admin permissions and it worked great! Unfortunately, some users are seeing a "Microsoft Edge can't read or write to its data directory" error when running it. + +When a Windows machine has two local accounts: + +- Alice, an admin +- Bob, a regular user + +Bob sees a UAC prompt when running your program. Bob enters Alice's admin credentials into this prompt. The app launches with admin permissions under Alice's account. + +Wails instructs WebView2 to store user data at the specified `WebviewUserDataPath`. It defaults to `%APPDATA%\[BinaryName.exe]`. + +Because the application is running under Alice's account, `%APPDATA%\[BinaryName.exe]` resolves to `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`. + +WebView2 [creates some child processes under Bob's logged-in account instead of Alice's admin account](https://github.com/MicrosoftEdge/WebView2Feedback/issues/932#issue-807464179). Since Bob cannot access `C:\Users\Alice\AppData\Roaming\[BinaryName.exe]`, the "Microsoft Edge can't read or write to its data directory" error is shown. + +Possible solution #1: + +Refactor your application to work without constant admin permissions. If you just need to perform a small set of admin tasks (such as running an updater), you can run your application with the minimum permissions and then use the `runas` command to run these tasks with admin permissions as needed: + +```go +//go:build windows + +package sample + +import ( + "golang.org/x/sys/windows" + "syscall" +) + +// Calling RunAs("C:\path\to\my\updater.exe") shows Bob a UAC prompt. Bob enters Alice's admin credentials. The updater launches with admin permissions under Alice's account. +func RunAs(path string) error { + verbPtr, _ := syscall.UTF16PtrFromString("runas") + exePtr, _ := syscall.UTF16PtrFromString(path) + cwdPtr, _ := syscall.UTF16PtrFromString("") + argPtr, _ := syscall.UTF16PtrFromString("") + + var showCmd int32 = 1 //SW_NORMAL + + err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) + if err != nil { + return err + } + return nil +} +``` + +Possible solution #2: + +Run your application with extended permissions. If you absolutely must run with constant admin permissions, WebView2 will function correctly if you use a data directory accessible by both users and you also launch your app with the `SeBackupPrivilege`, `SeDebugPrivilege`, and `SeRestorePrivilege` permissions. Here's an example: + +```go +package main + +import ( + "embed" + "os" + "runtime" + + "github.com/fourcorelabs/wintoken" + "github.com/hectane/go-acl" + "github.com/wailsapp/wails/v2" + "github.com/wailsapp/wails/v2/pkg/options" + "github.com/wailsapp/wails/v2/pkg/options/assetserver" + "github.com/wailsapp/wails/v2/pkg/options/windows" +) + +//go:embed all:frontend/dist +var assets embed.FS + +const ( + fixedTokenKey = "SAMPLE_RANDOM_KEY" + fixedTokenVal = "with-fixed-token" + webviewDir = "C:\\ProgramData\\Sample" +) + +func runWithFixedToken() { + println("Re-launching self") + token, err := wintoken.OpenProcessToken(0, wintoken.TokenPrimary) //pass 0 for own process + if err != nil { + panic(err) + } + defer token.Close() + + token.EnableTokenPrivileges([]string{ + "SeBackupPrivilege", + "SeDebugPrivilege", + "SeRestorePrivilege", + }) + + cmd := exec.Command(os.Args[0]) + cmd.Args = os.Args + cmd.Env = os.Environ() + cmd.Env = append(cmd.Env, fmt.Sprintf("%v=%v", fixedTokenKey, fixedTokenVal)) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.SysProcAttr = &syscall.SysProcAttr{Token: syscall.Token(token.Token())} + if err := cmd.Run(); err != nil { + println("Error after launching self:", err) + os.Exit(1) + } + println("Clean self launch :)") + os.Exit(0) +} + +func main() { + if runtime.GOOS == "windows" && os.Getenv(fixedTokenKey) != fixedTokenVal { + runWithFixedToken() + } + + println("Setting data dir to", webviewDir) + if err := os.MkdirAll(webviewDir, os.ModePerm); err != nil { + println("Failed creating dir:", err) + } + if err := acl.Chmod(webviewDir, 0777); err != nil { + println("Failed setting ACL on dir:", err) + } + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "sample-data-dir", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + }, + Windows: &windows.Options{ + WebviewUserDataPath: webviewDir, + }, + }) + + if err != nil { + println("Error:", err.Error()) + } +} +``` + +If you use a data directory accessible by both users but not the extended privileges, you will receive a WebView2 `80010108 The object invoked has disconnected from its clients` error. + +Possible future solution #3: [run WebView2 using an in-memory mode if implemented](https://github.com/MicrosoftEdge/WebView2Feedback/issues/3637#issuecomment-1728300982). + +## WebView2 installation succeeded, but the wails doctor command shows that it is not installed + +If you have installed WebView2, but the `wails doctor` command shows that it is not installed, it is likely that the +WebView2 runtime installed was for a different architecture. You can download the correct runtime from [here](https://developer.microsoft.com/en-us/microsoft-edge/webview2/). + +Source: https://github.com/wailsapp/wails/issues/2917 + +## WebVie2wProcess failed with kind + +If your Windows app generates this kind of error, you can check out what the error means [here](https://docs.microsoft.com/en-us/microsoft-edge/webview2/reference/winrt/microsoft_web_webview2_core/corewebview2processfailedkind?view=webview2-winrt-1.0.2045.28). + diff --git a/website/versioned_docs/version-v2.5.0/guides/vscode.mdx b/website/versioned_docs/version-v2.7.0/guides/vscode.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/vscode.mdx rename to website/versioned_docs/version-v2.7.0/guides/vscode.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/windows-installer.mdx b/website/versioned_docs/version-v2.7.0/guides/windows-installer.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/windows-installer.mdx rename to website/versioned_docs/version-v2.7.0/guides/windows-installer.mdx diff --git a/website/versioned_docs/version-v2.5.0/guides/windows.mdx b/website/versioned_docs/version-v2.7.0/guides/windows.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/guides/windows.mdx rename to website/versioned_docs/version-v2.7.0/guides/windows.mdx diff --git a/website/versioned_docs/version-v2.5.0/howdoesitwork.mdx b/website/versioned_docs/version-v2.7.0/howdoesitwork.mdx similarity index 93% rename from website/versioned_docs/version-v2.5.0/howdoesitwork.mdx rename to website/versioned_docs/version-v2.7.0/howdoesitwork.mdx index e9f2c6e3dc4..6e23d5eb94d 100644 --- a/website/versioned_docs/version-v2.5.0/howdoesitwork.mdx +++ b/website/versioned_docs/version-v2.7.0/howdoesitwork.mdx @@ -206,6 +206,57 @@ You may bind as many structs as you like. Just make sure you create an instance ``` +You may bind enums types as well. +In that case you should create array that will contain all possible enum values, instrument enum type and bind it to the app via `EnumBind`: + +```go {16-18} title="app.go" +type Weekday string + +const ( + Sunday Weekday = "Sunday" + Monday Weekday = "Monday" + Tuesday Weekday = "Tuesday" + Wednesday Weekday = "Wednesday" + Thursday Weekday = "Thursday" + Friday Weekday = "Friday" + Saturday Weekday = "Saturday" +) + +var AllWeekdays = []struct { + Value Weekday + TSName string +}{ + {Sunday, "SUNDAY"}, + {Monday, "MONDAY"}, + {Tuesday, "TUESDAY"}, + {Wednesday, "WEDNESDAY"}, + {Thursday, "THURSDAY"}, + {Friday, "FRIDAY"}, + {Saturday, "SATURDAY"}, +} +``` + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + EnumBind: []interface{}{ + AllWeekdays, + }, + }) + +``` + When you run `wails dev` (or `wails generate module`), a frontend module will be generated containing the following: - JavaScript bindings for all bound methods diff --git a/website/versioned_docs/version-v2.5.0/introduction.mdx b/website/versioned_docs/version-v2.7.0/introduction.mdx similarity index 97% rename from website/versioned_docs/version-v2.5.0/introduction.mdx rename to website/versioned_docs/version-v2.7.0/introduction.mdx index 1bfa099e80f..98f42806fa9 100644 --- a/website/versioned_docs/version-v2.5.0/introduction.mdx +++ b/website/versioned_docs/version-v2.7.0/introduction.mdx @@ -48,7 +48,7 @@ and TypeScript versions for each template. Wails uses a purpose built library for handling native elements such as Window, Menus, Dialogs, etc, so you can build good-looking, feature rich desktop applications. -**It does not embed a browser**, so it is resource efficient. Instead, it uses the native rendering engine for the +**It does not embed a browser**, so it delivers a small runtime. Instead, it reuses the native rendering engine for the platform. On Windows, this is the new Microsoft Webview2 library, built on Chromium. ### Go & JavaScript Interoperability diff --git a/website/versioned_docs/version-v2.7.0/reference/_category_.json b/website/versioned_docs/version-v2.7.0/reference/_category_.json new file mode 100644 index 00000000000..ebb337b8360 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/reference/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Reference", + "position": 40 +} diff --git a/website/versioned_docs/version-v2.5.0/reference/cli.mdx b/website/versioned_docs/version-v2.7.0/reference/cli.mdx similarity index 71% rename from website/versioned_docs/version-v2.5.0/reference/cli.mdx rename to website/versioned_docs/version-v2.7.0/reference/cli.mdx index 684dd88b4ad..bbfe5bea4c4 100644 --- a/website/versioned_docs/version-v2.5.0/reference/cli.mdx +++ b/website/versioned_docs/version-v2.7.0/reference/cli.mdx @@ -52,29 +52,35 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for `wails build` is used for compiling your project to a production-ready binary. -| Flag | Description | Default | -|:---------------------|:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|:----------------------------------------------------------------------------------------------------------------------------------------------| -| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | -| -clean | Cleans the `build/bin` directory | | -| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | -| -ldflags "flags" | Additional ldflags to pass to the compiler | | -| -nopackage | Do not package application | | -| -o filename | Output filename | | -| -s | Skip building the frontend | false | -| -f | Force build application | false | -| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | -| -upx | Compress final binary using "upx" | | -| -upxflags | Flags to pass to upx | | -| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | -| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | -| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | -| -debug | Retains debug information in the application. Allows the use of the devtools in the application window | false | -| -trimpath | Remove all file system paths from the resulting executable. | false | -| -race | Build with Go's race detector | false | -| -windowsconsole | Keep the console window for Windows builds | | -| -obfuscate | Obfuscate the application using [garble](https://github.com/burrowers/garble) | false | -| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | -| -nosyncgomod | Do not sync go.mod with the Wails version | false | +| Flag | Description | Default | +|:---------------------|:-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|:----------------------------------------------------------------------------------------------------------------------------------------------| +| -clean | Cleans the `build/bin` directory | | +| -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | +| -debug | Retains debug information in the application and shows the debug console. Allows the use of the devtools in the application window | | +| -devtools | Allows the use of the devtools in the application window in production (when -debug is not used). Ctrl/Cmd+Shift+F12 may be used to open the devtools window. *NOTE*: This option will make your application FAIL Mac appstore guidelines. Use for debugging only. | | +| -dryrun | Prints the build command without executing it | | +| -f | Force build application | | +| -garbleargs | Arguments to pass to garble | `-literals -tiny -seed=random` | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -m | Skip mod tidy before compile | | +| -nopackage | Do not package application | | +| -nocolour | Disable colour in output | | +| -nosyncgomod | Do not sync go.mod with the Wails version | | +| -nsis | Generate NSIS installer for Windows | | +| -o filename | Output filename | | +| -obfuscated | Obfuscate the application using [garble](https://github.com/burrowers/garble) | | +| -platform | Build for the given (comma delimited) [platforms](../reference/cli.mdx#platforms) eg. `windows/arm64`. Note, if you do not give the architecture, `runtime.GOARCH` is used. | platform = `GOOS` environment variable if given else `runtime.GOOS`.
arch = `GOARCH` envrionment variable if given else `runtime.GOARCH`. | +| -race | Build with Go's race detector | | +| -s | Skip building the frontend | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to Go compiler. Must be quoted. Space or comma (but not both) separated | | +| -trimpath | Remove all file system paths from the resulting executable. | | +| -u | Updates your project's `go.mod` to use the same version of Wails as the CLI | | +| -upx | Compress final binary using "upx" | | +| -upxflags | Flags to pass to upx | | +| -v int | Verbosity level (0 - silent, 1 - default, 2 - verbose) | 1 | +| -webview2 | WebView2 installer strategy: download,embed,browser,error | download | +| -windowsconsole | Keep the console window for Windows builds | | For a detailed description of the `webview2` flag, please refer to the [Windows](../guides/windows.mdx) Guide. @@ -164,37 +170,35 @@ Your system is ready for Wails development! - A webserver is started on `http://localhost:34115` which serves your application (not just frontend) over http. This allows you to use your favourite browser development extensions - All application assets are loaded from disk. If they are changed, the application will automatically reload (not rebuild). All connected browsers will also reload - A JS module is generated that provides the following: - - JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting - - TypeScript versions of your Go structs, that can be constructed and passed to your go methods +- JavaScript wrappers of your Go methods with autogenerated JSDoc, providing code hinting +- TypeScript versions of your Go structs, that can be constructed and passed to your go methods - A second JS module is generated that provides a wrapper + TS declaration for the runtime - On macOS, it will bundle the application into a `.app` file and run it. It will use a `build/darwin/Info.dev.plist` for development. | Flag | Description | Default | |:-----------------------------|:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|:----------------------| +| -appargs "args" | Arguments passed to the application in shell style | | | -assetdir "./path/to/assets" | Serve assets from the given directory instead of using the provided asset FS | Value in `wails.json` | | -browser | Opens a browser to `http://localhost:34115` on startup | | | -compiler "compiler" | Use a different go compiler to build, eg go1.15beta1 | go | -| -e | Extensions to trigger rebuilds (comma separated) | go | -| -reloaddirs | Additional directories to trigger reloads (comma separated) | Value in `wails.json` | -| -ldflags "flags" | Additional ldflags to pass to the compiler | | -| -tags "extra tags" | Build tags to pass to compiler (quoted and space separated) | | -| -loglevel "loglevel" | Loglevel to use - Trace, Debug, Info, Warning, Error | Debug | -| -nogoreload | Disable back end rebuilds on file change/create/delete. Useful for IDEs that autosave frequently. | | -| -noreload | Disable automatic reload when assets change | | -| -nocolour | Turn off colour cli output | false | -| -nogen | Disable generate module | | -| -v | Verbosity level (0 - silent, 1 - standard, 2 - verbose) | 1 | -| -wailsjsdir | The directory to generate the generated Wails JS modules | Value in `wails.json` | | -debounce | The time to wait for reload after an asset change is detected | 100 (milliseconds) | | -devserver "host:port" | The address to bind the wails dev server to | "localhost:34115" | +| -extensions | Extensions to trigger rebuilds (comma separated) | go | +| -forcebuild | Force build of application | | | -frontenddevserverurl "url" | Use 3rd party dev server url to serve assets, EG Vite | "" | -| -appargs "args" | Arguments passed to the application in shell style | | -| -save | Saves the given `assetdir`, `reloaddirs`, `wailsjsdir`, `debounce`, `devserver` and `frontenddevserverurl` flags in `wails.json` to become the defaults for subsequent invocations. | | +| -ldflags "flags" | Additional ldflags to pass to the compiler | | +| -loglevel "loglevel" | Loglevel to use - Trace, Debug, Info, Warning, Error | Debug | +| -nocolour | Turn off colour cli output | false | +| -noreload | Disable automatic reload when assets change | | +| -nosyncgomod | Do not sync go.mod with the Wails version | false | | -race | Build with Go's race detector | false | +| -reloaddirs | Additional directories to trigger reloads (comma separated) | Value in `wails.json` | | -s | Skip building the frontend | false | -| -nosyncgomod | Do not sync go.mod with the Wails version | false | - - +| -save | Saves the given `assetdir`, `reloaddirs`, `wailsjsdir`, `debounce`, `devserver` and `frontenddevserverurl` flags in `wails.json` to become the defaults for subsequent invocations. | | +| -skipbindings | Skip bindings generation | | +| -tags "extra tags" | Build tags to pass to compiler (quoted and space separated) | | +| -v | Verbosity level (0 - silent, 1 - standard, 2 - verbose) | 1 | +| -wailsjsdir | The directory to generate the generated Wails JS modules | Value in `wails.json` | Example: @@ -217,7 +221,7 @@ Wails uses templates for project generation. The `wails generate template` comma it may be used for generating projects. | Flag | Description | -| :--------------- | :------------------------------------------ | +|:-----------------|:--------------------------------------------| | -name | The template name (Mandatory) | | -frontend "path" | Path to frontend project to use in template | @@ -232,7 +236,7 @@ The `wails generate module` command allows you to manually generate the `wailsjs `wails update` will update the version of the Wails CLI. | Flag | Description | -| :----------------- | :------------------------------------ | +|:-------------------|:--------------------------------------| | -pre | Update to latest pre-release version | | -version "version" | Install a specific version of the CLI | diff --git a/website/versioned_docs/version-v2.5.0/reference/menus.mdx b/website/versioned_docs/version-v2.7.0/reference/menus.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/menus.mdx rename to website/versioned_docs/version-v2.7.0/reference/menus.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/options.mdx b/website/versioned_docs/version-v2.7.0/reference/options.mdx similarity index 82% rename from website/versioned_docs/version-v2.5.0/reference/options.mdx rename to website/versioned_docs/version-v2.7.0/reference/options.mdx index 4bc0d0ffd44..caa2d0f806c 100644 --- a/website/versioned_docs/version-v2.5.0/reference/options.mdx +++ b/website/versioned_docs/version-v2.7.0/reference/options.mdx @@ -51,12 +51,21 @@ func main() { OnBeforeClose: app.beforeClose, CSSDragProperty: "--wails-draggable", CSSDragValue: "drag", + EnableDefaultContextMenu: false, EnableFraudulentWebsiteDetection: false, ZoomFactor: 1.0, IsZoomControlEnabled: false, Bind: []interface{}{ app, }, + EnumBind: []interface{}{ + AllWeekdays, + }, + ErrorFormatter: func(err error) any { return err.Error() }, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "c9c8fd93-6758-4144-87d1-34bdb0a8bd60", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, Windows: &windows.Options{ WebviewIsTransparent: false, WindowIsTranslucent: false, @@ -90,6 +99,8 @@ func main() { FullSizeContent: false, UseToolbar: false, HideToolbarSeparator: true, + OnFileOpen: app.onFileOpen, + OnUrlOpen: app.onUrlOpen, }, Appearance: mac.NSAppearanceNameDarkAqua, WebviewIsTransparent: true, @@ -103,7 +114,8 @@ func main() { Linux: &linux.Options{ Icon: icon, WindowIsTranslucent: false, - WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + WebviewGpuPolicy: linux.WebviewGpuPolicyAlways, + ProgramName: "wails" }, Debug: options.Debug{ OpenInspectorOnStartup: false, @@ -417,6 +429,34 @@ Indicates what value the `CSSDragProperty` style should have to drag the window. Name: CSSDragValue
Type: `string` +### EnableDefaultContextMenu + +EnableDefaultContextMenu enables the browser's default context-menu in production. + +By default, the browser's default context-menu is only available in development and in a `-debug` [build](../reference/cli.mdx#build) along with the devtools inspector, Using this option you can enable the default context-menu in `production` while the devtools inspector won't be available unless the `-devtools` build flag is used. + +When this option is enabled, by default the context-menu will only be shown for text contexts (where Cut/Copy/Paste is needed), to override this behavior, you can use the CSS property `--default-contextmenu` on any HTML element (including the `body`) with the following values : + +| CSS Style | Behavior | +|--------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------| +| `--default-contextmenu: auto;` | (**default**) will show the default context menu only if :
contentEditable is true OR text has been selected OR element is input or textarea | +| `--default-contextmenu: show;` | will always show the default context menu | +| `--default-contextmenu: hide;` | will always hide the default context menu | + +This rule is inherited like any normal CSS rule, so nesting works as expected. + +:::note +This filtering functionality is only enabled in production, so in development and in debug build, the full context-menu is always available everywhere. +::: + +:::warning +This filtering functionality is NOT a security measure, the developer should expect that the full context-menu could be leaked anytime which could contain commands like (Download image, Reload, Save webpage), if this is a concern, the developer SHOULD NOT enable the default context-menu. +::: + + +Name: EnableDefaultContextMenu
+Type: `bool` + ### EnableFraudulentWebsiteDetection EnableFraudulentWebsiteDetection enables scan services for fraudulent content, such as malware or phishing attempts. @@ -448,6 +488,43 @@ A slice of struct instances defining methods that need to be bound to the fronte Name: Bind
Type: `[]interface{}` +### EnumBind + +A slice of Enum arrays that need to be bound to the frontend. + +Name: EnumBind
+Type: `[]interface{}` + +### ErrorFormatter + +A function that determines how errors are formatted when returned by a JS-to-Go +method call. The returned value will be marshalled as JSON. + +Name: ErrorFormatter
+Type: `func (error) any` + +### SingleInstanceLock + +Enables single instance locking. This means that only one instance of your application can be running at a time. + +Name: SingleInstanceLock
+Type: `*options.SingleInstanceLock` + +#### UniqueId + +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. + +Name: UniqueId
+Type: `string` + +#### OnSecondInstanceLaunch + +Callback that is called when a second instance of your app is launched. + +Name: OnSecondInstanceLaunch
+Type: `func(secondInstanceData SecondInstanceData)` + + ### Windows This defines [Windows specific options](#windows). @@ -648,6 +725,13 @@ Setting this to `true` will disable GPU hardware acceleration for the webview. Name: WebviewGpuIsDisabled
Type: `bool` +#### EnableSwipeGestures + +Setting this to `true` will enable swipe gestures for the webview. + +Name: EnableSwipeGestures
+Type: `bool` + ### Mac This defines [Mac specific options](#mac). @@ -751,6 +835,57 @@ with [WebviewIsTransparent](#WebviewIsTransparent) to make frosty-looking applic Name: WindowIsTranslucent
Type: `bool` +#### OnFileOpen + +Callback that is called when a file is opened with the application. + +Name: OnFileOpen
+Type: `func(filePath string)` + +#### OnUrlOpen + +Callback that is called when a URL is opened with the application. + +Name: OnUrlOpen
+Type: `func(filePath string)` + +#### Preferences + +The Preferences struct provides the ability to configure the Webview preferences. + +Name: Preferences
+Type: [`*mac.Preferences`](#preferences-struct) + +##### Preferences struct + +You can specify the webview preferences. + +```go +type Preferences struct { + TabFocusesLinks u.Bool + TextInteractionEnabled u.Bool + FullscreenEnabled u.Bool +} +``` + +| Name | Description | +| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| TabFocusesLinks | A Boolean value that indicates whether pressing the tab key changes the focus to links and form controls. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/2818595-tabfocuseslinks?language=objc) | +| TextInteractionEnabled | A Boolean value that indicates whether to allow people to select or otherwise interact with text. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3727362-textinteractionenabled?language=objc) | +| FullscreenEnabled | A Boolean value that indicates whether a web view can display content full screen. [Apple Docs](https://developer.apple.com/documentation/webkit/wkpreferences/3917769-elementfullscreenenabled?language=objc) | + +Example: + +```go +Mac: &mac.Options{ + Preferences: &mac.Preferences{ + TabFocusesLinks: mac.Enabled, + TextInteractionEnabled: mac.Disabled, + FullscreenEnabled: mac.Enabled, + } +} +``` + #### About This configuration lets you set the title, message and icon for the "About" menu item in the app menu created by the "AppMenu" role. @@ -860,6 +995,17 @@ Default: `WebviewGpuPolicyAlways` | WebviewGpuPolicyOnDemand | Hardware acceleration is enabled/disabled as request by web contents| | WebviewGpuPolicyNever | Hardware acceleration is always disabled | +#### ProgramName + +This option is used to set the program's name for the window manager via GTK's g_set_prgname(). +This name should not be localized, [see the docs](https://docs.gtk.org/glib/func.set_prgname.html). + +When a .desktop file is created this value helps with window grouping and desktop icons when the .desktop file's `Name` +property differs form the executable's filename. + +Name: ProgramName
+Type: string
+ ### Debug This defines [Debug specific options](#Debug) that apply to debug builds. diff --git a/website/versioned_docs/version-v2.7.0/reference/project-config.mdx b/website/versioned_docs/version-v2.7.0/reference/project-config.mdx new file mode 100644 index 00000000000..3a6f09495a0 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/reference/project-config.mdx @@ -0,0 +1,130 @@ +--- +sidebar_position: 5 +--- + +# Project Config + +The project config resides in the `wails.json` file in the project directory. The structure of the config is: + +```json5 +{ + // Project config version + "version": "", + // The project name + "name": "", + // Relative path to the directory containing the compiled assets, this is normally inferred and could be left empty + "assetdir": "", + // Additional directories to trigger reloads (comma separated), this is only used for some advanced asset configurations + "reloaddirs": "", + // The directory where the build files reside. Defaults to 'build' + "build:dir": "", + // Relative path to the frontend directory. Defaults to 'frontend' + "frontend:dir": "", + // The command to install node dependencies, run in the frontend directory - often `npm install` + "frontend:install": "", + // The command to build the assets, run in the frontend directory - often `npm run build` + "frontend:build": "", + // This command has been replaced by frontend:dev:build. If frontend:dev:build is not specified will falls back to this command. \nIf this command is also not specified will falls back to frontend:build + "frontend:dev": "", + // This command is the dev equivalent of frontend:build. If not specified falls back to frontend:dev + "frontend:dev:build": "", + // This command is the dev equivalent of frontend:install. If not specified falls back to frontend:install + "frontend:dev:install": "", + // This command is run in a separate process on `wails dev`. Useful for 3rd party watchers or starting 3d party dev servers + "frontend:dev:watcher": "", + // URL to a 3rd party dev server to be used to serve assets, EG Vite. \nIf this is set to 'auto' then the devServerUrl will be inferred from the Vite output + "frontend:dev:serverUrl": "", + // Relative path to the directory that the auto-generated JS modules will be created + "wailsjsdir": "", + // The name of the binary + "outputfilename": "", + // The default time the dev server waits to reload when it detects a change in assets + "debounceMS": 100, + // Address to bind the wails dev sever to. Default: localhost:34115 + "devServer": "", + // Arguments passed to the application in shell style when in dev mode + "appargs": "", + // Defines if build hooks should be run though they are defined for an OS other than the host OS. + "runNonNativeBuildHooks": false, + "preBuildHooks": { + // The command that will be executed before a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed before a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH". The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed before every build: ${platform} is replaced with the "GOOS/GOARCH". + "*/*": "" + }, + "postBuildHooks": { + // The command that will be executed after a build of the specified GOOS/GOARCH: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/GOARCH" hook is executed before the "GOOS/*" and "*/*" hook. + "GOOS/GOARCH": "", + // The command that will be executed after a build of the specified GOOS: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. The "GOOS/*" hook is executed before the "*/*" hook. + "GOOS/*": "", + // The command that will be executed after every build: ${platform} is replaced with the "GOOS/GOARCH" and ${bin} with the path to the compiled binary. + "*/*": "" + }, + // Data used to populate manifests and version info. + "info": { + // The company name. Default: [The project name] + "companyName": "", + // The product name. Default: [The project name] + "productName": "", + // The version of the product. Default: '1.0.0' + "productVersion": "", + // The copyright of the product. Default: 'Copyright.........' + "copyright": "", + // A short comment of the app. Default: 'Built using Wails (https://wails.app)' + "comments": "", + // File associations for the app + "fileAssociations": [ + { + // The extension (minus the leading period). e.g. png + "ext": "wails", + // The name. e.g. PNG File + "name": "Wails", + // Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. + "description": "Wails file", + // The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows) + "iconName": "fileIcon", + // macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. + "role": "Editor" + }, + ], + // Custom URI protocols that should be opened by the application + "protocols": [ + { + // protocol scheme. e.g. myapp + "scheme": "myapp", + // Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. + "description": "Myapp protocol", + // macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. + "role": "Editor" + } + ] + }, + // 'multiple': One installer per architecture. 'single': Single universal installer for all architectures being built. Default: 'multiple' + "nsisType": "", + // Whether the app should be obfuscated. Default: false + "obfuscated": "", + // The arguments to pass to the garble command when using the obfuscated flag + "garbleargs": "", + // Bindings configurations + "bindings": { + // model.ts file generation config + "ts_generation": { + // All generated JavaScript entities will be prefixed with this value + "prefix": "", + // All generated JavaScript entities will be suffixed with this value + "suffix": "", + // Type of output to generate (classes|interfaces) + "outputType": "classes", + } + } +} +``` + +This file is read by the Wails CLI when running `wails build` or `wails dev`. + +The `assetdir`, `reloaddirs`, `wailsjsdir`, `debounceMS`, `devserver` and `frontenddevserverurl` flags in `wails build/dev` will update the project config +and thus become defaults for subsequent runs. + +The JSON Schema for this file is located [here](https://wails.io/schemas/config.v2.json). diff --git a/website/versioned_docs/version-v2.7.0/reference/runtime/_category_.json b/website/versioned_docs/version-v2.7.0/reference/runtime/_category_.json new file mode 100644 index 00000000000..ac6d5548816 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/reference/runtime/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Runtime", + "position": 1 +} diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/browser.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/browser.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/browser.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/browser.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/clipboard.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/clipboard.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/clipboard.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/clipboard.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/dialog.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/dialog.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/dialog.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/dialog.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/events.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/events.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/events.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/events.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/intro.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/intro.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/intro.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/intro.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/log.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/log.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/log.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/log.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/menu.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/menu.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/menu.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/menu.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/screen.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/screen.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/reference/runtime/screen.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/screen.mdx diff --git a/website/versioned_docs/version-v2.5.0/reference/runtime/window.mdx b/website/versioned_docs/version-v2.7.0/reference/runtime/window.mdx similarity index 94% rename from website/versioned_docs/version-v2.5.0/reference/runtime/window.mdx rename to website/versioned_docs/version-v2.7.0/reference/runtime/window.mdx index 3719be554dd..69ed1fe2b1f 100644 --- a/website/versioned_docs/version-v2.5.0/reference/runtime/window.mdx +++ b/website/versioned_docs/version-v2.7.0/reference/runtime/window.mdx @@ -117,7 +117,7 @@ JS: `WindowIsNormal() bool` Sets the width and height of the window. Go: `WindowSetSize(ctx context.Context, width int, height int)`
-JS: `WindowSetSize(size: Size)` +JS: `WindowSetSize(width: number, height: number)` ### WindowGetSize @@ -134,7 +134,7 @@ Will resize the window if the window is currently smaller than the given dimensi Setting a size of `0,0` will disable this constraint. Go: `WindowSetMinSize(ctx context.Context, width int, height int)`
-JS: `WindowSetMinSize(size: Size)` +JS: `WindowSetMinSize(width: number, height: number)` ### WindowSetMaxSize @@ -144,7 +144,7 @@ Will resize the window if the window is currently larger than the given dimensio Setting a size of `0,0` will disable this constraint. Go: `WindowSetMaxSize(ctx context.Context, width int, height int)`
-JS: `WindowSetMaxSize(size: Size)` +JS: `WindowSetMaxSize(width: number, height: number)` ### WindowSetAlwaysOnTop @@ -158,7 +158,7 @@ JS: `WindowSetAlwaysOnTop(b: Boolen)` Sets the window position relative to the monitor the window is currently on. Go: `WindowSetPosition(ctx context.Context, x int, y int)`
-JS: `WindowSetPosition(position: Position)` +JS: `WindowSetPosition(x: number, y: number)` ### WindowGetPosition @@ -233,6 +233,13 @@ Any value that is not 0 will be considered 255. Go: `WindowSetBackgroundColour(ctx context.Context, R, G, B, A uint8)`
JS: `WindowSetBackgroundColour(R, G, B, A)` +### WindowPrint + +Opens tha native print dialog. + +Go: `WindowPrint(ctx context.Context)`
+JS: `WindowPrint()` + ## TypeScript Object Definitions ### Position diff --git a/website/versioned_docs/version-v2.7.0/tutorials/_category_.json b/website/versioned_docs/version-v2.7.0/tutorials/_category_.json new file mode 100644 index 00000000000..dfac1d175e6 --- /dev/null +++ b/website/versioned_docs/version-v2.7.0/tutorials/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Tutorials", + "position": 70 +} diff --git a/website/versioned_docs/version-v2.5.0/tutorials/dogsapi.mdx b/website/versioned_docs/version-v2.7.0/tutorials/dogsapi.mdx similarity index 100% rename from website/versioned_docs/version-v2.5.0/tutorials/dogsapi.mdx rename to website/versioned_docs/version-v2.7.0/tutorials/dogsapi.mdx diff --git a/website/versioned_docs/version-v2.5.0/tutorials/helloworld.mdx b/website/versioned_docs/version-v2.7.0/tutorials/helloworld.mdx similarity index 99% rename from website/versioned_docs/version-v2.5.0/tutorials/helloworld.mdx rename to website/versioned_docs/version-v2.7.0/tutorials/helloworld.mdx index c7d4d01c59b..f9e789bb10a 100644 --- a/website/versioned_docs/version-v2.5.0/tutorials/helloworld.mdx +++ b/website/versioned_docs/version-v2.7.0/tutorials/helloworld.mdx @@ -73,6 +73,7 @@ App Type: desktop Platforms: windows/amd64 Compiler: C:\Users\leaan\go\go1.18.3\bin\go.exe Build Mode: Production +Devtools: false Skip Frontend: false Compress: false Package: true diff --git a/website/versioned_sidebars/version-v2.5.0-sidebars.json b/website/versioned_sidebars/version-v2.7.0-sidebars.json similarity index 100% rename from website/versioned_sidebars/version-v2.5.0-sidebars.json rename to website/versioned_sidebars/version-v2.7.0-sidebars.json diff --git a/website/versions.json b/website/versions.json index 1bbe686c8d5..1e10a413099 100644 --- a/website/versions.json +++ b/website/versions.json @@ -1 +1 @@ -["v2.6.0","v2.5.0"] \ No newline at end of file +["v2.7.0","v2.6.0"] \ No newline at end of file From 73b547a730edb87038bfc55996789850488bd053 Mon Sep 17 00:00:00 2001 From: YOSHIKI HIDAKA <71998099+radish-miyazaki@users.noreply.github.com> Date: Sat, 9 Dec 2023 18:43:08 +0900 Subject: [PATCH 63/87] update Star History Chart url (#3106) --- README.ja.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.ja.md b/README.ja.md index 8d29c7d1cc2..09c1c9bbe69 100644 --- a/README.ja.md +++ b/README.ja.md @@ -120,7 +120,7 @@ Wails では Go のコードとウェブフロントエンドを単一のバイ ## スター数の推移 -[![スター数の推移](https://starchart.cc/wailsapp/wails.svg)](https://starchart.cc/wailsapp/wails) +[![Star History Chart](https://api.star-history.com/svg?repos=wailsapp/wails&type=Date)](https://star-history.com/#wailsapp/wails&Date) ## コントリビューター From f517c8b9803f04de2035505b7873884390258537 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sat, 9 Dec 2023 20:45:02 +1100 Subject: [PATCH 64/87] Update README.md Use updated starchart --- README.md | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 6d4427c03b9..b46aea19d66 100644 --- a/README.md +++ b/README.md @@ -122,7 +122,13 @@ This project is supported by these kind people / companies: ## Stargazers over time -[![Star History Chart](https://api.star-history.com/svg?repos=wailsapp/wails&type=Date)](https://star-history.com/#wailsapp/wails&Date) + + + + + Star History Chart + + ## Contributors From fd3d1801ed07ce7c49b718b06e475227344b91f5 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 07:32:48 +1100 Subject: [PATCH 65/87] Run full check on bugfix branches --- .github/workflows/build-and-test.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index f208e446a10..671712e0794 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -2,7 +2,7 @@ name: Build + Test v2 on: push: - branches: [release/*, master] + branches: [release/*, master, bugfix/*] workflow_dispatch: jobs: From 5d9fe49cf92ee3719b46e1f043234016e56dc66c Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 07:33:53 +1100 Subject: [PATCH 66/87] Include macos-11 build --- .github/workflows/build-and-test.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index 671712e0794..84325d10cd4 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -12,7 +12,7 @@ jobs: runs-on: ${{ matrix.os }} strategy: matrix: - os: [ubuntu-latest, windows-latest, macos-latest] + os: [ubuntu-latest, windows-latest, macos-latest, macos-11] go-version: [1.18, 1.19] steps: From e9cc68fe3a16a9864234d0dd643276f725df9beb Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 07:34:49 +1100 Subject: [PATCH 67/87] Update to Go 1.20+1.21 --- .github/workflows/build-and-test.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index 84325d10cd4..bad46f7d0ed 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -13,7 +13,7 @@ jobs: strategy: matrix: os: [ubuntu-latest, windows-latest, macos-latest, macos-11] - go-version: [1.18, 1.19] + go-version: [1.20, 1.21] steps: - name: Checkout code From c3d57d05c3eff66928b5d9ee747441e0a9f296c0 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 07:40:00 +1100 Subject: [PATCH 68/87] Update github.com/tc-hib/winres to v0.2.1 --- v2/go.mod | 8 +++++--- v2/go.sum | 4 ++++ 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/v2/go.mod b/v2/go.mod index 88bd7c75cda..61384b9edd7 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -1,6 +1,8 @@ module github.com/wailsapp/wails/v2 -go 1.18 +go 1.21 + +toolchain go1.21.0 require ( github.com/Masterminds/semver v1.5.0 @@ -33,7 +35,7 @@ require ( github.com/sabhiram/go-gitignore v0.0.0-20210923224102-525f6e181f06 github.com/samber/lo v1.38.1 github.com/stretchr/testify v1.8.1 - github.com/tc-hib/winres v0.1.5 + github.com/tc-hib/winres v0.2.1 github.com/tidwall/sjson v1.1.7 github.com/tkrajina/go-reflector v0.5.6 github.com/wailsapp/go-webview2 v1.0.10 @@ -94,7 +96,7 @@ require ( github.com/yuin/goldmark-emoji v1.0.1 // indirect golang.org/x/crypto v0.14.0 // indirect golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 // indirect - golang.org/x/image v0.10.0 // indirect + golang.org/x/image v0.12.0 // indirect golang.org/x/term v0.13.0 // indirect golang.org/x/text v0.13.0 // indirect gopkg.in/warnings.v0 v0.1.2 // indirect diff --git a/v2/go.sum b/v2/go.sum index 7e4691d0b95..6b9ebb71d5c 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -210,6 +210,8 @@ github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKs github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= github.com/tc-hib/winres v0.1.5 h1:2dA5yfjdoEA3UyRaOC92HNMt3jap66pLzoW4MjpC/0M= github.com/tc-hib/winres v0.1.5/go.mod h1:pe6dOR40VOrGz8PkzreVKNvEKnlE8t4yR8A8naL+t7A= +github.com/tc-hib/winres v0.2.1 h1:YDE0FiP0VmtRaDn7+aaChp1KiF4owBiJa5l964l5ujA= +github.com/tc-hib/winres v0.2.1/go.mod h1:C/JaNhH3KBvhNKVbvdlDWkbMDO9H4fKKDaN7/07SSuk= github.com/tidwall/gjson v1.8.0/go.mod h1:5/xDoumyyDNerp2U36lyolv46b3uF/9Bu6OfyQ9GImk= github.com/tidwall/gjson v1.9.3 h1:hqzS9wAHMO+KVBBkLxYdkEeeFHuqr95GfClRLKlgK0E= github.com/tidwall/gjson v1.9.3/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= @@ -258,6 +260,8 @@ golang.org/x/image v0.0.0-20200430140353-33d19683fad8/go.mod h1:FeLwcggjj3mMvU+o golang.org/x/image v0.0.0-20201208152932-35266b937fa6/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= golang.org/x/image v0.10.0 h1:gXjUUtwtx5yOE0VKWq1CH4IJAClq4UGgUA3i+rpON9M= golang.org/x/image v0.10.0/go.mod h1:jtrku+n79PfroUbvDdeUWMAI+heR786BofxrbiSF+J0= +golang.org/x/image v0.12.0 h1:w13vZbU4o5rKOFFR8y7M+c4A5jXDC0uXTdHYRP8X2DQ= +golang.org/x/image v0.12.0/go.mod h1:Lu90jvHG7GfemOIcldsh9A2hS01ocl6oNO7ype5mEnk= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/mod v0.12.0 h1:rmsUpXtvNzj340zd98LZ4KntptpfRHwpFOHG188oHXc= From 975c55a3d5f436ae53eb5ca1067c461be9523715 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 10:12:14 +1100 Subject: [PATCH 69/87] Add better macos guard for elementFullscreenEnabled (#3111) * Add better macos guard for elementFullscreenEnabled * Disable go linters as they constantly error * Run full check on bugfix branches * Update to Go 1.20+1.21 * Update github.com/tc-hib/winres to v0.2.1 * Update setup-go to v4 * Try fix for Go 1.20 * Fix go.mod * Update go sum * Revert to builds on Go 1.18 + 1.19 * Update Go version to 1.19 for all workflows and modules The Go version is updated to 1.19 across all GitHub Actions workflows and go.mod files. All builds and tests will now only run on Go 1.19, simplifying our build matrix, and ensuring we're testing on the latest stable Go version. * Update build-and-test workflow for MacOS version and Go version The build-and-test workflow has been updated to run tests on 'macos-11' in addition to 'macos-latest'. Furthermore, Go version for the tests has been set to '1.19' only. * Update actions versions * Move to go 1.20 and improve caching * Move to go 1.20 and improve caching * Add additional guards --- .github/workflows/build-and-test.yml | 20 ++++---- .github/workflows/pr.yml | 46 +++++++++---------- v2/go.mod | 4 +- v2/go.sum | 6 --- .../frontend/desktop/darwin/Application.m | 3 +- .../frontend/desktop/darwin/WailsContext.m | 25 +++++++--- .../frontend/desktop/darwin/inspector_dev.go | 3 +- 7 files changed, 58 insertions(+), 49 deletions(-) diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index bad46f7d0ed..163076e875c 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -13,30 +13,31 @@ jobs: strategy: matrix: os: [ubuntu-latest, windows-latest, macos-latest, macos-11] - go-version: [1.20, 1.21] + go-version: ['1.20'] steps: - name: Checkout code - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Install linux dependencies if: matrix.os == 'ubuntu-latest' run: sudo apt-get update -y && sudo apt-get install -y libgtk-3-dev libwebkit2gtk-4.0-dev build-essential pkg-config - name: Setup Go - uses: actions/setup-go@v3 + uses: actions/setup-go@v4 with: go-version: ${{ matrix.go-version }} + cache-dependency-path: ./v2/go.sum - name: Run tests (mac) - if: matrix.os == 'macos-latest' + if: matrix.os == 'macos-latest' || matrix.os == 'macos-11' env: CGO_LDFLAGS: -framework UniformTypeIdentifiers -mmacosx-version-min=10.13 working-directory: ./v2 run: go test -v ./... - name: Run tests (!mac) - if: matrix.os != 'macos-latest' + if: matrix.os != 'macos-latest' && matrix.os != 'macos-11' working-directory: ./v2 run: go test -v ./... @@ -50,7 +51,7 @@ jobs: steps: - name: Checkout code - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Use Node.js ${{ matrix.node-version }} uses: actions/setup-node@v3 @@ -89,15 +90,16 @@ jobs: vanilla-ts, plain, ] - go-version: [1.18, 1.19] + go-version: ['1.20'] steps: - name: Checkout - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Setup Go - uses: actions/setup-go@v3 + uses: actions/setup-go@v4 with: go-version: ${{ matrix.go-version }} + cache-dependency-path: ./v2/go.sum - name: Build Wails CLI run: | diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml index 46acb8ee434..55bed637aae 100644 --- a/.github/workflows/pr.yml +++ b/.github/workflows/pr.yml @@ -26,28 +26,28 @@ jobs: run: | echo "::warning::Feature branch does not contain any changes to the website." - lint_go: - name: Run Go Linters - runs-on: ubuntu-latest - steps: - - name: Checkout code - uses: actions/checkout@v4 - - - name: Setup Go - uses: actions/setup-go@v4 - with: - go-version: "1.21" - - - name: Update go modules - working-directory: ./v2 - run: go mod tidy - - - name: Run Linter - uses: golangci/golangci-lint-action@v3 - with: - version: v1.54 - working-directory: ./v2 - args: --timeout=10m0s --config ./.golangci.yml +# lint_go: +# name: Run Go Linters +# runs-on: ubuntu-latest +# steps: +# - name: Checkout code +# uses: actions/checkout@v4 +# +# - name: Setup Go +# uses: actions/setup-go@v4 +# with: +# go-version: "1.21" +# +# - name: Update go modules +# working-directory: ./v2 +# run: go mod tidy +# +# - name: Run Linter +# uses: golangci/golangci-lint-action@v3 +# with: +# version: v1.54 +# working-directory: ./v2 +# args: --timeout=10m0s --config ./.golangci.yml test_go: name: Run Go Tests @@ -56,7 +56,7 @@ jobs: strategy: matrix: os: [ubuntu-latest, windows-latest, macos-latest] - go-version: [1.18, 1.19] + go-version: [1.19] steps: - name: Checkout code diff --git a/v2/go.mod b/v2/go.mod index 61384b9edd7..a2d0ab416b8 100644 --- a/v2/go.mod +++ b/v2/go.mod @@ -1,8 +1,6 @@ module github.com/wailsapp/wails/v2 -go 1.21 - -toolchain go1.21.0 +go 1.20 require ( github.com/Masterminds/semver v1.5.0 diff --git a/v2/go.sum b/v2/go.sum index 6b9ebb71d5c..fdc18d024f6 100644 --- a/v2/go.sum +++ b/v2/go.sum @@ -208,8 +208,6 @@ github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/ github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= -github.com/tc-hib/winres v0.1.5 h1:2dA5yfjdoEA3UyRaOC92HNMt3jap66pLzoW4MjpC/0M= -github.com/tc-hib/winres v0.1.5/go.mod h1:pe6dOR40VOrGz8PkzreVKNvEKnlE8t4yR8A8naL+t7A= github.com/tc-hib/winres v0.2.1 h1:YDE0FiP0VmtRaDn7+aaChp1KiF4owBiJa5l964l5ujA= github.com/tc-hib/winres v0.2.1/go.mod h1:C/JaNhH3KBvhNKVbvdlDWkbMDO9H4fKKDaN7/07SSuk= github.com/tidwall/gjson v1.8.0/go.mod h1:5/xDoumyyDNerp2U36lyolv46b3uF/9Bu6OfyQ9GImk= @@ -257,9 +255,6 @@ golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 h1:k/i9J1pBpvlfR+9QsetwPyERsqu1GIbi967PQMq3Ivc= golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w= golang.org/x/image v0.0.0-20200430140353-33d19683fad8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= -golang.org/x/image v0.0.0-20201208152932-35266b937fa6/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= -golang.org/x/image v0.10.0 h1:gXjUUtwtx5yOE0VKWq1CH4IJAClq4UGgUA3i+rpON9M= -golang.org/x/image v0.10.0/go.mod h1:jtrku+n79PfroUbvDdeUWMAI+heR786BofxrbiSF+J0= golang.org/x/image v0.12.0 h1:w13vZbU4o5rKOFFR8y7M+c4A5jXDC0uXTdHYRP8X2DQ= golang.org/x/image v0.12.0/go.mod h1:Lu90jvHG7GfemOIcldsh9A2hS01ocl6oNO7ype5mEnk= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= @@ -320,7 +315,6 @@ golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= -golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= diff --git a/v2/internal/frontend/desktop/darwin/Application.m b/v2/internal/frontend/desktop/darwin/Application.m index aa35b2a6bea..c428b4ce628 100644 --- a/v2/internal/frontend/desktop/darwin/Application.m +++ b/v2/internal/frontend/desktop/darwin/Application.m @@ -396,7 +396,7 @@ void ReleaseContext(void *inctx) { // Credit: https://stackoverflow.com/q/33319295 void WindowPrint(void *inctx) { - // Check if macOS 11.0 or newer +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 110000 if (@available(macOS 11.0, *)) { ON_MAIN_THREAD( WailsContext *ctx = (__bridge WailsContext*) inctx; @@ -424,4 +424,5 @@ void WindowPrint(void *inctx) { [po runOperationModalForWindow:ctx.mainWindow delegate:ctx.mainWindow.delegate didRunSelector:nil contextInfo:nil]; ) } +#endif } diff --git a/v2/internal/frontend/desktop/darwin/WailsContext.m b/v2/internal/frontend/desktop/darwin/WailsContext.m index bf191e4725f..fd15465a836 100644 --- a/v2/internal/frontend/desktop/darwin/WailsContext.m +++ b/v2/internal/frontend/desktop/darwin/WailsContext.m @@ -219,23 +219,27 @@ - (void) CreateWindow:(int)width :(int)height :(bool)frameless :(bool)resizable config.preferences.tabFocusesLinks = *preferences.tabFocusesLinks; } +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 110300 if (@available(macOS 11.3, *)) { if (preferences.textInteractionEnabled != NULL) { config.preferences.textInteractionEnabled = *preferences.textInteractionEnabled; } } +#endif +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 120300 if (@available(macOS 12.3, *)) { - if (preferences.fullscreenEnabled != NULL) { - config.preferences.elementFullscreenEnabled = *preferences.fullscreenEnabled; - } + if (preferences.fullscreenEnabled != NULL) { + config.preferences.elementFullscreenEnabled = *preferences.fullscreenEnabled; + } } - -// [config.preferences setValue:[NSNumber numberWithBool:true] forKey:@"developerExtrasEnabled"]; +#endif +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 101500 if (@available(macOS 10.15, *)) { config.preferences.fraudulentWebsiteWarningEnabled = fraudulentWebsiteWarningEnabled; } +#endif WKUserContentController* userContentController = [WKUserContentController new]; [userContentController addScriptMessageHandler:self name:@"external"]; @@ -431,10 +435,11 @@ - (void)webView:(WKWebView *)webView runOpenPanelWithParameters:(WKOpenPanelPara NSOpenPanel *openPanel = [NSOpenPanel openPanel]; openPanel.allowsMultipleSelection = parameters.allowsMultipleSelection; +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 101400 if (@available(macOS 10.14, *)) { openPanel.canChooseDirectories = parameters.allowsDirectories; } - +#endif [openPanel beginSheetModalForWindow:webView.window completionHandler:^(NSInteger result) { @@ -558,14 +563,18 @@ -(void) OpenFileDialog :(NSString*)title :(NSString*)defaultFilename :(NSString* #ifdef USE_NEW_FILTERS NSMutableArray *contentTypes = [[NSMutableArray new] autorelease]; for (NSString *filter in filterList) { +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 110000 if (@available(macOS 11.0, *)) { UTType *t = [UTType typeWithFilenameExtension:filter]; [contentTypes addObject:t]; } +#endif } +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 110000 if (@available(macOS 11.0, *)) { [dialog setAllowedContentTypes:contentTypes]; } +#endif #else [dialog setAllowedFileTypes:filterList]; #endif @@ -638,17 +647,21 @@ -(void) SaveFileDialog :(NSString*)title :(NSString*)defaultFilename :(NSString* #ifdef USE_NEW_FILTERS NSMutableArray *contentTypes = [[NSMutableArray new] autorelease]; for (NSString *filter in filterList) { +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 110000 if (@available(macOS 11.0, *)) { UTType *t = [UTType typeWithFilenameExtension:filter]; [contentTypes addObject:t]; } +#endif } if( contentTypes.count == 0) { [dialog setAllowsOtherFileTypes:true]; } else { +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 110000 if (@available(macOS 11.0, *)) { [dialog setAllowedContentTypes:contentTypes]; } +#endif } #else diff --git a/v2/internal/frontend/desktop/darwin/inspector_dev.go b/v2/internal/frontend/desktop/darwin/inspector_dev.go index f3520cb3ed3..e79b9c3e7ea 100644 --- a/v2/internal/frontend/desktop/darwin/inspector_dev.go +++ b/v2/internal/frontend/desktop/darwin/inspector_dev.go @@ -23,6 +23,7 @@ extern void processMessage(const char *message); @end void showInspector(void *inctx) { +#if MAC_OS_X_VERSION_MAX_ALLOWED >= 120000 ON_MAIN_THREAD( if (@available(macOS 12.0, *)) { WailsContext *ctx = (__bridge WailsContext*) inctx; @@ -47,7 +48,7 @@ void showInspector(void *inctx) { NSLog(@"Opening the inspector needs at least MacOS 12"); } ); - +#endif } void setupF12hotkey() { From e3af0945d0a2c1fe27683a7212ff33b48a710d46 Mon Sep 17 00:00:00 2001 From: atterpac <89053530+atterpac@users.noreply.github.com> Date: Sat, 9 Dec 2023 16:21:21 -0700 Subject: [PATCH 70/87] Fix Nil pointer if GPU does not provide device info (#3108) * Fix Nil pointer if GPU does not provide device info * Updated changelog format --------- Co-authored-by: Lea Anthony --- v2/cmd/wails/doctor.go | 4 ++++ website/src/pages/changelog.mdx | 3 +++ 2 files changed, 7 insertions(+) diff --git a/v2/cmd/wails/doctor.go b/v2/cmd/wails/doctor.go index 74f9ec6cc88..35e3a13ffc4 100644 --- a/v2/cmd/wails/doctor.go +++ b/v2/cmd/wails/doctor.go @@ -104,6 +104,10 @@ func diagnoseEnvironment(f *flags.Doctor) error { if len(gpu.GraphicsCards) > 1 { prefix = "GPU " + strconv.Itoa(idx+1) + " " } + if card.DeviceInfo == nil { + systemTabledata = append(systemTabledata, []string{prefix, "Unknown"}) + continue + } details := fmt.Sprintf("%s (%s) - Driver: %s", card.DeviceInfo.Product.Name, card.DeviceInfo.Vendor.Name, card.DeviceInfo.Driver) systemTabledata = append(systemTabledata, []string{prefix, details}) } diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index a1476656608..1b795461fcd 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -14,6 +14,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +### Fixed +- Segfault in `wails doctor` when GPU device information is not provided. Thanks to [@atterpac](https://github.com/atterpac) for the fix ([#3108](https://github.com/wailsapp/wails/pull/3108)). + ## v2.7.0 - 2023-12-09 ### Added From a632ac506837a2367af3a5f2b5a15ebf5a9b3bf7 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 10:25:14 +1100 Subject: [PATCH 71/87] v2.7.1 --- v2/cmd/wails/internal/version.txt | 2 +- website/src/pages/changelog.mdx | 5 +++++ 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/v2/cmd/wails/internal/version.txt b/v2/cmd/wails/internal/version.txt index 5c9dc9d02a1..afe8e01d5ba 100644 --- a/v2/cmd/wails/internal/version.txt +++ b/v2/cmd/wails/internal/version.txt @@ -1 +1 @@ -v2.7.0 \ No newline at end of file +v2.7.1 \ No newline at end of file diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 1b795461fcd..8f96e143898 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -16,6 +16,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Fixed - Segfault in `wails doctor` when GPU device information is not provided. Thanks to [@atterpac](https://github.com/atterpac) for the fix ([#3108](https://github.com/wailsapp/wails/pull/3108)). +- Fixed building on macOS 11. Updated other build guards. Fixed in [#3111](https://github.com/wailsapp/wails/pull/3108) by [@leaanthony](https://github.com/leaanthony). + +### Changed +- Go 1.20 is now the minimum supported Go version. + ## v2.7.0 - 2023-12-09 From ae99d7a97377653abac0561d142ba05292f60f94 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sun, 10 Dec 2023 10:41:14 +1100 Subject: [PATCH 72/87] Update version in changelog --- website/src/pages/changelog.mdx | 2 ++ 1 file changed, 2 insertions(+) diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 8f96e143898..25fd96c9661 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -14,6 +14,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +## v2.7.1 - 2023-12-10 + ### Fixed - Segfault in `wails doctor` when GPU device information is not provided. Thanks to [@atterpac](https://github.com/atterpac) for the fix ([#3108](https://github.com/wailsapp/wails/pull/3108)). - Fixed building on macOS 11. Updated other build guards. Fixed in [#3111](https://github.com/wailsapp/wails/pull/3108) by [@leaanthony](https://github.com/leaanthony). From aa8cd7ce67b1427d75cb4f56259d76cc22b4a882 Mon Sep 17 00:00:00 2001 From: Ali Torabi Date: Mon, 11 Dec 2023 00:05:27 -0800 Subject: [PATCH 73/87] Add the esp studio show case (#3114) --- website/docs/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ .../current/community/showcase/espstudio.mdx | 13 +++++++++++++ website/static/img/showcase/esp-studio.png | Bin 0 -> 138285 bytes .../community/showcase/espstudio.mdx | 13 +++++++++++++ 10 files changed, 117 insertions(+) create mode 100644 website/docs/community/showcase/espstudio.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx create mode 100644 website/static/img/showcase/esp-studio.png create mode 100644 website/versioned_docs/version-v2.6.0/community/showcase/espstudio.mdx diff --git a/website/docs/community/showcase/espstudio.mdx b/website/docs/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..44db858f971 --- /dev/null +++ b/website/docs/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - Cross platform, Desktop, Cloud, and Embedded software +for controlling ESP/Arduino devices, and building complex IOT workflows and control systems diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..aae652e2c3c --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - Logiciels multiplateformes, de bureau, cloud et embarqués +pour contrôler les appareils ESP/Arduino et créer des flux de travail et des systèmes de contrôle IOT complexes \ No newline at end of file diff --git a/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..4e4295b26ab --- /dev/null +++ b/website/i18n/ja/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - クロスプラットフォーム、デスクトップ、クラウド、組み込みソフトウェア +ESP/Arduino デバイスの制御、複雑な IOT ワークフローと制御システムの構築用 \ No newline at end of file diff --git a/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..73f93417cf0 --- /dev/null +++ b/website/i18n/ko/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - 크로스 플랫폼, 데스크탑, 클라우드 및 임베디드 소프트웨어 +ESP/Arduino 장치 제어, 복잡한 IOT 워크플로 및 제어 시스템 구축용 \ No newline at end of file diff --git a/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..78c79769f7e --- /dev/null +++ b/website/i18n/pt/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - Software multiplataforma, desktop, nuvem e incorporado +para controlar dispositivos ESP/Arduino e construir fluxos de trabalho e sistemas de controle IOT complexos \ No newline at end of file diff --git a/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..0fc168d6300 --- /dev/null +++ b/website/i18n/ru/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - Кроссплатформенное, настольное, облачное и встроенное программное обеспечение +для управления устройствами ESP/Arduino и построения сложных рабочих процессов и систем управления IOT. \ No newline at end of file diff --git a/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..0f4b6b9ca7e --- /dev/null +++ b/website/i18n/vi/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - Đa nền tảng, Máy tính để bàn, Đám mây và Phần mềm nhúng +để điều khiển các thiết bị ESP/Arduino và xây dựng các hệ thống điều khiển và quy trình làm việc IOT phức tạp \ No newline at end of file diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..43f3289181e --- /dev/null +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - 跨平台、桌面、云和嵌入式软件 +用于控制 ESP/Arduino 设备,以及构建复杂的物联网工作流程和控制系统 \ No newline at end of file diff --git a/website/static/img/showcase/esp-studio.png b/website/static/img/showcase/esp-studio.png new file mode 100644 index 0000000000000000000000000000000000000000..fb7bc7c18108e8c8645d444af249163749eb767a GIT binary patch literal 138285 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�MuI@IP~pJgEYVmj)(?x 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�}nl@}UfPeNDlc4~9Mb-e3jet9%)mfb9RXVq)ug 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 bZtO8p$$uFx9_kK)4h+}Sv{iFe?)v;6T{B&| literal 0 HcmV?d00001 diff --git a/website/versioned_docs/version-v2.6.0/community/showcase/espstudio.mdx b/website/versioned_docs/version-v2.6.0/community/showcase/espstudio.mdx new file mode 100644 index 00000000000..44db858f971 --- /dev/null +++ b/website/versioned_docs/version-v2.6.0/community/showcase/espstudio.mdx @@ -0,0 +1,13 @@ +# ESP Studio + +```mdx-code-block +

+ +
+

+``` + +[ESP Studio](https://github.com/torabian/esp-studio) - Cross platform, Desktop, Cloud, and Embedded software +for controlling ESP/Arduino devices, and building complex IOT workflows and control systems From 5f412d0bfb9fb56fdf6c41333afc6fa299d00c3b Mon Sep 17 00:00:00 2001 From: "imgbot[bot]" <31301654+imgbot[bot]@users.noreply.github.com> Date: Tue, 12 Dec 2023 20:33:21 +1100 Subject: [PATCH 74/87] [ImgBot] Optimize images (#3094) *Total -- 454.86kb -> 406.40kb (10.65%) /website/static/img/showcase/snippetexpandergui-select-snippet.png -- 140.27kb -> 113.72kb (18.93%) /website/static/img/showcase/snippetexpandergui-search-and-paste.png -- 86.16kb -> 70.75kb (17.88%) /website/static/img/showcase/snippetexpandergui-add-snippet.png -- 36.61kb -> 30.74kb (16.04%) /website/static/img/icon-linux-dark.svg -- 2.56kb -> 2.38kb (7.02%) /v2/pkg/templates/templates/preact/frontend/src/assets/preact.svg -- 1.54kb -> 1.49kb (3.54%) /website/static/img/sponsors.svg -- 187.71kb -> 187.32kb (0.21%) Signed-off-by: ImgBotApp Co-authored-by: ImgBotApp --- .../preact/frontend/src/assets/preact.svg | 11 +- website/static/img/icon-linux-dark.svg | 9 +- .../snippetexpandergui-add-snippet.png | Bin 37492 -> 31478 bytes .../snippetexpandergui-search-and-paste.png | Bin 88231 -> 72453 bytes .../snippetexpandergui-select-snippet.png | Bin 143641 -> 116453 bytes website/static/img/sponsors.svg | 175 +----------------- 6 files changed, 3 insertions(+), 192 deletions(-) diff --git a/v2/pkg/templates/templates/preact/frontend/src/assets/preact.svg b/v2/pkg/templates/templates/preact/frontend/src/assets/preact.svg index 8d4155b1d88..23433fcf8e3 100644 --- a/v2/pkg/templates/templates/preact/frontend/src/assets/preact.svg +++ b/v2/pkg/templates/templates/preact/frontend/src/assets/preact.svg @@ -1,10 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/website/static/img/icon-linux-dark.svg b/website/static/img/icon-linux-dark.svg index 90e328d0765..b0ca94d950e 100644 --- a/website/static/img/icon-linux-dark.svg +++ b/website/static/img/icon-linux-dark.svg @@ -1,8 +1 @@ - - - - - + \ No newline at end of file diff --git a/website/static/img/showcase/snippetexpandergui-add-snippet.png b/website/static/img/showcase/snippetexpandergui-add-snippet.png index 2a88cb75f64846d512ecd7e51da4a0b8531c9598..0f1c60a869e3e077200278f972b2197974ea6267 100644 GIT binary patch literal 31478 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 Su=5c6M0puy=@Q8&Fa8%B-zOCS literal 37492 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 Si|e}niITj!T!qZrkN*ceN}P@W diff --git a/website/static/img/showcase/snippetexpandergui-search-and-paste.png b/website/static/img/showcase/snippetexpandergui-search-and-paste.png index e197805fc376395549a2557b8c2d896e503794c2..a7af8a415b179fbbac79d735e493df924500726b 100644 GIT binary patch literal 72453 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 A?EnA( literal 88231 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 Sc!od5UUGttI3Kqs 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

+
+

+ +

+ Go va Web texnologiyalaridan foydalangan holda ish stoli ilovalarini yarating +
+
+ + GitHub + + + + + + Go Reference + + + CodeFactor + + + + + + Awesome + + + Discord + +
+ + Build + + + GitHub tag (latest SemVer pre-release) + +

+ +
+ + + +[English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · +[한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz) + + + +
+ +## Tarkib + +- [Tarkib](#tarkib) +- [Kirish](#kirish) +- [Xususiyatlari](#xususiyatlari) + - [Yo'l xaritasi](#yol-xaritasi) +- [Ishni boshlash](#ishni-boshlash) +- [Homiylar](#homiylar) +- [FAQ](#faq) +- [Vaqt o'tishi bilan yulduzlar](#vaqt-otishi-bilan-yulduzlar) +- [Ishtirokchilar](#homiylar) +- [Litsenziya](#litsenziya) +- [Ilhomlanish](#ilhomlanish) + +## Kirish + +Odatda, Go dasturlari uchun veb-interfeyslar o'rnatilgan veb-server va veb-brauzerdir. +Walls boshqacha yondashuvni qo'llaydi: u Go kodini ham, veb-interfeysni ham bitta ikkilik (e.g: EXE)fayliga o'raydi. +Loyihalarni yaratish, kompilyatsiya qilish va birlashtirishni boshqarish orqali ilovangizni yaratishni osonlashtiradi. +Hamma narsa faqat sizning tasavvuringiz bilan cheklangan! + +## Xususiyatlari + +- Backend uchun standart Go dan foydalaning +- UI yaratish uchun siz allaqachon tanish bo'lgan har qanday frontend texnologiyasidan foydalaning +- Oldindan tayyorlangan shablonlardan foydalanib, Go dasturlaringiz uchun tezda boy frontendlarni yarating +- Javascriptdan Go methodlarini osongina chaqiring +- Go struktura va methodlari uchun avtomatik yaratilgan Typescript ta'riflari +- Mahalliy Dialoglar va Menyular +- Mahalliy Dark / Light rejimini qo'llab-quvvatlash +- Zamonaviy shaffoflik va "muzli oyna" effektlarini qo'llab-quvvatlaydi +- Go va Javascript o'rtasidagi yagona hodisa tizimi +- Loyihalaringizni tezda yaratish va qurish uchun kuchli cli vositasi +- Ko'p platformali +- Mahalliy renderlash mexanizmlaridan foydalanadi - _o'rnatilgan brauzer yo'q_! + +### Yo'l xaritasi + +Loyihaning yoʻl xaritasini [bu yerdan](https://github.com/wailsapp/wails/discussions/1484) topish mumkin. Iltimos, maslahatlashing +Buni yaxshilash so'rovini ochishdan oldin. + +## Ishni boshlash + +O'rnatish bo'yicha ko'rsatmalar [Rasmiy veb saytda](https://wails.io/docs/gettingstarted/installation) mavjud. + +## Homiylar + +Ushbu loyiha quyidagi mehribon odamlar / kompaniyalar tomonidan qo'llab-quvvatlanadi: + + +

+ +

+ +## FAQ + +- Bu Elektronga muqobilmi? + + Sizning talablaringizga bog'liq. Bu Go dasturchilariga yengil ish stoli yaratishni osonlashtirish uchun yaratilgan + ilovalar yoki ularning mavjud ilovalariga frontend qo'shing. Wails menyular kabi mahalliy elementlarni taklif qiladi + va dialoglar, shuning uchun uni yengil elektron muqobili deb hisoblash mumkin. + +- Ushbu loyiha kimlar uchun? + + Server yaratmasdan va uni ko'rish uchun brauzerni ochmasdan, o'z ilovalari bilan HTML/JS/CSS orqali frontendini birlashtirmoqchi bo'lgan dasturchilar uchun. + +- Bu qanday nom? + + Men WebViewni ko'rganimda, men shunday deb o'yladim: "Menga WebView ilovasini yaratish uchun vositalar kerak. + biroz Rails for Rubyga o'xshaydi." Demak, dastlab bu so'zlar ustida o'yin edi (Railsda Webview). Shunday bo'ldi. + u men kelgan [Mamlakat](https://en.wikipedia.org/wiki/Wales)ning inglizcha nomining omofonidir. + +## Vaqt o'tishi bilan yulduzlar + + + + + + Yulduzlar tarixi jadvali + + + +## Ishtirokchilar + +Ishtirokchilar roʻyxati oʻqish uchun juda kattalashib bormoqda! Bunga hissa qo'shgan barcha ajoyib odamlarning +loyihada o'z sahifasi bor [bu yerga](https://wails.io/credits#contributors). + +## Litsenziya + +[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fwailsapp%2Fwails.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fwailsapp%2Fwails?ref=badge_large) + +## Ilhomlanish + +Ushbu loyiha asosan quyidagi albomlar uchun kodlangan: + +- [Manic Street Preachers - Resistance Is Futile](https://open.spotify.com/album/1R2rsEUqXjIvAbzM0yHrxA) +- [Manic Street Preachers - This Is My Truth, Tell Me Yours](https://open.spotify.com/album/4VzCL9kjhgGQeKCiojK1YN) +- [The Midnight - Endless Summer](https://open.spotify.com/album/4Krg8zvprquh7TVn9OxZn8) +- [Gary Newman - Savage (Songs from a Broken World)](https://open.spotify.com/album/3kMfsD07Q32HRWKRrpcexr) +- [Steve Vai - Passion & Warfare](https://open.spotify.com/album/0oL0OhrE2rYVns4IGj8h2m) +- [Ben Howard - Every Kingdom](https://open.spotify.com/album/1nJsbWm3Yy2DW1KIc1OKle) +- [Ben Howard - Noonday Dream](https://open.spotify.com/album/6astw05cTiXEc2OvyByaPs) +- [Adwaith - Melyn](https://open.spotify.com/album/2vBE40Rp60tl7rNqIZjaXM) +- [Gwidaith Hen Fran - Cedors Hen Wrach](https://open.spotify.com/album/3v2hrfNGINPLuDP0YDTOjm) +- [Metallica - Metallica](https://open.spotify.com/album/2Kh43m04B1UkVcpcRa1Zug) +- [Bloc Party - Silent Alarm](https://open.spotify.com/album/6SsIdN05HQg2GwYLfXuzLB) +- [Maxthor - Another World](https://open.spotify.com/album/3tklE2Fgw1hCIUstIwPBJF) +- [Alun Tan Lan - Y Distawrwydd](https://open.spotify.com/album/0c32OywcLpdJCWWMC6vB8v) diff --git a/README.zh-Hans.md b/README.zh-Hans.md index a26639a17cb..ce3986e3636 100644 --- a/README.zh-Hans.md +++ b/README.zh-Hans.md @@ -44,7 +44,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) · [Francais](README.fr.md) +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) From 772f0215a672988c844ea200033140ced12869c0 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Fri, 15 Dec 2023 20:26:30 +1100 Subject: [PATCH 80/87] Remove quarantine attribute from bindings + compiled binary (#3118) * Remove quarantine attribute from bindings + compiled binary --------- Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> --- v2/pkg/commands/bindings/bindings.go | 8 ++++++++ v2/pkg/commands/build/build.go | 14 ++++++++++++++ 2 files changed, 22 insertions(+) diff --git a/v2/pkg/commands/bindings/bindings.go b/v2/pkg/commands/bindings/bindings.go index 310b1e9afc0..6ed2a27bab6 100644 --- a/v2/pkg/commands/bindings/bindings.go +++ b/v2/pkg/commands/bindings/bindings.go @@ -57,6 +57,14 @@ func GenerateBindings(options Options) (string, error) { return stdout, fmt.Errorf("%s\n%s\n%s", stdout, stderr, err) } + if runtime.GOOS == "darwin" { + // Remove quarantine attribute + stdout, stderr, err = shell.RunCommand(workingDirectory, "xattr", "-rc", filename) + if err != nil { + return stdout, fmt.Errorf("%s\n%s\n%s", stdout, stderr, err) + } + } + defer func() { // Best effort removal of temp file _ = os.Remove(filename) diff --git a/v2/pkg/commands/build/build.go b/v2/pkg/commands/build/build.go index 62c08e910d1..d061f2ce089 100644 --- a/v2/pkg/commands/build/build.go +++ b/v2/pkg/commands/build/build.go @@ -321,6 +321,20 @@ func execBuildApplication(builder Builder, options *Options) (string, error) { } } + if runtime.GOOS == "darwin" { + // Remove quarantine attribute + if _, err := os.Stat(options.CompiledBinary); os.IsNotExist(err) { + return "", fmt.Errorf("compiled binary does not exist at path: %s", options.CompiledBinary) + } + stdout, stderr, err := shell.RunCommand(options.BinDirectory, "xattr", "-rc", options.CompiledBinary) + if err != nil { + return "", fmt.Errorf("%s - %s", err.Error(), stderr) + } + if options.Verbosity == VERBOSE && stdout != "" { + pterm.Info.Println(stdout) + } + } + pterm.Println("Done.") // Do we need to pack the app for non-windows? From dc5f0b6e7bc6d21d3eeb88f65b77423e85256fb7 Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Fri, 15 Dec 2023 20:31:33 +1100 Subject: [PATCH 81/87] Update Changelog --- website/src/pages/changelog.mdx | 1 + 1 file changed, 1 insertion(+) diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 44255428a55..442b19b7c59 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -18,6 +18,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added windows options supports `DisablePinchZoom` configuration. Added by @tuuzed in [PR](https://github.com/wailsapp/wails/pull/3115) - Add Apple Silicon hardware detection to `wails doctor`. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/3129) +- Remove quarantine attribute on macOS binaries. Changed by @leaanthony in [PR](https://github.com/wailsapp/wails/pull/3118) ## v2.7.1 - 2023-12-10 From 1683e240d94b4aff94bd8e0bc878562860393cb4 Mon Sep 17 00:00:00 2001 From: Mark Snyder <20092441+mkwsnyder@users.noreply.github.com> Date: Sat, 16 Dec 2023 15:09:58 -0700 Subject: [PATCH 82/87] add gstreamer error documentation (#3134) * add gstreamer error documentation * add info to changelog * add profile url --- website/docs/guides/linux.mdx | 52 +++++++++++++++++++ website/src/pages/changelog.mdx | 1 + .../version-v2.6.0/guides/linux.mdx | 52 +++++++++++++++++++ .../version-v2.7.0/guides/linux.mdx | 52 +++++++++++++++++++ 4 files changed, 157 insertions(+) diff --git a/website/docs/guides/linux.mdx b/website/docs/guides/linux.mdx index fa74fe6cf86..1b55297b50d 100644 --- a/website/docs/guides/linux.mdx +++ b/website/docs/guides/linux.mdx @@ -18,3 +18,55 @@ videoTag.addEventListener("timeupdate", (event) => { Source: [Lyimmi](https://github.com/Lyimmi) on the [discussions board](https://github.com/wailsapp/wails/issues/1729#issuecomment-1212291275) + +## GStreamer error when using Audio or Video elements + +If you are seeing the following error when including `

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 k#6)t^_OD+sHPX@ADS2;p!AMN@0|c4LZMDU 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%0AhcdX$VcoK2IRg<%SELX- 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 S - - -Bronze Sponsors - Cody Bentley - - - - Kazuya Gokita - - - - Simon Thomas - -Covering Costs - Nick - - - - Marcus - - - - John - - - - Matt Holt - - - - Iain - -Buying Breakfast - tc-hib - - - - Tai Groot - - - - Michael - - - - Tom Wu - - - - Bironou - - - - Arden - - - - igops - -Buying Coffee - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -Helpers - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +Bronze SponsorsCody BentleyKazuya GokitaSimon ThomasCovering CostsNickMarcusJohnMatt HoltIainBuying Breakfasttc-hibTai GrootMichaelTom WuBironouArdenigopsBuying CoffeeHelpers \ No newline at end of file From aaa65ff7c2233c74ab8917bcb2a9d51c45581b43 Mon Sep 17 00:00:00 2001 From: Yong Hui Date: Tue, 12 Dec 2023 17:38:30 +0800 Subject: [PATCH 75/87] docs: fix doc typos (#3119) (cherry picked from commit acf55033e2e7dabb480fa3c00e18f945fcc18d93) --- .../current/reference/runtime/window.mdx | 2 +- .../version-v2.6.0/reference/runtime/window.mdx | 2 +- .../version-v2.7.0/reference/runtime/window.mdx | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx index f5397d8472d..a4b9c1263b5 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/current/reference/runtime/window.mdx @@ -104,7 +104,7 @@ Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` -### WindowSetSize 获取窗口尺寸 +### WindowGetSize 获取窗口尺寸 获取窗口的宽度和高度。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.6.0/reference/runtime/window.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.6.0/reference/runtime/window.mdx index 3c76752593a..9cad686ed39 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.6.0/reference/runtime/window.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.6.0/reference/runtime/window.mdx @@ -104,7 +104,7 @@ Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(size: Size)` -### WindowSetSize 获取窗口尺寸 +### WindowGetSize 获取窗口尺寸 获取窗口的宽度和高度。 diff --git a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx index f5397d8472d..a4b9c1263b5 100644 --- a/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx +++ b/website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx @@ -104,7 +104,7 @@ Go: `WindowIsNormal(ctx context.Context) bool`
JS: `WindowIsNormal() bool` Go: `WindowSetSize(ctx context.Context, width int, height int)`
JS: `WindowSetSize(width: number, height: number)` -### WindowSetSize 获取窗口尺寸 +### WindowGetSize 获取窗口尺寸 获取窗口的宽度和高度。 From 3aaa1f8e3408fdd7e00d408e273ad2b5453609b3 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 13 Dec 2023 23:15:03 +1100 Subject: [PATCH 76/87] chore: update sponsors.svg (#3122) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 175 +++++++++++++++++++++++++++++++- 1 file changed, 174 insertions(+), 1 deletion(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 317af8e6adb..ceb2bbcf8ee 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -1 +1,174 @@ -Bronze SponsorsCody BentleyKazuya GokitaSimon ThomasCovering CostsNickMarcusJohnMatt HoltIainBuying Breakfasttc-hibTai GrootMichaelTom WuBironouArdenigopsBuying CoffeeHelpers \ No newline at end of file + + + + +Bronze Sponsors + Cody Bentley + + + + Kazuya Gokita + + + + Simon Thomas + +Covering Costs + Nick + + + + Marcus + + + + John + + + + Matt Holt + + + + Iain + +Buying Breakfast + tc-hib + + + + Tai Groot + + + + Michael + + + + Tom Wu + + + + Bironou + + + + Arden + + + + igops + +Buying Coffee + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Helpers + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + From 5f457dba8e18be67f9139a038164b97144e5f09b Mon Sep 17 00:00:00 2001 From: ALMAS <9382335+almas1992@users.noreply.github.com> Date: Fri, 15 Dec 2023 17:14:15 +0800 Subject: [PATCH 77/87] [v2/Mac] Add `Apple Silicon` hardware detection to `wails doctor` (#3129) * [v2/Mac] Add Apple Silicon hardware detection to `wails doctor` * add change log --- v2/cmd/wails/doctor.go | 47 ++++++++++++++++++++++++++++++--- website/src/pages/changelog.mdx | 4 +++ 2 files changed, 48 insertions(+), 3 deletions(-) diff --git a/v2/cmd/wails/doctor.go b/v2/cmd/wails/doctor.go index 35e3a13ffc4..5306cab1742 100644 --- a/v2/cmd/wails/doctor.go +++ b/v2/cmd/wails/doctor.go @@ -2,6 +2,7 @@ package main import ( "fmt" + "github.com/wailsapp/wails/v2/internal/shell" "runtime" "runtime/debug" "strconv" @@ -93,7 +94,14 @@ func diagnoseEnvironment(f *flags.Doctor) error { systemTabledata = append(systemTabledata, []string{prefix, cpu.Model}) } } else { - systemTabledata = append(systemTabledata, []string{"CPU", "Unknown"}) + cpuInfo := "Unknown" + if runtime.GOOS == "darwin" { + // Try to get CPU info from sysctl + if stdout, _, err := shell.RunCommand("", "sysctl", "-n", "machdep.cpu.brand_string"); err == nil { + cpuInfo = strings.TrimSpace(stdout) + } + } + systemTabledata = append(systemTabledata, []string{"CPU", cpuInfo}) } // Probe GPU @@ -112,14 +120,47 @@ func diagnoseEnvironment(f *flags.Doctor) error { systemTabledata = append(systemTabledata, []string{prefix, details}) } } else { - systemTabledata = append(systemTabledata, []string{"GPU", "Unknown"}) + gpuInfo := "Unknown" + if runtime.GOOS == "darwin" { + // Try to get GPU info from system_profiler + if stdout, _, err := shell.RunCommand("", "system_profiler", "SPDisplaysDataType"); err == nil { + var ( + startCapturing bool + gpuInfoDetails []string + ) + for _, line := range strings.Split(stdout, "\n") { + if strings.Contains(line, "Chipset Model") { + startCapturing = true + } + if startCapturing { + gpuInfoDetails = append(gpuInfoDetails, strings.TrimSpace(line)) + } + if strings.Contains(line, "Metal Support") { + break + } + } + if len(gpuInfoDetails) > 0 { + gpuInfo = strings.Join(gpuInfoDetails, " ") + } + } + } + systemTabledata = append(systemTabledata, []string{"GPU", gpuInfo}) } memory, _ := ghw.Memory() if memory != nil { systemTabledata = append(systemTabledata, []string{"Memory", strconv.Itoa(int(memory.TotalPhysicalBytes/1024/1024/1024)) + "GB"}) } else { - systemTabledata = append(systemTabledata, []string{"Memory", "Unknown"}) + memInfo := "Unknown" + if runtime.GOOS == "darwin" { + // Try to get Memory info from sysctl + if stdout, _, err := shell.RunCommand("", "sysctl", "-n", "hw.memsize"); err == nil { + if memSize, err := strconv.Atoi(strings.TrimSpace(stdout)); err == nil { + memInfo = strconv.Itoa(memSize/1024/1024/1024) + "GB" + } + } + } + systemTabledata = append(systemTabledata, []string{"Memory", memInfo}) } err = pterm.DefaultTable.WithBoxed().WithData(systemTabledata).Render() diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 25fd96c9661..452b8c65ed8 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -14,6 +14,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +### Changed + +- Add Apple Silicon hardware detection to `wails doctor`. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/3129) + ## v2.7.1 - 2023-12-10 ### Fixed From 8efaaffadb58d8e5cc7f405bcc42e165b615209f Mon Sep 17 00:00:00 2001 From: Yong Hui Date: Fri, 15 Dec 2023 17:21:50 +0800 Subject: [PATCH 78/87] feat: add windows options supports `DisablePinchZoom` configuration(#2021) (#3115) * feat: add windows options supports `IsPinchZoomEnabled` configuration(#2021) * refactor: modify `IsPinchZoomEnabled` to `DisablePinchZoom` ensure default behavior is consistent * docs: add `DisablePinchZoom` to changelog * docs: update the description of `DisablePinchZoom` attributes in the document --------- Co-authored-by: Lea Anthony --- v2/internal/frontend/desktop/windows/frontend.go | 4 ++++ v2/pkg/options/windows/windows.go | 2 ++ website/docs/reference/options.mdx | 8 ++++++++ website/src/pages/changelog.mdx | 4 +++- 4 files changed, 17 insertions(+), 1 deletion(-) diff --git a/v2/internal/frontend/desktop/windows/frontend.go b/v2/internal/frontend/desktop/windows/frontend.go index 7671ad7420b..97eef741feb 100644 --- a/v2/internal/frontend/desktop/windows/frontend.go +++ b/v2/internal/frontend/desktop/windows/frontend.go @@ -543,6 +543,10 @@ func (f *Frontend) setupChromium() { if err != nil { log.Fatal(err) } + err = settings.PutIsPinchZoomEnabled(!opts.DisablePinchZoom) + if err != nil { + log.Fatal(err) + } } err = settings.PutIsStatusBarEnabled(false) diff --git a/v2/pkg/options/windows/windows.go b/v2/pkg/options/windows/windows.go index 073450c9f68..39b91ee8def 100644 --- a/v2/pkg/options/windows/windows.go +++ b/v2/pkg/options/windows/windows.go @@ -68,6 +68,8 @@ type Options struct { IsZoomControlEnabled bool ZoomFactor float64 + DisablePinchZoom bool + // Disable all window decorations in Frameless mode, which means no "Aero Shadow" and no "Rounded Corner" will be shown. // "Rounded Corners" are only available on Windows 11. DisableFramelessWindowDecorations bool diff --git a/website/docs/reference/options.mdx b/website/docs/reference/options.mdx index caa2d0f806c..d9a1ca4fc4a 100644 --- a/website/docs/reference/options.mdx +++ b/website/docs/reference/options.mdx @@ -70,6 +70,7 @@ func main() { WebviewIsTransparent: false, WindowIsTranslucent: false, BackdropType: windows.Mica, + DisablePinchZoom: false, DisableWindowIcon: false, DisableFramelessWindowDecorations: false, WebviewUserDataPath: "", @@ -577,6 +578,13 @@ The value can be one of the following: | Mica | Use [Mica](https://learn.microsoft.com/en-us/windows/apps/design/style/mica) effect | | Tabbed | Use Tabbed. This is a backdrop that is similar to Mica. | +#### DisablePinchZoom + +Setting this to `true` will disable pinch zoom gestures. + +Name: DisablePinchZoom
+Type: `bool` + #### DisableWindowIcon Setting this to `true` will remove the icon in the top left corner of the title bar. diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 452b8c65ed8..44255428a55 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -14,10 +14,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] -### Changed +### Added +- Added windows options supports `DisablePinchZoom` configuration. Added by @tuuzed in [PR](https://github.com/wailsapp/wails/pull/3115) - Add Apple Silicon hardware detection to `wails doctor`. Changed by @almas1992 in [PR](https://github.com/wailsapp/wails/pull/3129) + ## v2.7.1 - 2023-12-10 ### Fixed From 435aa99656cc4304d69f7fe50696f04ac077e5a1 Mon Sep 17 00:00:00 2001 From: Dilshod Fayzullayev <140431027+bug4you@users.noreply.github.com> Date: Fri, 15 Dec 2023 14:22:51 +0500 Subject: [PATCH 79/87] Add Uzbek README (#3123) * :memo: Added README.uz-oz.md link --- README.es.md | 2 +- README.fr.md | 2 +- README.ja.md | 2 +- README.ko.md | 2 +- README.md | 2 +- README.pt-br.md | 2 +- README.ru.md | 2 +- README.uz.md | 158 ++++++++++++++++++++++++++++++++++++++++++++++ README.zh-Hans.md | 2 +- 9 files changed, 166 insertions(+), 8 deletions(-) create mode 100644 README.uz.md diff --git a/README.es.md b/README.es.md index 16cd67414d5..f7c117f6c3c 100644 --- a/README.es.md +++ b/README.es.md @@ -42,7 +42,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) · [Francais](README.fr.md) +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.fr.md b/README.fr.md index 55379aa466b..45580841b48 100644 --- a/README.fr.md +++ b/README.fr.md @@ -42,7 +42,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) · [Francais](README.fr.md) +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.ja.md b/README.ja.md index 09c1c9bbe69..a96eccfc172 100644 --- a/README.ja.md +++ b/README.ja.md @@ -44,7 +44,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) · [Francais](README.fr.md) +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.ko.md b/README.ko.md index 4e72cffa705..a8cd3d1d743 100644 --- a/README.ko.md +++ b/README.ko.md @@ -44,7 +44,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) · [Francais](README.fr.md) +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.md b/README.md index b46aea19d66..bae60686388 100644 --- a/README.md +++ b/README.md @@ -42,7 +42,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · [한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · -[Русский](README.ru.md) · [Francais](README.fr.md) +[Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.pt-br.md b/README.pt-br.md index 6cf9c7d621b..da1f6f021bf 100644 --- a/README.pt-br.md +++ b/README.pt-br.md @@ -41,7 +41,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · -[한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · [Francais](README.fr.md) +[한국어](README.ko.md) · [Español](README.es.md) · [Português](README.pt-br.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.ru.md b/README.ru.md index 018c644fbd0..36492414110 100644 --- a/README.ru.md +++ b/README.ru.md @@ -41,7 +41,7 @@ [English](README.md) · [简体中文](README.zh-Hans.md) · [日本語](README.ja.md) · -[한국어](README.ko.md) · [Español](README.es.md) · [Русский](README.ru.md) · [Francais](README.fr.md) +[한국어](README.ko.md) · [Español](README.es.md) · [Русский](README.ru.md) · [Francais](README.fr.md) · [Uzbek](README.uz.md) diff --git a/README.uz.md b/README.uz.md new file mode 100644 index 00000000000..3b86b199a22 --- /dev/null +++ b/README.uz.md @@ -0,0 +1,158 @@ +

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%0AhcdX$VcoK2IRg<%SELX- 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 S Date: Sat, 25 Nov 2023 19:50:49 +0000 Subject: [PATCH 52/87] Add support for interface generation and enums (#3047) * Add support to output ts models as interfaces * Add support to generate enums from golang * cleanup logs * add missing documentation * fix package names for enum. Fix processing enums that are in separate packages * revert golang 1.21 * Fix spelling * Add support for simplified version of Enum for typescriptify * update docs * removed unused logs * Add tests. Fix imported enums types in models --------- Co-authored-by: Lea Anthony --- v2/cmd/wails/generate.go | 11 +- v2/internal/app/app_bindings.go | 12 +- v2/internal/app/app_dev.go | 2 +- v2/internal/app/app_production.go | 2 +- v2/internal/binding/binding.go | 107 +++++++- .../binding_conflicting_package_name_test.go | 2 +- .../binding_test/binding_importedenum_test.go | 50 ++++ .../binding_returned_promises_test.go | 2 +- .../binding/binding_test/binding_test.go | 14 +- .../binding_test_import.go | 17 ++ .../binding_test/binding_tsgeneration_test.go | 232 ++++++++++++++++++ .../binding_test/binding_type_alias_test.go | 2 +- v2/internal/binding/generate_test.go | 2 +- v2/internal/project/project.go | 5 +- v2/internal/typescriptify/typescriptify.go | 12 +- v2/pkg/commands/bindings/bindings.go | 2 + v2/pkg/commands/build/build.go | 13 +- v2/pkg/options/options.go | 1 + .../docs/guides/application-development.mdx | 59 +++++ website/docs/howdoesitwork.mdx | 51 ++++ website/docs/reference/options.mdx | 56 ++++- website/docs/reference/project-config.mdx | 42 +++- website/src/pages/changelog.mdx | 1 + website/static/schemas/config.v2.json | 64 +++-- 24 files changed, 717 insertions(+), 44 deletions(-) create mode 100644 v2/internal/binding/binding_test/binding_importedenum_test.go diff --git a/v2/cmd/wails/generate.go b/v2/cmd/wails/generate.go index 159df90a1b5..05290051987 100644 --- a/v2/cmd/wails/generate.go +++ b/v2/cmd/wails/generate.go @@ -43,10 +43,15 @@ func generateModule(f *flags.GenerateModule) error { return err } + if projectConfig.Bindings.TsGeneration.OutputType == "" { + projectConfig.Bindings.TsGeneration.OutputType = "classes" + } + _, err = bindings.GenerateBindings(bindings.Options{ - Tags: buildTags, - TsPrefix: projectConfig.Bindings.TsGeneration.Prefix, - TsSuffix: projectConfig.Bindings.TsGeneration.Suffix, + Tags: buildTags, + TsPrefix: projectConfig.Bindings.TsGeneration.Prefix, + TsSuffix: projectConfig.Bindings.TsGeneration.Suffix, + TsOutputType: projectConfig.Bindings.TsGeneration.OutputType, }) if err != nil { return err diff --git a/v2/internal/app/app_bindings.go b/v2/internal/app/app_bindings.go index d079790aa7f..be031819c65 100644 --- a/v2/internal/app/app_bindings.go +++ b/v2/internal/app/app_bindings.go @@ -31,6 +31,7 @@ func (a *App) Run() error { var tsPrefixFlag *string var tsPostfixFlag *string + var tsOutputTypeFlag *string tsPrefix := os.Getenv("tsprefix") if tsPrefix == "" { @@ -42,6 +43,11 @@ func (a *App) Run() error { tsPostfixFlag = bindingFlags.String("tssuffix", "", "Suffix for generated typescript entities") } + tsOutputType := os.Getenv("tsoutputtype") + if tsOutputType == "" { + tsOutputTypeFlag = bindingFlags.String("tsoutputtype", "", "Output type for generated typescript entities (classes|interfaces)") + } + _ = bindingFlags.Parse(os.Args[1:]) if tsPrefixFlag != nil { tsPrefix = *tsPrefixFlag @@ -49,11 +55,15 @@ func (a *App) Run() error { if tsPostfixFlag != nil { tsSuffix = *tsPostfixFlag } + if tsOutputTypeFlag != nil { + tsOutputType = *tsOutputTypeFlag + } - appBindings := binding.NewBindings(a.logger, a.options.Bind, bindingExemptions, IsObfuscated()) + appBindings := binding.NewBindings(a.logger, a.options.Bind, bindingExemptions, IsObfuscated(), a.options.EnumBind) appBindings.SetTsPrefix(tsPrefix) appBindings.SetTsSuffix(tsSuffix) + appBindings.SetOutputType(tsOutputType) err := generateBindings(appBindings) if err != nil { diff --git a/v2/internal/app/app_dev.go b/v2/internal/app/app_dev.go index 51e3e63ba1c..58cd94ef0e0 100644 --- a/v2/internal/app/app_dev.go +++ b/v2/internal/app/app_dev.go @@ -209,7 +209,7 @@ func CreateApp(appoptions *options.App) (*App, error) { appoptions.OnDomReady, appoptions.OnBeforeClose, } - appBindings := binding.NewBindings(myLogger, appoptions.Bind, bindingExemptions, false) + appBindings := binding.NewBindings(myLogger, appoptions.Bind, bindingExemptions, false, appoptions.EnumBind) eventHandler := runtime.NewEvents(myLogger) ctx = context.WithValue(ctx, "events", eventHandler) diff --git a/v2/internal/app/app_production.go b/v2/internal/app/app_production.go index 96ed84e30c0..4c217b17c4a 100644 --- a/v2/internal/app/app_production.go +++ b/v2/internal/app/app_production.go @@ -72,7 +72,7 @@ func CreateApp(appoptions *options.App) (*App, error) { appoptions.OnDomReady, appoptions.OnBeforeClose, } - appBindings := binding.NewBindings(myLogger, appoptions.Bind, bindingExemptions, IsObfuscated()) + appBindings := binding.NewBindings(myLogger, appoptions.Bind, bindingExemptions, IsObfuscated(), appoptions.EnumBind) eventHandler := runtime.NewEvents(myLogger) ctx = context.WithValue(ctx, "events", eventHandler) // Attach logger to context diff --git a/v2/internal/binding/binding.go b/v2/internal/binding/binding.go index d911e12a3d7..568e11b0318 100644 --- a/v2/internal/binding/binding.go +++ b/v2/internal/binding/binding.go @@ -23,17 +23,20 @@ type Bindings struct { exemptions slicer.StringSlicer structsToGenerateTS map[string]map[string]interface{} + enumsToGenerateTS map[string]map[string]interface{} tsPrefix string tsSuffix string + tsInterface bool obfuscate bool } // NewBindings returns a new Bindings object -func NewBindings(logger *logger.Logger, structPointersToBind []interface{}, exemptions []interface{}, obfuscate bool) *Bindings { +func NewBindings(logger *logger.Logger, structPointersToBind []interface{}, exemptions []interface{}, obfuscate bool, enumsToBind []interface{}) *Bindings { result := &Bindings{ db: newDB(), logger: logger.CustomLogger("Bindings"), structsToGenerateTS: make(map[string]map[string]interface{}), + enumsToGenerateTS: make(map[string]map[string]interface{}), obfuscate: obfuscate, } @@ -47,6 +50,10 @@ func NewBindings(logger *logger.Logger, structPointersToBind []interface{}, exem result.exemptions.Add(name) } + for _, enum := range enumsToBind { + result.AddEnumToGenerateTS(enum) + } + // Add the structs to bind for _, ptr := range structPointersToBind { err := result.Add(ptr) @@ -88,16 +95,21 @@ func (b *Bindings) ToJSON() (string, error) { func (b *Bindings) GenerateModels() ([]byte, error) { models := map[string]string{} var seen slicer.StringSlicer + var seenEnumsPackages slicer.StringSlicer allStructNames := b.getAllStructNames() allStructNames.Sort() + allEnumNames := b.getAllEnumNames() + allEnumNames.Sort() for packageName, structsToGenerate := range b.structsToGenerateTS { thisPackageCode := "" w := typescriptify.New() w.WithPrefix(b.tsPrefix) w.WithSuffix(b.tsSuffix) + w.WithInterface(b.tsInterface) w.Namespace = packageName w.WithBackupDir("") w.KnownStructs = allStructNames + w.KnownEnums = allEnumNames // sort the structs var structNames []string for structName := range structsToGenerate { @@ -112,6 +124,20 @@ func (b *Bindings) GenerateModels() ([]byte, error) { structInterface := structsToGenerate[structName] w.Add(structInterface) } + + // if we have enums for this package, add them as well + var enums, enumsExist = b.enumsToGenerateTS[packageName] + if enumsExist { + for enumName, enum := range enums { + fqemumname := packageName + "." + enumName + if seen.Contains(fqemumname) { + continue + } + w.AddEnum(enum) + } + seenEnumsPackages.Add(packageName) + } + str, err := w.Convert(nil) if err != nil { return nil, err @@ -121,6 +147,35 @@ func (b *Bindings) GenerateModels() ([]byte, error) { models[packageName] = thisPackageCode } + // Add outstanding enums to the models that were not in packages with structs + for packageName, enumsToGenerate := range b.enumsToGenerateTS { + if seenEnumsPackages.Contains(packageName) { + continue + } + + thisPackageCode := "" + w := typescriptify.New() + w.WithPrefix(b.tsPrefix) + w.WithSuffix(b.tsSuffix) + w.WithInterface(b.tsInterface) + w.Namespace = packageName + w.WithBackupDir("") + + for enumName, enum := range enumsToGenerate { + fqemumname := packageName + "." + enumName + if seen.Contains(fqemumname) { + continue + } + w.AddEnum(enum) + } + str, err := w.Convert(nil) + if err != nil { + return nil, err + } + thisPackageCode += str + models[packageName] = thisPackageCode + } + // Sort the package names first to make the output deterministic sortedPackageNames := make([]string, 0) for packageName := range models { @@ -163,6 +218,39 @@ func (b *Bindings) WriteModels(modelsDir string) error { return nil } +func (b *Bindings) AddEnumToGenerateTS(e interface{}) { + enumType := reflect.TypeOf(e) + + var packageName string + var enumName string + // enums should be represented as array of all possible values + if hasElements(enumType) { + enum := enumType.Elem() + // simple enum represented by struct with Value/TSName fields + if enum.Kind() == reflect.Struct { + _, tsNamePresented := enum.FieldByName("TSName") + enumT, valuePresented := enum.FieldByName("Value") + if tsNamePresented && valuePresented { + packageName = getPackageName(enumT.Type.String()) + enumName = enumT.Type.Name() + } else { + return + } + // otherwise expecting implementation with TSName() https://github.com/tkrajina/typescriptify-golang-structs#enums-with-tsname + } else { + packageName = getPackageName(enumType.Elem().String()) + enumName = enumType.Elem().Name() + } + if b.enumsToGenerateTS[packageName] == nil { + b.enumsToGenerateTS[packageName] = make(map[string]interface{}) + } + if b.enumsToGenerateTS[packageName][enumName] != nil { + return + } + b.enumsToGenerateTS[packageName][enumName] = e + } +} + func (b *Bindings) AddStructToGenerateTS(packageName string, structName string, s interface{}) { if b.structsToGenerateTS[packageName] == nil { b.structsToGenerateTS[packageName] = make(map[string]interface{}) @@ -231,6 +319,13 @@ func (b *Bindings) SetTsSuffix(postfix string) *Bindings { return b } +func (b *Bindings) SetOutputType(outputType string) *Bindings { + if outputType == "interfaces" { + b.tsInterface = true + } + return b +} + func (b *Bindings) getAllStructNames() *slicer.StringSlicer { var result slicer.StringSlicer for packageName, structsToGenerate := range b.structsToGenerateTS { @@ -241,6 +336,16 @@ func (b *Bindings) getAllStructNames() *slicer.StringSlicer { return &result } +func (b *Bindings) getAllEnumNames() *slicer.StringSlicer { + var result slicer.StringSlicer + for packageName, enumsToGenerate := range b.enumsToGenerateTS { + for enumName := range enumsToGenerate { + result.Add(packageName + "." + enumName) + } + } + return &result +} + func (b *Bindings) hasExportedJSONFields(typeOf reflect.Type) bool { for i := 0; i < typeOf.NumField(); i++ { jsonFieldName := "" diff --git a/v2/internal/binding/binding_test/binding_conflicting_package_name_test.go b/v2/internal/binding/binding_test/binding_conflicting_package_name_test.go index 2309d6daf32..b37334ec325 100644 --- a/v2/internal/binding/binding_test/binding_conflicting_package_name_test.go +++ b/v2/internal/binding/binding_test/binding_conflicting_package_name_test.go @@ -42,7 +42,7 @@ func TestConflictingPackageName(t *testing.T) { // setup testLogger := &logger.Logger{} - b := binding.NewBindings(testLogger, []interface{}{&HandlerTest{}}, []interface{}{}, false) + b := binding.NewBindings(testLogger, []interface{}{&HandlerTest{}}, []interface{}{}, false, []interface{}{}) // then err := b.GenerateGoBindings(generationDir) diff --git a/v2/internal/binding/binding_test/binding_importedenum_test.go b/v2/internal/binding/binding_test/binding_importedenum_test.go new file mode 100644 index 00000000000..5b5b4419edb --- /dev/null +++ b/v2/internal/binding/binding_test/binding_importedenum_test.go @@ -0,0 +1,50 @@ +package binding_test + +import "github.com/wailsapp/wails/v2/internal/binding/binding_test/binding_test_import" + +type ImportedEnumStruct struct { + EnumValue binding_test_import.ImportedEnum `json:"EnumValue"` +} + +func (s ImportedEnumStruct) Get() ImportedEnumStruct { + return s +} + +var ImportedEnumTest = BindingTest{ + name: "ImportedEnum", + structs: []interface{}{ + &ImportedEnumStruct{}, + }, + enums: []interface{}{ + binding_test_import.AllImportedEnumValues, + }, + exemptions: nil, + shouldError: false, + want: `export namespace binding_test { + + export class ImportedEnumStruct { + EnumValue: binding_test_import.ImportedEnum; + + static createFrom(source: any = {}) { + return new ImportedEnumStruct(source); + } + + constructor(source: any = {}) { + if ('string' === typeof source) source = JSON.parse(source); + this.EnumValue = source["EnumValue"]; + } + } + + } + + export namespace binding_test_import { + + export enum ImportedEnum { + Value1 = "value1", + Value2 = "value2", + Value3 = "value3", + } + + } +`, +} diff --git a/v2/internal/binding/binding_test/binding_returned_promises_test.go b/v2/internal/binding/binding_test/binding_returned_promises_test.go index 837d5fad338..94941d0a390 100644 --- a/v2/internal/binding/binding_test/binding_returned_promises_test.go +++ b/v2/internal/binding/binding_test/binding_returned_promises_test.go @@ -59,7 +59,7 @@ func TestPromises(t *testing.T) { // setup testLogger := &logger.Logger{} - b := binding.NewBindings(testLogger, []interface{}{&PromisesTest{}}, []interface{}{}, false) + b := binding.NewBindings(testLogger, []interface{}{&PromisesTest{}}, []interface{}{}, false, []interface{}{}) // then err := b.GenerateGoBindings(generationDir) diff --git a/v2/internal/binding/binding_test/binding_test.go b/v2/internal/binding/binding_test/binding_test.go index c2e35191517..6d643b92d5a 100644 --- a/v2/internal/binding/binding_test/binding_test.go +++ b/v2/internal/binding/binding_test/binding_test.go @@ -13,6 +13,7 @@ import ( type BindingTest struct { name string structs []interface{} + enums []interface{} exemptions []interface{} want string shouldError bool @@ -20,8 +21,9 @@ type BindingTest struct { } type TsGenerationOptionsTest struct { - TsPrefix string - TsSuffix string + TsPrefix string + TsSuffix string + TsOutputType string } func TestBindings_GenerateModels(t *testing.T) { @@ -31,12 +33,17 @@ func TestBindings_GenerateModels(t *testing.T) { ImportedStructTest, ImportedSliceTest, ImportedMapTest, + ImportedEnumTest, NestedFieldTest, NonStringMapKeyTest, SingleFieldTest, MultistructTest, EmptyStructTest, GeneratedJsEntityTest, + GeneratedJsEntityWithIntEnumTest, + GeneratedJsEntityWithStringEnumTest, + GeneratedJsEntityWithEnumTsName, + GeneratedJsEntityWithNestedStructInterfacesTest, AnonymousSubStructTest, AnonymousSubStructMultiLevelTest, GeneratedJsEntityWithNestedStructTest, @@ -46,13 +53,14 @@ func TestBindings_GenerateModels(t *testing.T) { testLogger := &logger.Logger{} for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { - b := binding.NewBindings(testLogger, tt.structs, tt.exemptions, false) + b := binding.NewBindings(testLogger, tt.structs, tt.exemptions, false, tt.enums) for _, s := range tt.structs { err := b.Add(s) require.NoError(t, err) } b.SetTsPrefix(tt.TsPrefix) b.SetTsSuffix(tt.TsSuffix) + b.SetOutputType(tt.TsOutputType) got, err := b.GenerateModels() if (err != nil) != tt.shouldError { t.Errorf("GenerateModels() error = %v, shouldError %v", err, tt.shouldError) diff --git a/v2/internal/binding/binding_test/binding_test_import/binding_test_import.go b/v2/internal/binding/binding_test/binding_test_import/binding_test_import.go index 6b99d43bed3..e7080c6940e 100644 --- a/v2/internal/binding/binding_test/binding_test_import/binding_test_import.go +++ b/v2/internal/binding/binding_test/binding_test_import/binding_test_import.go @@ -13,3 +13,20 @@ type ASliceWrapper struct { type AMapWrapper struct { AMap map[string]binding_test_nestedimport.A `json:"AMap"` } + +type ImportedEnum string + +const ( + ImportedEnumValue1 ImportedEnum = "value1" + ImportedEnumValue2 ImportedEnum = "value2" + ImportedEnumValue3 ImportedEnum = "value3" +) + +var AllImportedEnumValues = []struct { + Value ImportedEnum + TSName string +}{ + {ImportedEnumValue1, "Value1"}, + {ImportedEnumValue2, "Value2"}, + {ImportedEnumValue3, "Value3"}, +} diff --git a/v2/internal/binding/binding_test/binding_tsgeneration_test.go b/v2/internal/binding/binding_test/binding_tsgeneration_test.go index d2c5349c58a..b627772fe18 100644 --- a/v2/internal/binding/binding_test/binding_tsgeneration_test.go +++ b/v2/internal/binding/binding_test/binding_tsgeneration_test.go @@ -275,3 +275,235 @@ export namespace binding_test { `, } + +type IntEnum int + +const ( + IntEnumValue1 IntEnum = iota + IntEnumValue2 + IntEnumValue3 +) + +var AllIntEnumValues = []struct { + Value IntEnum + TSName string +}{ + {IntEnumValue1, "Value1"}, + {IntEnumValue2, "Value2"}, + {IntEnumValue3, "Value3"}, +} + +type EntityWithIntEnum struct { + Name string `json:"name"` + Enum IntEnum `json:"enum"` +} + +func (e EntityWithIntEnum) Get() EntityWithIntEnum { + return e +} + +var GeneratedJsEntityWithIntEnumTest = BindingTest{ + name: "GeneratedJsEntityWithIntEnumTest", + structs: []interface{}{ + &EntityWithIntEnum{}, + }, + enums: []interface{}{ + AllIntEnumValues, + }, + exemptions: nil, + shouldError: false, + TsGenerationOptionsTest: TsGenerationOptionsTest{ + TsPrefix: "MY_PREFIX_", + TsSuffix: "_MY_SUFFIX", + }, + want: `export namespace binding_test { + + export enum MY_PREFIX_IntEnum_MY_SUFFIX { + Value1 = 0, + Value2 = 1, + Value3 = 2, + } + export class MY_PREFIX_EntityWithIntEnum_MY_SUFFIX { + name: string; + enum: MY_PREFIX_IntEnum_MY_SUFFIX; + + static createFrom(source: any = {}) { + return new MY_PREFIX_EntityWithIntEnum_MY_SUFFIX(source); + } + + constructor(source: any = {}) { + if ('string' === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.enum = source["enum"]; + } + } + + } +`, +} + +type StringEnum string + +const ( + StringEnumValue1 StringEnum = "value1" + StringEnumValue2 StringEnum = "value2" + StringEnumValue3 StringEnum = "value3" +) + +var AllStringEnumValues = []struct { + Value StringEnum + TSName string +}{ + {StringEnumValue1, "Value1"}, + {StringEnumValue2, "Value2"}, + {StringEnumValue3, "Value3"}, +} + +type EntityWithStringEnum struct { + Name string `json:"name"` + Enum StringEnum `json:"enum"` +} + +func (e EntityWithStringEnum) Get() EntityWithStringEnum { + return e +} + +var GeneratedJsEntityWithStringEnumTest = BindingTest{ + name: "GeneratedJsEntityWithStringEnumTest", + structs: []interface{}{ + &EntityWithStringEnum{}, + }, + enums: []interface{}{ + AllStringEnumValues, + }, + exemptions: nil, + shouldError: false, + TsGenerationOptionsTest: TsGenerationOptionsTest{ + TsPrefix: "MY_PREFIX_", + TsSuffix: "_MY_SUFFIX", + }, + want: `export namespace binding_test { + + export enum MY_PREFIX_StringEnum_MY_SUFFIX { + Value1 = "value1", + Value2 = "value2", + Value3 = "value3", + } + export class MY_PREFIX_EntityWithStringEnum_MY_SUFFIX { + name: string; + enum: MY_PREFIX_StringEnum_MY_SUFFIX; + + static createFrom(source: any = {}) { + return new MY_PREFIX_EntityWithStringEnum_MY_SUFFIX(source); + } + + constructor(source: any = {}) { + if ('string' === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.enum = source["enum"]; + } + } + + } +`, +} + +type EnumWithTsName string + +const ( + EnumWithTsName1 EnumWithTsName = "value1" + EnumWithTsName2 EnumWithTsName = "value2" + EnumWithTsName3 EnumWithTsName = "value3" +) + +var AllEnumWithTsNameValues = []EnumWithTsName{EnumWithTsName1, EnumWithTsName2, EnumWithTsName3} + +func (v EnumWithTsName) TSName() string { + switch v { + case EnumWithTsName1: + return "TsName1" + case EnumWithTsName2: + return "TsName2" + case EnumWithTsName3: + return "TsName3" + default: + return "???" + } +} + +type EntityWithEnumTsName struct { + Name string `json:"name"` + Enum EnumWithTsName `json:"enum"` +} + +func (e EntityWithEnumTsName) Get() EntityWithEnumTsName { + return e +} + +var GeneratedJsEntityWithEnumTsName = BindingTest{ + name: "GeneratedJsEntityWithEnumTsName", + structs: []interface{}{ + &EntityWithEnumTsName{}, + }, + enums: []interface{}{ + AllEnumWithTsNameValues, + }, + exemptions: nil, + shouldError: false, + TsGenerationOptionsTest: TsGenerationOptionsTest{ + TsPrefix: "MY_PREFIX_", + TsSuffix: "_MY_SUFFIX", + }, + want: `export namespace binding_test { + + export enum MY_PREFIX_EnumWithTsName_MY_SUFFIX { + TsName1 = "value1", + TsName2 = "value2", + TsName3 = "value3", + } + export class MY_PREFIX_EntityWithEnumTsName_MY_SUFFIX { + name: string; + enum: MY_PREFIX_EnumWithTsName_MY_SUFFIX; + + static createFrom(source: any = {}) { + return new MY_PREFIX_EntityWithEnumTsName_MY_SUFFIX(source); + } + + constructor(source: any = {}) { + if ('string' === typeof source) source = JSON.parse(source); + this.name = source["name"]; + this.enum = source["enum"]; + } + } + + } +`, +} + +var GeneratedJsEntityWithNestedStructInterfacesTest = BindingTest{ + name: "GeneratedJsEntityWithNestedStructInterfacesTest", + structs: []interface{}{ + &ParentEntity{}, + }, + exemptions: nil, + shouldError: false, + TsGenerationOptionsTest: TsGenerationOptionsTest{ + TsPrefix: "MY_PREFIX_", + TsSuffix: "_MY_SUFFIX", + TsOutputType: "interfaces", + }, + want: `export namespace binding_test { + + export interface MY_PREFIX_ChildEntity_MY_SUFFIX { + name: string; + childProp: number; + } + export interface MY_PREFIX_ParentEntity_MY_SUFFIX { + name: string; + ref: MY_PREFIX_ChildEntity_MY_SUFFIX; + parentProp: string; + } + + } +`, +} diff --git a/v2/internal/binding/binding_test/binding_type_alias_test.go b/v2/internal/binding/binding_test/binding_type_alias_test.go index 8e7c7ca6de9..498c5976ceb 100644 --- a/v2/internal/binding/binding_test/binding_type_alias_test.go +++ b/v2/internal/binding/binding_test/binding_type_alias_test.go @@ -41,7 +41,7 @@ func TestAliases(t *testing.T) { // setup testLogger := &logger.Logger{} - b := binding.NewBindings(testLogger, []interface{}{&AliasTest{}}, []interface{}{}, false) + b := binding.NewBindings(testLogger, []interface{}{&AliasTest{}}, []interface{}{}, false, []interface{}{}) // then err := b.GenerateGoBindings(generationDir) diff --git a/v2/internal/binding/generate_test.go b/v2/internal/binding/generate_test.go index 565fba31caa..8d6a833b817 100644 --- a/v2/internal/binding/generate_test.go +++ b/v2/internal/binding/generate_test.go @@ -25,7 +25,7 @@ type B struct { func TestNestedStruct(t *testing.T) { bind := &BindForTest{} - testBindings := NewBindings(logger.New(nil), []interface{}{bind}, []interface{}{}, false) + testBindings := NewBindings(logger.New(nil), []interface{}{bind}, []interface{}{}, false, []interface{}{}) namesStrSlicer := testBindings.getAllStructNames() names := []string{} diff --git a/v2/internal/project/project.go b/v2/internal/project/project.go index 34cbe88da59..0d84f18552f 100644 --- a/v2/internal/project/project.go +++ b/v2/internal/project/project.go @@ -242,8 +242,9 @@ type Bindings struct { } type TsGeneration struct { - Prefix string `json:"prefix"` - Suffix string `json:"suffix"` + Prefix string `json:"prefix"` + Suffix string `json:"suffix"` + OutputType string `json:"outputType"` } // Parse the given JSON data into a Project struct diff --git a/v2/internal/typescriptify/typescriptify.go b/v2/internal/typescriptify/typescriptify.go index bb72e6fb8b5..c06a8b2ec79 100644 --- a/v2/internal/typescriptify/typescriptify.go +++ b/v2/internal/typescriptify/typescriptify.go @@ -104,6 +104,7 @@ type TypeScriptify struct { Namespace string KnownStructs *slicer.StringSlicer + KnownEnums *slicer.StringSlicer } func New() *TypeScriptify { @@ -723,7 +724,16 @@ func (t *TypeScriptify) convertType(depth int, typeOf reflect.Type, customCode m } } else { // Simple field: t.logf(depth, "- simple field %s.%s", typeOf.Name(), field.Name) - err = builder.AddSimpleField(jsonFieldName, field, fldOpts) + // check if type is in known enum. If so, then replace TStype with enum name to avoid missing types + isKnownEnum := t.KnownEnums.Contains(getStructFQN(field.Type.String())) + if isKnownEnum { + err = builder.AddSimpleField(jsonFieldName, field, TypeOptions{ + TSType: getStructFQN(field.Type.String()), + TSTransform: fldOpts.TSTransform, + }) + } else { + err = builder.AddSimpleField(jsonFieldName, field, fldOpts) + } } if err != nil { return "", err diff --git a/v2/pkg/commands/bindings/bindings.go b/v2/pkg/commands/bindings/bindings.go index 1432acee11c..310b1e9afc0 100644 --- a/v2/pkg/commands/bindings/bindings.go +++ b/v2/pkg/commands/bindings/bindings.go @@ -21,6 +21,7 @@ type Options struct { GoModTidy bool TsPrefix string TsSuffix string + TsOutputType string } // GenerateBindings generates bindings for the Wails project in the given ProjectDirectory. @@ -65,6 +66,7 @@ func GenerateBindings(options Options) (string, error) { env := os.Environ() env = shell.SetEnv(env, "tsprefix", options.TsPrefix) env = shell.SetEnv(env, "tssuffix", options.TsSuffix) + env = shell.SetEnv(env, "tsoutputtype", options.TsOutputType) stdout, stderr, err = shell.RunCommandWithEnv(env, workingDirectory, filename) if err != nil { diff --git a/v2/pkg/commands/build/build.go b/v2/pkg/commands/build/build.go index 2223bf575f0..62c08e910d1 100644 --- a/v2/pkg/commands/build/build.go +++ b/v2/pkg/commands/build/build.go @@ -219,12 +219,17 @@ func GenerateBindings(buildOptions *Options) error { printBulletPoint("Generating bindings: ") } + if buildOptions.ProjectData.Bindings.TsGeneration.OutputType == "" { + buildOptions.ProjectData.Bindings.TsGeneration.OutputType = "classes" + } + // Generate Bindings output, err := bindings.GenerateBindings(bindings.Options{ - Tags: buildOptions.UserTags, - GoModTidy: !buildOptions.SkipModTidy, - TsPrefix: buildOptions.ProjectData.Bindings.TsGeneration.Prefix, - TsSuffix: buildOptions.ProjectData.Bindings.TsGeneration.Suffix, + Tags: buildOptions.UserTags, + GoModTidy: !buildOptions.SkipModTidy, + TsPrefix: buildOptions.ProjectData.Bindings.TsGeneration.Prefix, + TsSuffix: buildOptions.ProjectData.Bindings.TsGeneration.Suffix, + TsOutputType: buildOptions.ProjectData.Bindings.TsGeneration.OutputType, }) if err != nil { return err diff --git a/v2/pkg/options/options.go b/v2/pkg/options/options.go index 088e7c46ab8..66d56ceaa5e 100644 --- a/v2/pkg/options/options.go +++ b/v2/pkg/options/options.go @@ -63,6 +63,7 @@ type App struct { OnShutdown func(ctx context.Context) `json:"-"` OnBeforeClose func(ctx context.Context) (prevent bool) `json:"-"` Bind []interface{} + EnumBind []interface{} WindowStartState WindowStartState // ErrorFormatter overrides the formatting of errors returned by backend methods diff --git a/website/docs/guides/application-development.mdx b/website/docs/guides/application-development.mdx index 9d04fe9172d..78a6df3bcf1 100644 --- a/website/docs/guides/application-development.mdx +++ b/website/docs/guides/application-development.mdx @@ -144,6 +144,65 @@ func main() { } ``` +Also you might want to use Enums in your structs and have models for them on frontend. +In that case you should create array that will contain all possible enum values, instrument enum type and bind it to the app: + +```go {16-18} title="app.go" +type Weekday string + +const ( + Sunday Weekday = "Sunday" + Monday Weekday = "Monday" + Tuesday Weekday = "Tuesday" + Wednesday Weekday = "Wednesday" + Thursday Weekday = "Thursday" + Friday Weekday = "Friday" + Saturday Weekday = "Saturday" +) + +var AllWeekdays = []struct { + Value Weekday + TSName string +}{ + {Sunday, "SUNDAY"}, + {Monday, "MONDAY"}, + {Tuesday, "TUESDAY"}, + {Wednesday, "WEDNESDAY"}, + {Thursday, "THURSDAY"}, + {Friday, "FRIDAY"}, + {Saturday, "SATURDAY"}, +} +``` + +In the main application configuration, the `EnumBind` key is where we can tell Wails what we want to bind enums as well: + +```go {11-13} title="main.go" +func main() { + + app := NewApp() + + err := wails.Run(&options.App{ + Title: "My App", + Width: 800, + Height: 600, + OnStartup: app.startup, + OnShutdown: app.shutdown, + Bind: []interface{}{ + app, + }, + EnumBind: []interface{}{ + AllWeekdays, + }, + }) + if err != nil { + log.Fatal(err) + } +} + +``` + +This will add missing enums to your `model.ts` file. + More information on Binding can be found [here](../howdoesitwork.mdx#method-binding). ## Application Menu diff --git a/website/docs/howdoesitwork.mdx b/website/docs/howdoesitwork.mdx index e9f2c6e3dc4..6e23d5eb94d 100644 --- a/website/docs/howdoesitwork.mdx +++ b/website/docs/howdoesitwork.mdx @@ -206,6 +206,57 @@ You may bind as many structs as you like. Just make sure you create an instance ``` +You may bind enums types as well. +In that case you should create array that will contain all possible enum values, instrument enum type and bind it to the app via `EnumBind`: + +```go {16-18} title="app.go" +type Weekday string + +const ( + Sunday Weekday = "Sunday" + Monday Weekday = "Monday" + Tuesday Weekday = "Tuesday" + Wednesday Weekday = "Wednesday" + Thursday Weekday = "Thursday" + Friday Weekday = "Friday" + Saturday Weekday = "Saturday" +) + +var AllWeekdays = []struct { + Value Weekday + TSName string +}{ + {Sunday, "SUNDAY"}, + {Monday, "MONDAY"}, + {Tuesday, "TUESDAY"}, + {Wednesday, "WEDNESDAY"}, + {Thursday, "THURSDAY"}, + {Friday, "FRIDAY"}, + {Saturday, "SATURDAY"}, +} +``` + +```go {10-12} + //... + err := wails.Run(&options.App{ + Title: "Basic Demo", + Width: 1024, + Height: 768, + AssetServer: &assetserver.Options{ + Assets: assets, + }, + Bind: []interface{}{ + app, + &mystruct1{}, + &mystruct2{}, + }, + EnumBind: []interface{}{ + AllWeekdays, + }, + }) + +``` + When you run `wails dev` (or `wails generate module`), a frontend module will be generated containing the following: - JavaScript bindings for all bound methods diff --git a/website/docs/reference/options.mdx b/website/docs/reference/options.mdx index a2a19327725..caa2d0f806c 100644 --- a/website/docs/reference/options.mdx +++ b/website/docs/reference/options.mdx @@ -58,7 +58,14 @@ func main() { Bind: []interface{}{ app, }, + EnumBind: []interface{}{ + AllWeekdays, + }, ErrorFormatter: func(err error) any { return err.Error() }, + SingleInstanceLock: &options.SingleInstanceLock{ + UniqueId: "c9c8fd93-6758-4144-87d1-34bdb0a8bd60", + OnSecondInstanceLaunch: app.onSecondInstanceLaunch, + }, Windows: &windows.Options{ WebviewIsTransparent: false, WindowIsTranslucent: false, @@ -92,6 +99,8 @@ func main() { FullSizeContent: false, UseToolbar: false, HideToolbarSeparator: true, + OnFileOpen: app.onFileOpen, + OnUrlOpen: app.onUrlOpen, }, Appearance: mac.NSAppearanceNameDarkAqua, WebviewIsTransparent: true, @@ -479,6 +488,13 @@ A slice of struct instances defining methods that need to be bound to the fronte Name: Bind
Type: `[]interface{}` +### EnumBind + +A slice of Enum arrays that need to be bound to the frontend. + +Name: EnumBind
+Type: `[]interface{}` + ### ErrorFormatter A function that determines how errors are formatted when returned by a JS-to-Go @@ -487,6 +503,28 @@ method call. The returned value will be marshalled as JSON. Name: ErrorFormatter
Type: `func (error) any` +### SingleInstanceLock + +Enables single instance locking. This means that only one instance of your application can be running at a time. + +Name: SingleInstanceLock
+Type: `*options.SingleInstanceLock` + +#### UniqueId + +This id is used to generate the mutex name on Windows and macOS and the dbus name on Linux. Use a UUID to ensure that the id is unique. + +Name: UniqueId
+Type: `string` + +#### OnSecondInstanceLaunch + +Callback that is called when a second instance of your app is launched. + +Name: OnSecondInstanceLaunch
+Type: `func(secondInstanceData SecondInstanceData)` + + ### Windows This defines [Windows specific options](#windows). @@ -797,6 +835,20 @@ with [WebviewIsTransparent](#WebviewIsTransparent) to make frosty-looking applic Name: WindowIsTranslucent
Type: `bool` +#### OnFileOpen + +Callback that is called when a file is opened with the application. + +Name: OnFileOpen
+Type: `func(filePath string)` + +#### OnUrlOpen + +Callback that is called when a URL is opened with the application. + +Name: OnUrlOpen
+Type: `func(filePath string)` + #### Preferences The Preferences struct provides the ability to configure the Webview preferences. @@ -804,7 +856,7 @@ The Preferences struct provides the ability to configure the Webview preferences Name: Preferences
Type: [`*mac.Preferences`](#preferences-struct) -##### Preferences struct +##### Preferences struct You can specify the webview preferences. @@ -812,7 +864,7 @@ You can specify the webview preferences. type Preferences struct { TabFocusesLinks u.Bool TextInteractionEnabled u.Bool - FullscreenEnabled u.Bool + FullscreenEnabled u.Bool } ``` diff --git a/website/docs/reference/project-config.mdx b/website/docs/reference/project-config.mdx index 8e763502bcc..3a6f09495a0 100644 --- a/website/docs/reference/project-config.mdx +++ b/website/docs/reference/project-config.mdx @@ -73,14 +73,52 @@ The project config resides in the `wails.json` file in the project directory. Th // The copyright of the product. Default: 'Copyright.........' "copyright": "", // A short comment of the app. Default: 'Built using Wails (https://wails.app)' - "comments": "" + "comments": "", + // File associations for the app + "fileAssociations": [ + { + // The extension (minus the leading period). e.g. png + "ext": "wails", + // The name. e.g. PNG File + "name": "Wails", + // Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. + "description": "Wails file", + // The icon name without extension. Icons should be located in build folder. Proper icons will be generated from .png file for both macOS and Windows) + "iconName": "fileIcon", + // macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. + "role": "Editor" + }, + ], + // Custom URI protocols that should be opened by the application + "protocols": [ + { + // protocol scheme. e.g. myapp + "scheme": "myapp", + // Windows-only. The description. It is displayed on the `Type` column on Windows Explorer. + "description": "Myapp protocol", + // macOS-only. The app’s role with respect to the type. Corresponds to CFBundleTypeRole. + "role": "Editor" + } + ] }, // 'multiple': One installer per architecture. 'single': Single universal installer for all architectures being built. Default: 'multiple' "nsisType": "", // Whether the app should be obfuscated. Default: false "obfuscated": "", // The arguments to pass to the garble command when using the obfuscated flag - "garbleargs": "" + "garbleargs": "", + // Bindings configurations + "bindings": { + // model.ts file generation config + "ts_generation": { + // All generated JavaScript entities will be prefixed with this value + "prefix": "", + // All generated JavaScript entities will be suffixed with this value + "suffix": "", + // Type of output to generate (classes|interfaces) + "outputType": "classes", + } + } } ``` diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index 8b65b549057..c6d63750617 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -29,6 +29,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added CPU/GPU/Memory detection for `wails doctor`. Added by @leaanthony in #d51268b8d0680430f3a614775b13e6cd2b906d1c - The [AssetServer](/docs/reference/options#assetserver) now injects the runtime/IPC into all index html files and into all html files returned when requesting a folder path. Added by @stffabi in [PR](https://github.com/wailsapp/wails/pull/2203) - Added Custom Protocol Schemes associations support for macOS and Windows. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3000) +– Added support for TS interfaces generation as an option. Add support for Enums in TS types. Added by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3047) ### Changed diff --git a/website/static/schemas/config.v2.json b/website/static/schemas/config.v2.json index f215415e641..7a6d26f1509 100644 --- a/website/static/schemas/config.v2.json +++ b/website/static/schemas/config.v2.json @@ -251,6 +251,33 @@ "garbleargs": { "type": "string", "description": "The arguments to pass to the garble command when using the obfuscated flag" + }, + "bindings": { + "type": "object", + "description": "Bindings configurations", + "properties": { + "ts_generation": { + "type": "object", + "description": "model.ts file generation config", + "properties": { + "prefix": { + "type": "string", + "description": "All generated JavaScript entities will be prefixed with this value" + }, + "suffix": { + "type": "string", + "description": "All generated JavaScript entities will be suffixed with this value" + }, + "outputType": { + "allOf": [ + { + "$ref": "#/definitions/BindingsOutputTypes" + } + ] + } + } + } + } } }, "dependencies": { @@ -346,26 +373,25 @@ ] } ] - } - }, - "bindings": { - "type": "object", - "description": "Bindings configurations", - "properties": { - "ts_generation": { - "type": "object", - "description": "model.ts file generation config", - "properties": { - "prefix": { - "type": "string", - "description": "All generated JavaScript entities will be prefixed with this value" - }, - "suffix": { - "type": "string", - "description": "All generated JavaScript entities will be suffixed with this value" - } + }, + "BindingsOutputTypes": { + "description": "Type of output to generate", + "oneOf": [ + { + "description": "Classes", + "type": "string", + "enum": [ + "classes" + ] + }, + { + "description": "Interfaces", + "type": "string", + "enum": [ + "interfaces" + ] } - } + ] } } } From 674042df36ac3e98d788001d81f3765e2b03cbc0 Mon Sep 17 00:00:00 2001 From: Andrey Pshenkin Date: Sat, 25 Nov 2023 20:16:24 +0000 Subject: [PATCH 53/87] fix obfuscated build binding ordering (#3071) * fix obfuscated build binding ordering * remove unused string method * add to changelog --------- Co-authored-by: Lea Anthony --- v2/internal/binding/db.go | 33 +++++++++++++++++---------------- website/src/pages/changelog.mdx | 5 +++-- 2 files changed, 20 insertions(+), 18 deletions(-) diff --git a/v2/internal/binding/db.go b/v2/internal/binding/db.go index 1fc7e8c662e..f7b793839d8 100644 --- a/v2/internal/binding/db.go +++ b/v2/internal/binding/db.go @@ -2,7 +2,6 @@ package binding import ( "encoding/json" - "sort" "sync" "unsafe" ) @@ -17,17 +16,22 @@ type DB struct { methodMap map[string]*BoundMethod // This uses ids to reference bound methods at runtime - obfuscatedMethodMap map[int]*BoundMethod + obfuscatedMethodArray []*ObfuscatedMethod // Lock to ensure sync access to the data lock sync.RWMutex } +type ObfuscatedMethod struct { + method *BoundMethod + methodName string +} + func newDB() *DB { return &DB{ - store: make(map[string]map[string]map[string]*BoundMethod), - methodMap: make(map[string]*BoundMethod), - obfuscatedMethodMap: make(map[int]*BoundMethod), + store: make(map[string]map[string]map[string]*BoundMethod), + methodMap: make(map[string]*BoundMethod), + obfuscatedMethodArray: []*ObfuscatedMethod{}, } } @@ -65,7 +69,11 @@ func (d *DB) GetObfuscatedMethod(id int) *BoundMethod { d.lock.RLock() defer d.lock.RUnlock() - return d.obfuscatedMethodMap[id] + if len(d.obfuscatedMethodArray) <= id { + return nil + } + + return d.obfuscatedMethodArray[id].method } // AddMethod adds the given method definition to the db using the given qualified path: packageName.structName.methodName @@ -96,6 +104,7 @@ func (d *DB) AddMethod(packageName string, structName string, methodName string, // Store in the methodMap key := packageName + "." + structName + "." + methodName d.methodMap[key] = methodDefinition + d.obfuscatedMethodArray = append(d.obfuscatedMethodArray, &ObfuscatedMethod{method: methodDefinition, methodName: key}) } // ToJSON converts the method map to JSON @@ -117,17 +126,9 @@ func (d *DB) ToJSON() (string, error) { func (d *DB) UpdateObfuscatedCallMap() map[string]int { mappings := make(map[string]int) - // Iterate map keys and sort them - keys := make([]string, 0, len(d.methodMap)) - for k := range d.methodMap { - keys = append(keys, k) + for id, k := range d.obfuscatedMethodArray { + mappings[k.methodName] = id } - sort.Strings(keys) - // Iterate sorted keys and add to obfuscated method map - for id, k := range keys { - mappings[k] = id - d.obfuscatedMethodMap[id] = d.methodMap[k] - } return mappings } diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index c6d63750617..b539d5ef1c5 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -41,8 +41,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 #### Fixed - Fixed typo on docs/reference/options page. Added by [@pylotlight](https://github.com/pylotlight) in [PR](https://github.com/wailsapp/wails/pull/2887) -- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR] in (https://github.com/wailsapp/wails/pull/2941) -- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR] in (https://github.com/wailsapp/wails/pull/3002) +- Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR](https://github.com/wailsapp/wails/pull/2941) +- Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/3002) +- Fixed binding mapping for obfuscated build, when binding are in different structs. Fixed by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3071) ## v2.6.0 - 2023-09-06 From b2839ed7197fa2499767d0d42d892387272d1245 Mon Sep 17 00:00:00 2001 From: Andrey Pshenkin Date: Sun, 26 Nov 2023 12:14:50 +0000 Subject: [PATCH 54/87] Pass obfuscation settings from wails.json (#3080) * pass obfuscation settings from wails.json * add to changelog --- v2/cmd/wails/build.go | 10 ++++++++++ website/src/pages/changelog.mdx | 1 + 2 files changed, 11 insertions(+) diff --git a/v2/cmd/wails/build.go b/v2/cmd/wails/build.go index 858ecb4c9e2..7364df8bae5 100644 --- a/v2/cmd/wails/build.go +++ b/v2/cmd/wails/build.go @@ -49,6 +49,16 @@ func buildApplication(f *flags.Build) error { return err } + // Set obfuscation from project file + if projectOptions.Obfuscated { + f.Obfuscated = projectOptions.Obfuscated + } + + // Set garble args from project file + if projectOptions.GarbleArgs != "" { + f.GarbleArgs = projectOptions.GarbleArgs + } + // Create BuildOptions buildOptions := &build.Options{ Logger: logger, diff --git a/website/src/pages/changelog.mdx b/website/src/pages/changelog.mdx index b539d5ef1c5..3cbadd5b616 100644 --- a/website/src/pages/changelog.mdx +++ b/website/src/pages/changelog.mdx @@ -44,6 +44,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixed issue with npm being called npm20 on openSUSE-Tumbleweed. Fixed by @TuffenDuffen in [PR](https://github.com/wailsapp/wails/pull/2941) - Fixed memory corruption on Windows when using accelerator keys. Fixed by @stffabi in [PR](https://github.com/wailsapp/wails/pull/3002) - Fixed binding mapping for obfuscated build, when binding are in different structs. Fixed by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3071) +- Fixed issue with obfuscation settings in wails.json. Fixed by @APshenkin in [PR](https://github.com/wailsapp/wails/pull/3080) ## v2.6.0 - 2023-09-06 From 5547792b6844376142cc2adb2514d1312fe73c65 Mon Sep 17 00:00:00 2001 From: Marcus Crane Date: Mon, 27 Nov 2023 21:27:32 +1300 Subject: [PATCH 55/87] Update Gon usage to reflect Bearer fork (#3063) Signed-off-by: Marcus Crane --- website/docs/guides/signing.mdx | 27 +++++++++++++++++++-------- 1 file changed, 19 insertions(+), 8 deletions(-) diff --git a/website/docs/guides/signing.mdx b/website/docs/guides/signing.mdx index b2a060c5f62..e57e99e76de 100644 --- a/website/docs/guides/signing.mdx +++ b/website/docs/guides/signing.mdx @@ -232,7 +232,7 @@ jobs: path: build/bin/* ``` -For code signing on macOS, [gon](https://github.com/mitchellh/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and +For code signing on macOS, [gon](https://github.com/Bearer/gon) is a very handy tool for code signing and communicating with Apple servers, also written in Go, and will be used in this guide. After the `Build Wails app` step, add the following to the workflow: @@ -241,7 +241,7 @@ After the `Build Wails app` step, add the following to the workflow: - name: MacOS download gon for code signing and app notarization if: matrix.platform == 'macos-latest' run: | - brew install mitchellh/gon/gon + brew install Bearer/tap/gon ``` Now we need to configure some gon config files in our `build/darwin` directory: @@ -254,19 +254,30 @@ Now we need to configure some gon config files in our `build/darwin` directory: "bundle_id": "app.myapp", "apple_id": { "username": "my-appleid@email.com", - "password": "@env:APPLE_PASSWORD" + "password": "@env:APPLE_PASSWORD", + "provider": "ABCDE12345" }, "sign": { - "application_identity": "Developer ID Application: My Name" + "application_identity": "Developer ID Application: Human User" } } ``` -Where `source` is your Wails binary, `bundle_id` is your bundle ID, `apple_id` contains your Apple ID username and App-Specific password -which you created earlier, and `sign.application_identity` is your identity which you can find by running the following command: +Here is a brief break down of the above fields: + +- `source`: The location of your wails binary to be signed +- `apple_id`: + - `username`: Your Apple ID email address + - `password`: Your app-specific password, referenced using Gon's environment variable syntax + - `provider`: Your team ID for your App Store Connect account +- `sign`: + - `application_identity`: Your Apple developer identity + +Your developer identity and team ID can both by found on macOS by running the following command: ```bash -security find-identity -v -p codesigning +$ security find-identity -v -p codesigning + 1) 00000000000000000000000000000000000000000 "Developer ID Application: Human User (ABCDE12345)" ``` 2. entitlements.plist: @@ -369,7 +380,7 @@ jobs: - name: MacOS download gon for code signing and app notarization if: matrix.platform == 'macos-latest' run: | - brew install mitchellh/gon/gon + brew install Bearer/tap/gon - name: Import Code-Signing Certificates for macOS if: matrix.platform == 'macos-latest' uses: Apple-Actions/import-codesign-certs@v1 From 7d14a75ff35dc085f007b010d0974a709c3b6fbe Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 29 Nov 2023 19:25:54 +1100 Subject: [PATCH 56/87] chore: update sponsors.svg (#3085) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 13 +++++-------- 1 file changed, 5 insertions(+), 8 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index cbc17a445b2..1b27f97d605 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -1,5 +1,5 @@ - +
- - - - - - + + + - + From 92c9f546ebef5f63eaad3c93fdafa6b4fa2fca4d Mon Sep 17 00:00:00 2001 From: "semgrep.dev on behalf of @leaanthony" Date: Sun, 3 Dec 2023 21:42:21 +0000 Subject: [PATCH 57/87] Add Semgrep CI --- .github/workflows/semgrep.yml | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) create mode 100644 .github/workflows/semgrep.yml diff --git a/.github/workflows/semgrep.yml b/.github/workflows/semgrep.yml new file mode 100644 index 00000000000..453e4cb8546 --- /dev/null +++ b/.github/workflows/semgrep.yml @@ -0,0 +1,24 @@ +on: + workflow_dispatch: {} + pull_request: {} + push: + branches: + - main + - master + paths: + - .github/workflows/semgrep.yml + schedule: + # random HH:MM to avoid a load spike on GitHub Actions at 00:00 + - cron: 14 16 * * * +name: Semgrep +jobs: + semgrep: + name: semgrep/ci + runs-on: ubuntu-20.04 + env: + SEMGREP_APP_TOKEN: ${{ secrets.SEMGREP_APP_TOKEN }} + container: + image: returntocorp/semgrep + steps: + - uses: actions/checkout@v3 + - run: semgrep ci From 3c5d850a30290a3c2f386ae1cfbac6952373ee62 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Mon, 4 Dec 2023 18:47:20 +1100 Subject: [PATCH 58/87] chore: update sponsors.svg (#3088) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 28 ++++++++++++---------------- 1 file changed, 12 insertions(+), 16 deletions(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 1b27f97d605..7ea8aaf265d 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -125,54 +125,50 @@ text { Helpers - + - + - + - + - + - + - + - + - + - + - - - - - + - + From a1f067377fa4deb87648bbfc2d2e551d9557c90c Mon Sep 17 00:00:00 2001 From: gurshan <116788218+gursheyss@users.noreply.github.com> Date: Sun, 3 Dec 2023 23:49:34 -0800 Subject: [PATCH 59/87] add .gitignore modification to sveltekit doc (#3074) * add gitignore modification to sveltekit doc * Fix Typo --------- Co-authored-by: Lea Anthony --- website/docs/guides/sveltekit.mdx | 3 +++ 1 file changed, 3 insertions(+) diff --git a/website/docs/guides/sveltekit.mdx b/website/docs/guides/sveltekit.mdx index 3f82eb813eb..e0357ca3c51 100644 --- a/website/docs/guides/sveltekit.mdx +++ b/website/docs/guides/sveltekit.mdx @@ -24,6 +24,9 @@ This guide will go into: ##### Modify main.go. - The first comment `//go:embed all:frontend/dist` needs to be changed to `//go:embed all:frontend/build` +##### Modify .gitignore +- The line `frontend/dist` needs to be replaced with `frontend/build` + ##### Install/remove dependencies using your favorite package manager. - Navigate into your "frontend" folder. - `npm i` From 4cbc7f3eecf0d3d641394374fa33d21daccfab03 Mon Sep 17 00:00:00 2001 From: venkadeshshiva <40267253+venkadeshshiva@users.noreply.github.com> Date: Mon, 4 Dec 2023 13:21:27 +0530 Subject: [PATCH 60/87] added wails-template-nextjs-app-router under react (#3082) --- website/docs/community/templates.mdx | 1 + 1 file changed, 1 insertion(+) diff --git a/website/docs/community/templates.mdx b/website/docs/community/templates.mdx index 1a781c758a2..98726c60aff 100644 --- a/website/docs/community/templates.mdx +++ b/website/docs/community/templates.mdx @@ -40,6 +40,7 @@ If you are unsure about a template, inspect `package.json` and `wails.json` for - [wails-react-template](https://github.com/AlienRecall/wails-react-template) - A template using reactjs - [wails-react-template](https://github.com/flin7/wails-react-template) - A minimal template for React that supports live development - [wails-template-nextjs](https://github.com/LGiki/wails-template-nextjs) - A template using Next.js and TypeScript +- [wails-template-nextjs-app-router](https://github.com/thisisvk-in/wails-template-nextjs-app-router) - A template using Next.js and TypeScript with App router - [wails-vite-react-ts-tailwind-template](https://github.com/hotafrika/wails-vite-react-ts-tailwind-template) - A template for React + TypeScript + Vite + TailwindCSS - [wails-vite-react-ts-tailwind-shadcnui-template](https://github.com/Mahcks/wails-vite-react-tailwind-shadcnui-ts) - A template with Vite, React, TypeScript, TailwindCSS, and shadcn/ui From 07fc8e37077683044454545fcc359aad459b3be6 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sat, 9 Dec 2023 11:20:45 +1100 Subject: [PATCH 61/87] chore: update sponsors.svg (#3105) Co-authored-by: leaanthony --- website/static/img/sponsors.svg | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/website/static/img/sponsors.svg b/website/static/img/sponsors.svg index 7ea8aaf265d..ceb2bbcf8ee 100644 --- a/website/static/img/sponsors.svg +++ b/website/static/img/sponsors.svg @@ -75,7 +75,7 @@ text { Buying Coffee - + From 6a8322cdb5194048c19b3b7f20fcabd32a00f9df Mon Sep 17 00:00:00 2001 From: Lea Anthony Date: Sat, 9 Dec 2023 16:32:29 +1100 Subject: [PATCH 62/87] v2.7.0 --- v2/cmd/wails/internal/version.txt | 2 +- .../version-v2.7.0.json | 38 + .../version-v2.7.0/community/links.mdx | 26 + .../community/showcase/bulletinboard.mdx | 10 + .../community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 12 + .../community/showcase/filehound.mdx | 16 + .../community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../community/showcase/october.mdx | 14 + .../community/showcase/optimus.mdx | 10 + .../community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 12 + .../community/showcase/riftshare.mdx | 21 + .../community/showcase/scriptbar.mdx | 10 + .../community/showcase/surge.mdx | 10 + .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 10 + .../community/showcase/warmine.mdx | 19 + .../community/showcase/wombat.mdx | 10 + .../version-v2.7.0/community/showcase/ytd.mdx | 10 + .../version-v2.7.0/community/templates.mdx | 69 ++ .../gettingstarted/building.mdx | 22 + .../gettingstarted/development.mdx | 16 + .../gettingstarted/firstproject.mdx | 130 +++ .../gettingstarted/installation.mdx | 90 ++ .../version-v2.7.0/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../guides/crossplatform-build.mdx | 66 ++ .../guides/custom-protocol-schemes.mdx | 204 ++++ .../version-v2.7.0/guides/dynamic-assets.mdx | 136 +++ .../guides/file-association.mdx | 244 +++++ .../version-v2.7.0/guides/frameless.mdx | 87 ++ .../version-v2.7.0/guides/frontend.mdx | 72 ++ .../version-v2.7.0/guides/ides.mdx | 127 +++ .../guides/linux-distro-support.mdx | 103 +++ .../version-v2.7.0/guides/linux.mdx | 18 + .../guides/local-development.mdx | 55 ++ .../version-v2.7.0/guides/mac-appstore.mdx | 97 ++ .../version-v2.7.0/guides/manual-builds.mdx | 95 ++ .../version-v2.7.0/guides/migrating.mdx | 191 ++++ .../version-v2.7.0/guides/mouse-buttons.mdx | 25 + .../version-v2.7.0/guides/obfuscated.mdx | 40 + .../version-v2.7.0/guides/overscroll.mdx | 10 + .../version-v2.7.0}/guides/routing.mdx | 0 .../version-v2.7.0/guides/signing.mdx | 387 ++++++++ .../guides/single-instance-lock.mdx | 81 ++ .../version-v2.7.0/guides/sveltekit.mdx | 153 +++ .../version-v2.7.0/guides/templates.mdx | 97 ++ .../version-v2.7.0/guides/troubleshooting.mdx | 368 ++++++++ .../version-v2.7.0/guides/vscode.mdx | 82 ++ .../guides/windows-installer.mdx | 58 ++ .../version-v2.7.0/guides/windows.mdx | 61 ++ .../version-v2.7.0/howdoesitwork.mdx | 369 ++++++++ .../version-v2.7.0/introduction.mdx | 75 ++ .../version-v2.7.0/reference/cli.mdx | 241 +++++ .../version-v2.7.0/reference/menus.mdx | 230 +++++ .../version-v2.7.0/reference/options.mdx | 853 +++++++++++++++++ .../reference/project-config.mdx | 92 ++ .../reference/runtime/browser.mdx | 13 + .../reference/runtime/clipboard.mdx | 23 + .../reference/runtime/dialog.mdx | 302 ++++++ .../reference/runtime/events.mdx | 37 + .../reference/runtime/intro.mdx | 85 ++ .../version-v2.7.0/reference/runtime/log.mdx | 130 +++ .../version-v2.7.0/reference/runtime/menu.mdx | 25 + .../reference/runtime/screen.mdx | 38 + .../reference/runtime/window.mdx | 227 +++++ .../version-v2.7.0/tutorials/dogsapi.mdx | 245 +++++ .../version-v2.7.0/tutorials/helloworld.mdx | 123 +++ .../version-v2.7.0}/appendix/_category_.json | 0 .../version-v2.7.0}/community/_category_.json | 0 .../version-v2.7.0/community/links.mdx | 26 + .../community/showcase/_category_.json | 0 .../community/showcase/bulletinboard.mdx | 10 + .../community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 12 + .../community/showcase/filehound.mdx | 16 + .../community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../community/showcase/october.mdx | 14 + .../community/showcase/optimus.mdx | 10 + .../community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 12 + .../community/showcase/riftshare.mdx | 21 + .../community/showcase/scriptbar.mdx | 10 + .../community/showcase/surge.mdx | 10 + .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 10 + .../community/showcase/warmine.mdx | 19 + .../community/showcase/wombat.mdx | 10 + .../version-v2.7.0/community/showcase/ytd.mdx | 10 + .../version-v2.7.0/community/templates.mdx | 69 ++ .../gettingstarted/_category_.json | 0 .../gettingstarted/building.mdx | 22 + .../gettingstarted/development.mdx | 16 + .../gettingstarted/firstproject.mdx | 130 +++ .../gettingstarted/installation.mdx | 89 ++ .../version-v2.7.0}/guides/_category_.json | 0 .../version-v2.7.0/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../guides/crossplatform-build.mdx | 66 ++ .../guides/custom-protocol-schemes.mdx | 204 ++++ .../version-v2.7.0/guides/dynamic-assets.mdx | 136 +++ .../guides/file-association.mdx | 244 +++++ .../version-v2.7.0/guides/frameless.mdx | 87 ++ .../version-v2.7.0/guides/frontend.mdx | 72 ++ .../version-v2.7.0/guides/ides.mdx | 127 +++ .../guides/linux-distro-support.mdx | 103 +++ .../version-v2.7.0/guides/linux.mdx | 18 + .../guides/local-development.mdx | 55 ++ .../version-v2.7.0/guides/mac-appstore.mdx | 97 ++ .../version-v2.7.0/guides/manual-builds.mdx | 95 ++ .../version-v2.7.0/guides/migrating.mdx | 191 ++++ .../version-v2.7.0/guides/mouse-buttons.mdx | 25 + .../version-v2.7.0/guides/obfuscated.mdx | 40 + .../version-v2.7.0/guides/overscroll.mdx | 10 + .../version-v2.7.0/guides/routing.mdx | 47 + .../version-v2.7.0/guides/signing.mdx | 387 ++++++++ .../guides/single-instance-lock.mdx | 81 ++ .../version-v2.7.0/guides/sveltekit.mdx | 153 +++ .../version-v2.7.0/guides/templates.mdx | 97 ++ .../version-v2.7.0/guides/troubleshooting.mdx | 368 ++++++++ .../version-v2.7.0/guides/vscode.mdx | 82 ++ .../guides/windows-installer.mdx | 58 ++ .../version-v2.7.0/guides/windows.mdx | 61 ++ .../version-v2.7.0/howdoesitwork.mdx | 369 ++++++++ .../version-v2.7.0/introduction.mdx | 75 ++ .../version-v2.7.0}/reference/_category_.json | 0 .../version-v2.7.0/reference/cli.mdx | 240 +++++ .../version-v2.7.0/reference/menus.mdx | 230 +++++ .../version-v2.7.0/reference/options.mdx | 853 +++++++++++++++++ .../reference/project-config.mdx | 91 ++ .../reference/runtime/_category_.json | 0 .../reference/runtime/browser.mdx | 13 + .../reference/runtime/clipboard.mdx | 23 + .../reference/runtime/dialog.mdx | 302 ++++++ .../reference/runtime/events.mdx | 37 + .../reference/runtime/intro.mdx | 85 ++ .../version-v2.7.0/reference/runtime/log.mdx | 130 +++ .../version-v2.7.0/reference/runtime/menu.mdx | 25 + .../reference/runtime/screen.mdx | 38 + .../reference/runtime/window.mdx | 227 +++++ .../version-v2.7.0}/tutorials/_category_.json | 0 .../version-v2.7.0/tutorials/dogsapi.mdx | 245 +++++ .../version-v2.7.0/tutorials/helloworld.mdx | 123 +++ .../version-v2.7.0/community/links.mdx | 26 + .../community/showcase/bulletinboard.mdx | 0 .../community/showcase/emailit.mdx | 0 .../community/showcase/encrypteasy.mdx | 12 + .../community/showcase/filehound.mdx | 16 + .../community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../community/showcase/october.mdx | 0 .../community/showcase/optimus.mdx | 10 + .../community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 0 .../community/showcase/riftshare.mdx | 0 .../community/showcase/scriptbar.mdx | 0 .../community/showcase/surge.mdx | 0 .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 0 .../community/showcase/warmine.mdx | 19 + .../community/showcase/wombat.mdx | 10 + .../community/showcase/ytd.mdx | 0 .../version-v2.7.0/community/templates.mdx | 69 ++ .../gettingstarted/building.mdx | 22 + .../gettingstarted/development.mdx | 16 + .../gettingstarted/firstproject.mdx | 130 +++ .../gettingstarted/installation.mdx | 89 ++ .../version-v2.7.0/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../guides/crossplatform-build.mdx | 66 ++ .../guides/custom-protocol-schemes.mdx | 204 ++++ .../version-v2.7.0/guides/dynamic-assets.mdx | 136 +++ .../guides/file-association.mdx | 244 +++++ .../version-v2.7.0/guides/frameless.mdx | 87 ++ .../version-v2.7.0/guides/frontend.mdx | 72 ++ .../version-v2.7.0/guides/ides.mdx | 127 +++ .../guides/linux-distro-support.mdx | 103 +++ .../version-v2.7.0/guides/linux.mdx | 18 + .../guides/local-development.mdx | 55 ++ .../version-v2.7.0/guides/mac-appstore.mdx | 97 ++ .../version-v2.7.0/guides/manual-builds.mdx | 95 ++ .../version-v2.7.0/guides/migrating.mdx | 191 ++++ .../version-v2.7.0/guides/mouse-buttons.mdx | 25 + .../version-v2.7.0/guides/obfuscated.mdx | 40 + .../version-v2.7.0/guides/overscroll.mdx | 10 + .../version-v2.7.0/guides/routing.mdx | 47 + .../version-v2.7.0/guides/signing.mdx | 387 ++++++++ .../guides/single-instance-lock.mdx | 81 ++ .../version-v2.7.0/guides/sveltekit.mdx | 153 +++ .../version-v2.7.0/guides/templates.mdx | 97 ++ .../version-v2.7.0/guides/troubleshooting.mdx | 368 ++++++++ .../version-v2.7.0/guides/vscode.mdx | 82 ++ .../guides/windows-installer.mdx | 58 ++ .../version-v2.7.0/guides/windows.mdx | 61 ++ .../version-v2.7.0/howdoesitwork.mdx | 369 ++++++++ .../version-v2.7.0/introduction.mdx | 75 ++ .../version-v2.7.0/reference/cli.mdx | 240 +++++ .../version-v2.7.0/reference/menus.mdx | 230 +++++ .../version-v2.7.0/reference/options.mdx | 853 +++++++++++++++++ .../reference/project-config.mdx | 2 +- .../reference/runtime/browser.mdx | 13 + .../reference/runtime/clipboard.mdx | 23 + .../reference/runtime/dialog.mdx | 302 ++++++ .../reference/runtime/events.mdx | 37 + .../reference/runtime/intro.mdx | 85 ++ .../version-v2.7.0/reference/runtime/log.mdx | 130 +++ .../version-v2.7.0/reference/runtime/menu.mdx | 25 + .../reference/runtime/screen.mdx | 38 + .../reference/runtime/window.mdx | 227 +++++ .../version-v2.7.0/tutorials/dogsapi.mdx | 245 +++++ .../version-v2.7.0/tutorials/helloworld.mdx | 123 +++ .../version-v2.7.0/community/links.mdx | 26 + .../community/showcase/bulletinboard.mdx | 10 + .../community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 12 + .../community/showcase/filehound.mdx | 16 + .../community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../community/showcase/october.mdx | 14 + .../community/showcase/optimus.mdx | 10 + .../community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 12 + .../community/showcase/riftshare.mdx | 21 + .../community/showcase/scriptbar.mdx | 10 + .../community/showcase/surge.mdx | 10 + .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 10 + .../community/showcase/warmine.mdx | 19 + .../community/showcase/wombat.mdx | 10 + .../version-v2.7.0/community/showcase/ytd.mdx | 10 + .../version-v2.7.0/community/templates.mdx | 69 ++ .../gettingstarted/building.mdx | 22 + .../gettingstarted/development.mdx | 16 + .../gettingstarted/firstproject.mdx | 130 +++ .../gettingstarted/installation.mdx | 90 ++ .../version-v2.7.0/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../guides/crossplatform-build.mdx | 66 ++ .../guides/custom-protocol-schemes.mdx | 204 ++++ .../version-v2.7.0/guides/dynamic-assets.mdx | 136 +++ .../guides/file-association.mdx | 244 +++++ .../version-v2.7.0/guides/frameless.mdx | 87 ++ .../version-v2.7.0/guides/frontend.mdx | 72 ++ .../version-v2.7.0/guides/ides.mdx | 127 +++ .../guides/linux-distro-support.mdx | 103 +++ .../version-v2.7.0/guides/linux.mdx | 18 + .../guides/local-development.mdx | 55 ++ .../version-v2.7.0/guides/mac-appstore.mdx | 97 ++ .../version-v2.7.0/guides/manual-builds.mdx | 95 ++ .../version-v2.7.0/guides/migrating.mdx | 191 ++++ .../version-v2.7.0/guides/mouse-buttons.mdx | 25 + .../version-v2.7.0/guides/obfuscated.mdx | 40 + .../version-v2.7.0/guides/overscroll.mdx | 10 + .../version-v2.7.0/guides/routing.mdx | 47 + .../version-v2.7.0/guides/signing.mdx | 387 ++++++++ .../guides/single-instance-lock.mdx | 81 ++ .../version-v2.7.0/guides/sveltekit.mdx | 153 +++ .../version-v2.7.0/guides/templates.mdx | 97 ++ .../version-v2.7.0/guides/troubleshooting.mdx | 368 ++++++++ .../version-v2.7.0/guides/vscode.mdx | 82 ++ .../guides/windows-installer.mdx | 58 ++ .../version-v2.7.0/guides/windows.mdx | 61 ++ .../version-v2.7.0/howdoesitwork.mdx | 369 ++++++++ .../version-v2.7.0/introduction.mdx | 75 ++ .../version-v2.7.0/reference/cli.mdx | 240 +++++ .../version-v2.7.0/reference/menus.mdx | 230 +++++ .../version-v2.7.0/reference/options.mdx | 853 +++++++++++++++++ .../reference/project-config.mdx | 92 ++ .../reference/runtime/browser.mdx | 13 + .../reference/runtime/clipboard.mdx | 23 + .../reference/runtime/dialog.mdx | 302 ++++++ .../reference/runtime/events.mdx | 37 + .../reference/runtime/intro.mdx | 85 ++ .../version-v2.7.0/reference/runtime/log.mdx | 130 +++ .../version-v2.7.0/reference/runtime/menu.mdx | 25 + .../reference/runtime/screen.mdx | 38 + .../reference/runtime/window.mdx | 227 +++++ .../version-v2.7.0/tutorials/dogsapi.mdx | 245 +++++ .../version-v2.7.0/tutorials/helloworld.mdx | 123 +++ .../version-v2.7.0/community/links.mdx | 26 + .../community/showcase/bulletinboard.mdx | 10 + .../community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 12 + .../community/showcase/filehound.mdx | 16 + .../community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../community/showcase/october.mdx | 14 + .../community/showcase/optimus.mdx | 10 + .../community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 12 + .../community/showcase/riftshare.mdx | 21 + .../community/showcase/scriptbar.mdx | 10 + .../community/showcase/surge.mdx | 10 + .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 10 + .../community/showcase/warmine.mdx | 19 + .../community/showcase/wombat.mdx | 10 + .../version-v2.7.0/community/showcase/ytd.mdx | 10 + .../version-v2.7.0/community/templates.mdx | 69 ++ .../gettingstarted/building.mdx | 22 + .../gettingstarted/development.mdx | 16 + .../gettingstarted/firstproject.mdx | 130 +++ .../gettingstarted/installation.mdx | 90 ++ .../version-v2.7.0/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../guides/crossplatform-build.mdx | 66 ++ .../guides/custom-protocol-schemes.mdx | 204 ++++ .../version-v2.7.0/guides/dynamic-assets.mdx | 136 +++ .../guides/file-association.mdx | 244 +++++ .../version-v2.7.0/guides/frameless.mdx | 87 ++ .../version-v2.7.0/guides/frontend.mdx | 72 ++ .../version-v2.7.0/guides/ides.mdx | 127 +++ .../guides/linux-distro-support.mdx | 103 +++ .../version-v2.7.0/guides/linux.mdx | 18 + .../guides/local-development.mdx | 55 ++ .../version-v2.7.0/guides/mac-appstore.mdx | 97 ++ .../version-v2.7.0/guides/manual-builds.mdx | 95 ++ .../version-v2.7.0/guides/migrating.mdx | 191 ++++ .../version-v2.7.0/guides/mouse-buttons.mdx | 25 + .../version-v2.7.0/guides/obfuscated.mdx | 40 + .../version-v2.7.0/guides/overscroll.mdx | 10 + .../version-v2.7.0/guides/routing.mdx | 47 + .../version-v2.7.0/guides/signing.mdx | 387 ++++++++ .../guides/single-instance-lock.mdx | 81 ++ .../version-v2.7.0/guides/sveltekit.mdx | 153 +++ .../version-v2.7.0/guides/templates.mdx | 97 ++ .../version-v2.7.0/guides/troubleshooting.mdx | 368 ++++++++ .../version-v2.7.0/guides/vscode.mdx | 82 ++ .../guides/windows-installer.mdx | 58 ++ .../version-v2.7.0/guides/windows.mdx | 61 ++ .../version-v2.7.0/howdoesitwork.mdx | 369 ++++++++ .../version-v2.7.0/introduction.mdx | 75 ++ .../version-v2.7.0/reference/cli.mdx | 240 +++++ .../version-v2.7.0/reference/menus.mdx | 230 +++++ .../version-v2.7.0/reference/options.mdx | 853 +++++++++++++++++ .../reference/project-config.mdx | 92 ++ .../reference/runtime/browser.mdx | 13 + .../reference/runtime/clipboard.mdx | 23 + .../reference/runtime/dialog.mdx | 302 ++++++ .../reference/runtime/events.mdx | 37 + .../reference/runtime/intro.mdx | 85 ++ .../version-v2.7.0/reference/runtime/log.mdx | 130 +++ .../version-v2.7.0/reference/runtime/menu.mdx | 25 + .../reference/runtime/screen.mdx | 38 + .../reference/runtime/window.mdx | 227 +++++ .../version-v2.7.0/tutorials/dogsapi.mdx | 245 +++++ .../version-v2.7.0/tutorials/helloworld.mdx | 123 +++ .../version-v2.7.0/appendix/_category_.json | 4 + .../version-v2.7.0/community/_category_.json | 4 + .../version-v2.7.0/community/links.mdx | 26 + .../community/showcase/_category_.json | 4 + .../community/showcase/bulletinboard.mdx | 10 + .../community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 12 + .../community/showcase/filehound.mdx | 16 + .../community/showcase/hiposter.mdx | 10 + .../community/showcase/minecraftupdater.mdx | 14 + .../community/showcase/modalfilemanager.mdx | 14 + .../community/showcase/mollywallet.mdx | 10 + .../community/showcase/october.mdx | 14 + .../community/showcase/optimus.mdx | 10 + .../community/showcase/portfall.mdx | 10 + .../community/showcase/restic-browser.mdx | 12 + .../community/showcase/riftshare.mdx | 21 + .../community/showcase/scriptbar.mdx | 10 + .../community/showcase/surge.mdx | 10 + .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 10 + .../community/showcase/warmine.mdx | 19 + .../community/showcase/wombat.mdx | 10 + .../version-v2.7.0/community/showcase/ytd.mdx | 10 + .../version-v2.7.0/community/templates.mdx | 69 ++ .../contributing/developing-new-features.mdx | 34 + .../contributing/documenting.mdx | 34 + .../contributing/fixing-bugs.mdx | 29 + .../setting-up-a-dev-environment.mdx | 30 + .../contributing/ways-of-contributing.mdx | 18 + .../gettingstarted/_category_.json | 4 + .../gettingstarted/building.mdx | 22 + .../gettingstarted/development.mdx | 16 + .../gettingstarted/firstproject.mdx | 130 +++ .../gettingstarted/installation.mdx | 89 ++ .../version-v2.7.0/guides/_category_.json | 4 + .../version-v2.7.0/guides/angular.mdx | 14 + .../guides/application-development.mdx | 215 +++++ .../guides/crossplatform-build.mdx | 66 ++ .../guides/custom-protocol-schemes.mdx | 204 ++++ .../version-v2.7.0/guides/dynamic-assets.mdx | 136 +++ .../guides/file-association.mdx | 244 +++++ .../version-v2.7.0/guides/frameless.mdx | 87 ++ .../version-v2.7.0/guides/frontend.mdx | 72 ++ .../version-v2.7.0/guides/ides.mdx | 127 +++ .../guides/linux-distro-support.mdx | 103 +++ .../version-v2.7.0/guides/linux.mdx | 18 + .../guides/local-development.mdx | 55 ++ .../version-v2.7.0/guides/mac-appstore.mdx | 97 ++ .../version-v2.7.0/guides/manual-builds.mdx | 95 ++ .../version-v2.7.0/guides/migrating.mdx | 193 ++++ .../version-v2.7.0/guides/mouse-buttons.mdx | 25 + .../version-v2.7.0/guides/obfuscated.mdx | 40 + .../version-v2.7.0/guides/overscroll.mdx | 10 + .../version-v2.7.0/guides/routing.mdx | 47 + .../version-v2.7.0/guides/signing.mdx | 387 ++++++++ .../guides/single-instance-lock.mdx | 81 ++ .../version-v2.7.0/guides/sveltekit.mdx | 153 +++ .../version-v2.7.0/guides/templates.mdx | 97 ++ .../version-v2.7.0/guides/troubleshooting.mdx | 368 ++++++++ .../version-v2.7.0/guides/vscode.mdx | 82 ++ .../guides/windows-installer.mdx | 58 ++ .../version-v2.7.0/guides/windows.mdx | 61 ++ .../version-v2.7.0/howdoesitwork.mdx | 369 ++++++++ .../version-v2.7.0/introduction.mdx | 75 ++ .../version-v2.7.0/reference/_category_.json | 4 + .../version-v2.7.0/reference/cli.mdx | 240 +++++ .../version-v2.7.0/reference/menus.mdx | 230 +++++ .../version-v2.7.0/reference/options.mdx | 869 ++++++++++++++++++ .../reference/project-config.mdx | 104 +++ .../reference/runtime/_category_.json | 4 + .../reference/runtime/browser.mdx | 13 + .../reference/runtime/clipboard.mdx | 23 + .../reference/runtime/dialog.mdx | 302 ++++++ .../reference/runtime/events.mdx | 37 + .../reference/runtime/intro.mdx | 85 ++ .../version-v2.7.0/reference/runtime/log.mdx | 130 +++ .../version-v2.7.0/reference/runtime/menu.mdx | 25 + .../reference/runtime/screen.mdx | 38 + .../reference/runtime/window.mdx | 227 +++++ .../version-v2.7.0/tutorials/_category_.json | 4 + .../version-v2.7.0/tutorials/dogsapi.mdx | 245 +++++ .../version-v2.7.0/tutorials/helloworld.mdx | 123 +++ website/src/pages/changelog.mdx | 2 + .../version-v2.5.0/guides/troubleshooting.mdx | 180 ---- .../version-v2.7.0/appendix/_category_.json | 4 + .../version-v2.7.0/community/_category_.json | 4 + .../community/links.mdx | 0 .../community/showcase/_category_.json | 4 + .../community/showcase/bulletinboard.mdx | 10 + .../community/showcase/emailit.mdx | 10 + .../community/showcase/encrypteasy.mdx | 0 .../community/showcase/filehound.mdx | 0 .../community/showcase/hiposter.mdx | 0 .../community/showcase/minecraftupdater.mdx | 0 .../community/showcase/modalfilemanager.mdx | 0 .../community/showcase/mollywallet.mdx | 0 .../community/showcase/october.mdx | 14 + .../community/showcase/optimus.mdx | 0 .../community/showcase/portfall.mdx | 0 .../community/showcase/restic-browser.mdx | 12 + .../community/showcase/riftshare.mdx | 21 + .../community/showcase/scriptbar.mdx | 10 + .../community/showcase/snippetexpander.mdx | 27 + .../community/showcase/surge.mdx | 10 + .../community/showcase/tinyrdm.mdx | 10 + .../community/showcase/wally.mdx | 10 + .../community/showcase/warmine.mdx | 0 .../community/showcase/wombat.mdx | 0 .../version-v2.7.0/community/showcase/ytd.mdx | 10 + .../community/templates.mdx | 15 +- .../gettingstarted/_category_.json | 4 + .../gettingstarted/building.mdx | 0 .../gettingstarted/development.mdx | 0 .../gettingstarted/firstproject.mdx | 0 .../gettingstarted/installation.mdx | 0 .../version-v2.7.0/guides/_category_.json | 4 + .../guides/angular.mdx | 0 .../guides/application-development.mdx | 59 ++ .../guides/crossplatform-build.mdx | 65 ++ .../guides/custom-protocol-schemes.mdx | 189 ++++ .../guides/dynamic-assets.mdx | 0 .../guides/file-association.mdx | 228 +++++ .../guides/frameless.mdx | 0 .../guides/frontend.mdx | 0 .../guides/ides.mdx | 0 .../guides/linux-distro-support.mdx | 0 .../guides/linux.mdx | 0 .../guides/local-development.mdx | 0 .../guides/mac-appstore.mdx | 0 .../guides/manual-builds.mdx | 0 .../guides/migrating.mdx | 0 .../guides/mouse-buttons.mdx | 0 .../guides/obfuscated.mdx | 0 .../guides/overscroll.mdx | 0 .../version-v2.7.0/guides/routing.mdx | 47 + .../guides/signing.mdx | 27 +- .../guides/single-instance-lock.mdx | 80 ++ .../version-v2.7.0/guides/sveltekit.mdx | 136 +++ .../guides/templates.mdx | 0 .../version-v2.7.0/guides/troubleshooting.mdx | 385 ++++++++ .../guides/vscode.mdx | 0 .../guides/windows-installer.mdx | 0 .../guides/windows.mdx | 0 .../howdoesitwork.mdx | 51 + .../introduction.mdx | 2 +- .../version-v2.7.0/reference/_category_.json | 4 + .../reference/cli.mdx | 90 +- .../reference/menus.mdx | 0 .../reference/options.mdx | 148 ++- .../reference/project-config.mdx | 130 +++ .../reference/runtime/_category_.json | 4 + .../reference/runtime/browser.mdx | 0 .../reference/runtime/clipboard.mdx | 0 .../reference/runtime/dialog.mdx | 0 .../reference/runtime/events.mdx | 0 .../reference/runtime/intro.mdx | 0 .../reference/runtime/log.mdx | 0 .../reference/runtime/menu.mdx | 0 .../reference/runtime/screen.mdx | 0 .../reference/runtime/window.mdx | 15 +- .../version-v2.7.0/tutorials/_category_.json | 4 + .../tutorials/dogsapi.mdx | 0 .../tutorials/helloworld.mdx | 1 + ...bars.json => version-v2.7.0-sidebars.json} | 0 website/versions.json | 2 +- 526 files changed, 42790 insertions(+), 248 deletions(-) create mode 100644 website/i18n/en/docusaurus-plugin-content-docs/version-v2.7.0.json create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0}/guides/routing.mdx (100%) create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx create mode 100644 website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/appendix/_category_.json (100%) rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/community/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/gettingstarted/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/guides/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/reference/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/reference/runtime/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0}/tutorials/_category_.json (100%) create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx create mode 100644 website/i18n/ja/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/bulletinboard.mdx (100%) rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/emailit.mdx (100%) create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/october.mdx (100%) create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/restic-browser.mdx (100%) rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/riftshare.mdx (100%) rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/scriptbar.mdx (100%) rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/surge.mdx (100%) create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/wally.mdx (100%) create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/community/showcase/ytd.mdx (100%) create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx rename website/{versioned_docs/version-v2.5.0 => i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0}/reference/project-config.mdx (99%) create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx create mode 100644 website/i18n/ko/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx create mode 100644 website/i18n/pt/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx create mode 100644 website/i18n/ru/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/appendix/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/emailit.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/encrypteasy.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/filehound.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/hiposter.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/minecraftupdater.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/modalfilemanager.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/mollywallet.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/october.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/optimus.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/portfall.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/restic-browser.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/riftshare.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/scriptbar.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/surge.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/tinyrdm.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wally.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/warmine.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/wombat.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/ytd.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/community/templates.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/developing-new-features.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/documenting.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/fixing-bugs.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/setting-up-a-dev-environment.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/contributing/ways-of-contributing.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/building.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/development.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/firstproject.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/gettingstarted/installation.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/angular.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/application-development.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/custom-protocol-schemes.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/dynamic-assets.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/file-association.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frameless.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/frontend.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/ides.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux-distro-support.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/linux.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/local-development.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mac-appstore.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/manual-builds.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/migrating.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/mouse-buttons.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/obfuscated.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/overscroll.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/routing.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/signing.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/sveltekit.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/templates.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/troubleshooting.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/vscode.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows-installer.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/guides/windows.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/howdoesitwork.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/introduction.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/cli.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/menus.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/options.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/project-config.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/browser.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/clipboard.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/dialog.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/events.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/intro.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/log.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/menu.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/screen.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/reference/runtime/window.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/_category_.json create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/dogsapi.mdx create mode 100644 website/i18n/zh-Hans/docusaurus-plugin-content-docs/version-v2.7.0/tutorials/helloworld.mdx delete mode 100644 website/versioned_docs/version-v2.5.0/guides/troubleshooting.mdx create mode 100644 website/versioned_docs/version-v2.7.0/appendix/_category_.json create mode 100644 website/versioned_docs/version-v2.7.0/community/_category_.json rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/links.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/_category_.json create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/bulletinboard.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/emailit.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/encrypteasy.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/filehound.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/hiposter.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/minecraftupdater.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/modalfilemanager.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/mollywallet.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/october.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/optimus.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/portfall.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/restic-browser.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/riftshare.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/scriptbar.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/snippetexpander.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/surge.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/tinyrdm.mdx create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/wally.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/warmine.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/showcase/wombat.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/community/showcase/ytd.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/community/templates.mdx (84%) create mode 100644 website/versioned_docs/version-v2.7.0/gettingstarted/_category_.json rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/gettingstarted/building.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/gettingstarted/development.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/gettingstarted/firstproject.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/gettingstarted/installation.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/guides/_category_.json rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/angular.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/application-development.mdx (87%) create mode 100644 website/versioned_docs/version-v2.7.0/guides/crossplatform-build.mdx create mode 100644 website/versioned_docs/version-v2.7.0/guides/custom-protocol-schemes.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/dynamic-assets.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/guides/file-association.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/frameless.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/frontend.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/ides.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/linux-distro-support.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/linux.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/local-development.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/mac-appstore.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/manual-builds.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/migrating.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/mouse-buttons.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/obfuscated.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/overscroll.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/guides/routing.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/signing.mdx (94%) create mode 100644 website/versioned_docs/version-v2.7.0/guides/single-instance-lock.mdx create mode 100644 website/versioned_docs/version-v2.7.0/guides/sveltekit.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/templates.mdx (100%) create mode 100644 website/versioned_docs/version-v2.7.0/guides/troubleshooting.mdx rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/vscode.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/windows-installer.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/guides/windows.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/howdoesitwork.mdx (93%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/introduction.mdx (97%) create mode 100644 website/versioned_docs/version-v2.7.0/reference/_category_.json rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/cli.mdx (71%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/menus.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/options.mdx (82%) create mode 100644 website/versioned_docs/version-v2.7.0/reference/project-config.mdx create mode 100644 website/versioned_docs/version-v2.7.0/reference/runtime/_category_.json rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/browser.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/clipboard.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/dialog.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/events.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/intro.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/log.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/menu.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/screen.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/reference/runtime/window.mdx (94%) create mode 100644 website/versioned_docs/version-v2.7.0/tutorials/_category_.json rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/tutorials/dogsapi.mdx (100%) rename website/versioned_docs/{version-v2.5.0 => version-v2.7.0}/tutorials/helloworld.mdx (99%) rename website/versioned_sidebars/{version-v2.5.0-sidebars.json => version-v2.7.0-sidebars.json} (100%) diff --git a/v2/cmd/wails/internal/version.txt b/v2/cmd/wails/internal/version.txt index 7433fb30ccc..5c9dc9d02a1 100644 --- a/v2/cmd/wails/internal/version.txt +++ b/v2/cmd/wails/internal/version.txt @@ -1 +1 @@ -v2.6.0 \ No newline at end of file +v2.7.0 \ No newline at end of file diff --git a/website/i18n/en/docusaurus-plugin-content-docs/version-v2.7.0.json b/website/i18n/en/docusaurus-plugin-content-docs/version-v2.7.0.json new file mode 100644 index 00000000000..12dbd6c8378 --- /dev/null +++ b/website/i18n/en/docusaurus-plugin-content-docs/version-v2.7.0.json @@ -0,0 +1,38 @@ +{ + "version.label": { + "message": "v2.7.0", + "description": "The label for version v2.7.0" + }, + "sidebar.docs.category.Getting Started": { + "message": "Getting Started", + "description": "The label for category Getting Started in sidebar docs" + }, + "sidebar.docs.category.Reference": { + "message": "Reference", + "description": "The label for category Reference in sidebar docs" + }, + "sidebar.docs.category.Runtime": { + "message": "Runtime", + "description": "The label for category Runtime in sidebar docs" + }, + "sidebar.docs.category.Community": { + "message": "Community", + "description": "The label for category Community in sidebar docs" + }, + "sidebar.docs.category.Showcase": { + "message": "Showcase", + "description": "The label for category Showcase in sidebar docs" + }, + "sidebar.docs.category.Guides": { + "message": "Guides", + "description": "The label for category Guides in sidebar docs" + }, + "sidebar.docs.category.Tutorials": { + "message": "Tutorials", + "description": "The label for category Tutorials in sidebar docs" + }, + "sidebar.docs.link.Contributing": { + "message": "Contributing", + "description": "The label for link Contributing in sidebar docs, linking to /community-guide#ways-of-contributing" + } +} diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx new file mode 100644 index 00000000000..d6b742435e4 --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/links.mdx @@ -0,0 +1,26 @@ +--- +sidebar_position: 2 +--- + +# Liens + +Cette page sert de liste pour les liens liés à la communauté. Veuillez soumettre une PR (cliquez sur `Modifier cette page` en bas) pour soumettre des liens. + +## Awesome Wails + +La [liste définitive](https://github.com/wailsapp/awesome-wails) de liens relatifs à Wails. + +## Canaux de support + +- [Serveur Discord Wails](https://discord.gg/JDdSxwjhGf) +- [Github Issues](https://github.com/wailsapp/wails/issues) +- [canal de discussion sur la bêta v2](https://github.com/wailsapp/wails/discussions/828) + +## Réseaux sociaux + +- [Twitter](https://twitter.com/wailsapp) +- [Groupe QQ pour la communauté chinoise de Wails](https://qm.qq.com/cgi-bin/qm/qr?k=PmIURne5hFGNd7QWzW5qd6FV-INEjNJv&jump_from=webapi) - Numéro de groupe : 1067173054 + +## Autres tutoriels et articles + +- [Construction d'un Panneau d'Affichage](https://blog.customct.com/building-bulletin-board) diff --git a/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx new file mode 100644 index 00000000000..2c56a2e836c --- /dev/null +++ b/website/i18n/fr/docusaurus-plugin-content-docs/version-v2.7.0/community/showcase/bulletinboard.mdx @@ -0,0 +1,10 @@ +# BulletinBoard + +```mdx-code-block +