Skip to content

Latest commit

 

History

History
227 lines (168 loc) · 9.33 KB

README.md

File metadata and controls

227 lines (168 loc) · 9.33 KB

🛠️ Compose Native Tray Library

logo

Compose Native Tray is a Kotlin library that provides a simple way to create system tray applications with native support for Linux, Windows, and macOS. This library was created to address several issues with the Compose for Desktop tray, including poor HDPI support on Windows and Linux, as well as the outdated appearance of the tray on Linux, which resembled Windows 95. In addition to these fixes, ComposeTray also adds support for checkable items, dividers, submenus, and even nested submenus, offering a more feature-rich and modern solution. The Linux implementation uses GTK, while macOS uses AWT, and the Windows implementation is based on native system calls. Additionally, it allows you to enable or disable individual tray items dynamically. This library allows you to add a system tray icon, tooltip, and menu with various options in a Kotlin DSL-style syntax.

Warning: This library is highly experimental, and its APIs are subject to change.

✨ Features

  • Cross-platform support for Linux, Windows, and macOS.
  • DSL-style syntax to define tray menus with ease.
  • Supports standard items, submenus, dividers, and checkable items.
  • Ability to enable/disable menu items dynamically.
  • Corrects issues with the Compose for Desktop tray, particularly HDPI support on Windows and Linux.
  • Improves the appearance of the tray on Linux, which previously resembled Windows 95.
  • Adds support for checkable items, dividers, and submenus, including nested submenus.
  • Supports primary action for Windows, macOS, and Linux.
    • On Windows and macOS, the primary action is triggered by a left-click on the tray icon.
    • On Linux, due to the limitations of libappindicator, the primary action creates an item at the top of the context menu (with a customizable label). If the context menu is empty, the library uses gtkstatusicon to capture the primary action without needing to add an item to the context menu.
  • Single Instance Management: Ensures that only one instance of the application can run at a time and allows restoring focus to the running instance when another instance is attempted.
  • Tray Position Detection: Allows determining the position of the system tray, which helps in positioning related windows appropriately.
  • Compose Recomposition Support: The tray supports Compose recomposition, making it possible to dynamically show or hide the tray icon, for example:

demo

var isWindowVisible by remember { mutableStateOf(true) }

if (!isWindowVisible) {
    Tray(
        // Tray parameters
    )
}

🚀 Getting Started

📦 Installation

To use the ComposeTray library, add it as a dependency in your build.gradle.kts file:

dependencies {
  implementation("io.github.kdroidfilter:composenativetray:0.5.2")
}

🛠️ Usage

Here's a basic example of how to use the ComposeTray library to create a system tray icon with a menu: The Tray must be executed inside an Application Scope, like this:

application {
  val iconPath = Paths.get("src/test/resources/icon.png").toAbsolutePath().toString()
  val windowsIconPath = Paths.get("src/test/resources/icon.ico").toAbsolutePath().toString()
  Tray(
    iconPath = iconPath,
    windowsIconPath = windowsIconPath,
    tooltip = "My Application",
    primaryAction = {
      Log.i(logTag, "Primary action triggered")
    },
    primaryActionLinuxLabel = "Open Application"
  ) {
    SubMenu(label = "Options") {
      Item(label = "Setting 1") {
        Log.i(logTag, "Setting 1 selected")
      }
      SubMenu(label = "Advanced Sub-options") {
        Item(label = "Advanced Option 1") {
          Log.i(logTag, "Advanced Option 1 selected")
        }
        Item(label = "Advanced Option 2") {
          Log.i(logTag, "Advanced Option 2 selected")
        }
      }
    }

    Divider()

    SubMenu(label = "Tools") {
      Item(label = "Calculator") {
        Log.i(logTag, "Calculator launched")
      }
      Item(label = "Notepad") {
        Log.i(logTag, "Notepad opened")
      }
    }

    Divider()

    CheckableItem(label = "Enable notifications") { isChecked ->
      Log.i(logTag, "Notifications ${if (isChecked) "enabled" else "disabled"}")
    }

    Divider()

    Item(label = "About") {
      Log.i(logTag, "Application v1.0 - Developed by Elyahou")
    }

    Divider()

    Item(label = "Exit", isEnabled = true) {
      Log.i(logTag, "Exiting the application")
      dispose()
      exitProcess(0)
    }

    Item(label = "Version 1.0.0", isEnabled = false)
  }
}

📋 Components of the Tray Menu

  • Item: A standard clickable item that can be enabled or disabled.
  • CheckableItem: A menu item with a checkbox that can be toggled on or off.
  • SubMenu: A submenu that can contain multiple items, including other submenus.
  • Divider: A separator used to visually separate menu items.
  • dispose: Call to remove the system tray icon and exit gracefully.
  • Primary Action: An action triggered by a left-click on the tray icon (Windows and macOS) or as a top item in the context menu (Linux).

🖥️ Platform-Specific Implementation

The NativeTray class automatically detects the operating system and initializes the appropriate tray implementation:

  • Linux: Uses LinuxTrayInitializer, which is implemented using GTK via JNA.
  • Windows: Uses WindowsTrayInitializer, which is based on the tray library by dmikushin.
  • macOS: Uses AwtTrayInitializer, which is implemented using AWT.

Note: For Windows, you must use an .ico file for the tray icon, while for Linux and macOS, a .png file is required.

🗂️ Windows DLL Generation

To generate the Windows DLL, execute the following command:

gcc -shared -o ../resources/win32-x86-64/tray.dll tray_windows.c

📜 Logging

This example uses the kmplog library for logging, which allows you to log messages when certain items are selected or toggled.

🔄 Single Instance Management

The SingleInstanceManager ensures that only one instance of the application is running at a time. When a second instance tries to start, it sends a restore request to the existing instance to bring it to the foreground.

Usage

To use the SingleInstanceManager, include it as follows:

var isWindowVisible by remember { mutableStateOf(true) }

val isSingleInstance = SingleInstanceManager.isSingleInstance(onRestoreRequest = {
  isWindowVisible = true
})

if (!isSingleInstance) {
  exitApplication()
  return@application
}

In this example, the SingleInstanceManager will check if an instance is already running. If not, it will acquire the lock and start watching for restore requests. If an instance is already running, it will send a restore request to bring the existing window to the foreground, allowing you to focus on the already-running application rather than starting a new instance.

📌 Tray Position Detection

The getTrayPosition() function allows you to determine the current position of the system tray on the screen. This information can be useful for aligning application windows relative to the tray icon.

Note: Currently, on macOS, the getTrayPosition() function always returns TOP_RIGHT, and the tray position is not dynamically detected. On Windows and Linux, the position is determined dynamically.

Additionally, the getTrayWindowPosition(windowWidth: Int, windowHeight: Int): WindowPosition function can be used to determine the appropriate position for a window of specific dimensions, taking into account the location of the tray.

Usage

To use the tray position functions:

val windowWidth = 800
val windowHeight = 600
val windowPosition = getTrayWindowPosition(windowWidth, windowHeight)

Window(
  state = rememberWindowState(
    width = windowWidth.dp,
    height = windowHeight.dp,
    position = windowPosition
  ),
  ...
)

In this example, getTrayWindowPosition is used to determine the ideal position for the window based on the system tray's location. This helps in creating a seamless and user-friendly experience when displaying windows near the tray icon.

📸 Screenshots

Here are some screenshots of ComposeTray running on different platforms:

Windows

Windows

Ubuntu Gnome

Gnome

Ubuntu KDE

KDE

📄 License

This library is licensed under the MIT License.

🤝 Contributing

Feel free to open issues or pull requests if you find any bugs or have suggestions for new features. If you have other ideas for improvements or new functionalities, please let me know! If you use this library in your project, I would be happy to link to those projects as well.

✅ Things to Implement

  • ✅ Implement a system to check if an instance of the application is already running
  • ✅ Add the ability to locate the position of the systray (top left, top right, etc.)
  • Add the ability to dynamically change the tray icon

🙏 Acknowledgements

This library is developed and maintained by Elie G, aiming to provide an easy and cross-platform system tray solution for Kotlin applications.