mirror of
https://github.com/wailsapp/wails.git
synced 2026-03-15 15:15:51 +01:00
This commit adds comprehensive Android support for Wails v3, enabling Go applications to run as native Android apps with WebView-based UI. Key features: - Android-specific application implementation with JNI bridge - WebView integration via WebViewAssetLoader for serving assets - JavaScript runtime injection and execution via JNI callbacks - Binding call support with async result callbacks - Event system support for Android platform - Full example Android app with Gradle build system Technical details: - Uses CGO with Android NDK for cross-compilation - Implements JNI callbacks for Go <-> Java communication - Supports both ARM64 and x86_64 architectures - WebView debugging support via Chrome DevTools Protocol - Handles empty response body case in binding calls to prevent panic Files added: - v3/pkg/application/*_android.go - Android platform implementations - v3/pkg/events/events_android.go - Android event definitions - v3/internal/*/\*_android.go - Android-specific internal packages - v3/examples/android/ - Complete example Android application - v3/ANDROID_ARCHITECTURE.md - Architecture documentation 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
118 lines
2.3 KiB
Go
118 lines
2.3 KiB
Go
//go:build linux && !android
|
|
|
|
package application
|
|
|
|
type linuxMenu struct {
|
|
menu *Menu
|
|
native pointer
|
|
}
|
|
|
|
func newMenuImpl(menu *Menu) *linuxMenu {
|
|
result := &linuxMenu{
|
|
menu: menu,
|
|
native: menuBarNew(),
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (m *linuxMenu) run() {
|
|
m.update()
|
|
}
|
|
|
|
func (m *linuxMenu) update() {
|
|
m.processMenu(m.menu)
|
|
}
|
|
|
|
func (m *linuxMenu) processMenu(menu *Menu) {
|
|
if menu.impl == nil {
|
|
menu.impl = &linuxMenu{
|
|
menu: menu,
|
|
native: menuNew(),
|
|
}
|
|
}
|
|
var currentRadioGroup GSListPointer
|
|
|
|
for _, item := range menu.items {
|
|
// drop the group if we have run out of radio items
|
|
if item.itemType != radio {
|
|
currentRadioGroup = nilRadioGroup
|
|
}
|
|
|
|
switch item.itemType {
|
|
case submenu:
|
|
menuItem := newMenuItemImpl(item)
|
|
item.impl = menuItem
|
|
m.processMenu(item.submenu)
|
|
m.addSubMenuToItem(item.submenu, item)
|
|
m.addMenuItem(menu, item)
|
|
case text, checkbox:
|
|
menuItem := newMenuItemImpl(item)
|
|
item.impl = menuItem
|
|
m.addMenuItem(menu, item)
|
|
case radio:
|
|
menuItem := newRadioItemImpl(item, currentRadioGroup)
|
|
item.impl = menuItem
|
|
m.addMenuItem(menu, item)
|
|
currentRadioGroup = menuGetRadioGroup(menuItem)
|
|
case separator:
|
|
m.addMenuSeparator(menu)
|
|
}
|
|
|
|
}
|
|
|
|
for _, item := range menu.items {
|
|
if item.callback != nil {
|
|
m.attachHandler(item)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func (m *linuxMenu) attachHandler(item *MenuItem) {
|
|
(item.impl).(*linuxMenuItem).handlerId = attachMenuHandler(item)
|
|
}
|
|
|
|
func (m *linuxMenu) addSubMenuToItem(menu *Menu, item *MenuItem) {
|
|
if menu.impl == nil {
|
|
menu.impl = &linuxMenu{
|
|
menu: menu,
|
|
native: menuNew(),
|
|
}
|
|
}
|
|
menuSetSubmenu(item, menu)
|
|
}
|
|
|
|
func (m *linuxMenu) addMenuItem(parent *Menu, menu *MenuItem) {
|
|
menuAppend(parent, menu)
|
|
}
|
|
|
|
func (m *linuxMenu) addMenuSeparator(menu *Menu) {
|
|
menuAddSeparator(menu)
|
|
|
|
}
|
|
|
|
func (m *linuxMenu) addServicesMenu(menu *Menu) {
|
|
// FIXME: Should this be required?
|
|
}
|
|
|
|
func (l *linuxMenu) createMenu(name string, items []*MenuItem) *Menu {
|
|
impl := newMenuImpl(&Menu{label: name})
|
|
menu := &Menu{
|
|
label: name,
|
|
items: items,
|
|
impl: impl,
|
|
}
|
|
impl.menu = menu
|
|
return menu
|
|
}
|
|
|
|
func DefaultApplicationMenu() *Menu {
|
|
menu := NewMenu()
|
|
menu.AddRole(AppMenu)
|
|
menu.AddRole(FileMenu)
|
|
menu.AddRole(EditMenu)
|
|
menu.AddRole(ViewMenu)
|
|
menu.AddRole(WindowMenu)
|
|
menu.AddRole(HelpMenu)
|
|
return menu
|
|
}
|