mirror of
https://github.com/feschber/lan-mouse.git
synced 2026-04-30 17:33:20 +03:00
Compare commits
6 Commits
macos-inpu
...
64d5058544
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
64d5058544 | ||
|
|
61259445c0 | ||
|
|
b9a4497fa4 | ||
|
|
2f824d8bd3 | ||
|
|
4397ce9f1c | ||
|
|
acb067bfde |
@@ -813,7 +813,7 @@ impl Dispatch<WlPointer, ()> for State {
|
|||||||
})),
|
})),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
wl_pointer::Event::Frame => {
|
wl_pointer::Event::Frame {} => {
|
||||||
// TODO properly handle frame events
|
// TODO properly handle frame events
|
||||||
// we simply insert a frame event on the client side
|
// we simply insert a frame event on the client side
|
||||||
// after each event for now
|
// after each event for now
|
||||||
|
|||||||
@@ -390,9 +390,9 @@ fn create_event_tap<'a>(
|
|||||||
|
|
||||||
if let Some(pos) = pos {
|
if let Some(pos) = pos {
|
||||||
res_events.iter().for_each(|e| {
|
res_events.iter().for_each(|e| {
|
||||||
// error must be ignored, since the event channel
|
event_tx
|
||||||
// may already be closed when the InputCapture instance is dropped.
|
.blocking_send((pos, *e))
|
||||||
let _ = event_tx.blocking_send((pos, *e));
|
.expect("Failed to send event");
|
||||||
});
|
});
|
||||||
// Returning None should stop the event from being processed
|
// Returning None should stop the event from being processed
|
||||||
// but core fundation still returns the event
|
// but core fundation still returns the event
|
||||||
|
|||||||
@@ -22,9 +22,9 @@ use windows::Win32::UI::WindowsAndMessaging::{
|
|||||||
RegisterClassW, SetWindowsHookExW, TranslateMessage, EDD_GET_DEVICE_INTERFACE_NAME, HHOOK,
|
RegisterClassW, SetWindowsHookExW, TranslateMessage, EDD_GET_DEVICE_INTERFACE_NAME, HHOOK,
|
||||||
HMENU, HOOKPROC, KBDLLHOOKSTRUCT, LLKHF_EXTENDED, MSG, MSLLHOOKSTRUCT, WH_KEYBOARD_LL,
|
HMENU, HOOKPROC, KBDLLHOOKSTRUCT, LLKHF_EXTENDED, MSG, MSLLHOOKSTRUCT, WH_KEYBOARD_LL,
|
||||||
WH_MOUSE_LL, WINDOW_STYLE, WM_DISPLAYCHANGE, WM_KEYDOWN, WM_KEYUP, WM_LBUTTONDOWN,
|
WH_MOUSE_LL, WINDOW_STYLE, WM_DISPLAYCHANGE, WM_KEYDOWN, WM_KEYUP, WM_LBUTTONDOWN,
|
||||||
WM_LBUTTONUP, WM_MBUTTONDOWN, WM_MBUTTONUP, WM_MOUSEHWHEEL, WM_MOUSEMOVE, WM_MOUSEWHEEL,
|
WM_LBUTTONUP, WM_MBUTTONDOWN, WM_MBUTTONUP, WM_MOUSEMOVE, WM_MOUSEWHEEL, WM_RBUTTONDOWN,
|
||||||
WM_RBUTTONDOWN, WM_RBUTTONUP, WM_SYSKEYDOWN, WM_SYSKEYUP, WM_USER, WM_XBUTTONDOWN,
|
WM_RBUTTONUP, WM_SYSKEYDOWN, WM_SYSKEYUP, WM_USER, WM_XBUTTONDOWN, WM_XBUTTONUP, WNDCLASSW,
|
||||||
WM_XBUTTONUP, WNDCLASSW, WNDPROC,
|
WNDPROC,
|
||||||
};
|
};
|
||||||
|
|
||||||
use input_event::{
|
use input_event::{
|
||||||
@@ -537,10 +537,6 @@ fn to_mouse_event(wparam: WPARAM, lparam: LPARAM) -> Option<PointerEvent> {
|
|||||||
state: if p == WM_XBUTTONDOWN as usize { 1 } else { 0 },
|
state: if p == WM_XBUTTONDOWN as usize { 1 } else { 0 },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
WPARAM(p) if p == WM_MOUSEHWHEEL as usize => Some(PointerEvent::AxisDiscrete120 {
|
|
||||||
axis: 1, // Horizontal
|
|
||||||
value: mouse_low_level.mouseData as i32 >> 16,
|
|
||||||
}),
|
|
||||||
w => {
|
w => {
|
||||||
log::warn!("unknown mouse event: {w:?}");
|
log::warn!("unknown mouse event: {w:?}");
|
||||||
None
|
None
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ impl X11Emulation {
|
|||||||
pub(crate) fn new() -> Result<Self, X11EmulationCreationError> {
|
pub(crate) fn new() -> Result<Self, X11EmulationCreationError> {
|
||||||
let display = unsafe {
|
let display = unsafe {
|
||||||
match xlib::XOpenDisplay(ptr::null()) {
|
match xlib::XOpenDisplay(ptr::null()) {
|
||||||
d if std::ptr::eq(d, ptr::null_mut::<xlib::Display>()) => {
|
d if d == ptr::null::<xlib::Display>() as *mut xlib::Display => {
|
||||||
Err(X11EmulationCreationError::OpenDisplay)
|
Err(X11EmulationCreationError::OpenDisplay)
|
||||||
}
|
}
|
||||||
display => Ok(display),
|
display => Ok(display),
|
||||||
|
|||||||
@@ -143,6 +143,7 @@ impl Emulation for DesktopPortalEmulation<'_> {
|
|||||||
|
|
||||||
impl AsyncDrop for DesktopPortalEmulation<'_> {
|
impl AsyncDrop for DesktopPortalEmulation<'_> {
|
||||||
#[doc = r" Perform the async cleanup."]
|
#[doc = r" Perform the async cleanup."]
|
||||||
|
#[must_use]
|
||||||
#[allow(clippy::type_complexity, clippy::type_repetition_in_bounds)]
|
#[allow(clippy::type_complexity, clippy::type_repetition_in_bounds)]
|
||||||
fn async_drop<'async_trait>(
|
fn async_drop<'async_trait>(
|
||||||
self,
|
self,
|
||||||
|
|||||||
@@ -1,102 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<interface>
|
|
||||||
<requires lib="gtk" version="4.0"/>
|
|
||||||
<requires lib="libadwaita" version="1.0"/>
|
|
||||||
<template class="AuthorizationWindow" parent="AdwWindow">
|
|
||||||
<property name="modal">True</property>
|
|
||||||
<property name="width-request">180</property>
|
|
||||||
<property name="default-width">180</property>
|
|
||||||
<property name="height-request">180</property>
|
|
||||||
<property name="default-height">180</property>
|
|
||||||
<property name="title" translatable="yes">Unauthorized Device</property>
|
|
||||||
<property name="content">
|
|
||||||
<object class="GtkBox">
|
|
||||||
<property name="orientation">vertical</property>
|
|
||||||
<property name="vexpand">True</property>
|
|
||||||
<child type="top">
|
|
||||||
<object class="AdwHeaderBar">
|
|
||||||
<style>
|
|
||||||
<class name="flat"/>
|
|
||||||
</style>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkBox">
|
|
||||||
<property name="orientation">vertical</property>
|
|
||||||
<property name="spacing">30</property>
|
|
||||||
<property name="margin-start">30</property>
|
|
||||||
<property name="margin-end">30</property>
|
|
||||||
<property name="margin-top">30</property>
|
|
||||||
<property name="margin-bottom">30</property>
|
|
||||||
<child>
|
|
||||||
<object class="GtkLabel">
|
|
||||||
<property name="label">An unauthorized Device is trying to connect. Do you want to authorize this Device?</property>
|
|
||||||
<property name="width-request">100</property>
|
|
||||||
<property name="wrap">word-wrap</property>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="AdwPreferencesGroup">
|
|
||||||
<property name="title">sha256 fingerprint</property>
|
|
||||||
<child>
|
|
||||||
<object class="AdwActionRow">
|
|
||||||
<property name="child">
|
|
||||||
<object class="GtkLabel" id="fingerprint">
|
|
||||||
<property name="ellipsize">PANGO_ELLIPSIZE_NONE</property>
|
|
||||||
<property name="vexpand">True</property>
|
|
||||||
<property name="hexpand">False</property>
|
|
||||||
<property name="wrap">True</property>
|
|
||||||
<property name="wrap-mode">word-char</property>
|
|
||||||
<property name="justify">center</property>
|
|
||||||
<property name="xalign">0.5</property>
|
|
||||||
<property name="margin-top">10</property>
|
|
||||||
<property name="margin-bottom">10</property>
|
|
||||||
<property name="margin-start">10</property>
|
|
||||||
<property name="margin-end">10</property>
|
|
||||||
<property name="width-chars">64</property>
|
|
||||||
</object>
|
|
||||||
</property>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkBox">
|
|
||||||
<property name="margin-start">30</property>
|
|
||||||
<property name="margin-end">30</property>
|
|
||||||
<property name="margin-top">30</property>
|
|
||||||
<property name="margin-bottom">30</property>
|
|
||||||
<property name="orientation">horizontal</property>
|
|
||||||
<property name="spacing">30</property>
|
|
||||||
<property name="hexpand">True</property>
|
|
||||||
<property name="vexpand">True</property>
|
|
||||||
<property name="valign">end</property>
|
|
||||||
<child>
|
|
||||||
<object class="GtkButton" id="cancel_button">
|
|
||||||
<signal name="clicked" handler="handle_cancel" swapped="true"/>
|
|
||||||
<property name="label" translatable="yes">Cancel</property>
|
|
||||||
<property name="can-shrink">True</property>
|
|
||||||
<property name="height-request">50</property>
|
|
||||||
<property name="hexpand">True</property>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkButton" id="confirm_button">
|
|
||||||
<signal name="clicked" handler="handle_confirm" swapped="true"/>
|
|
||||||
<property name="label" translatable="yes">Authorize</property>
|
|
||||||
<property name="can-shrink">True</property>
|
|
||||||
<property name="height-request">50</property>
|
|
||||||
<property name="hexpand">True</property>
|
|
||||||
<style>
|
|
||||||
<class name="destructive-action"/>
|
|
||||||
</style>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
</object>
|
|
||||||
</property>
|
|
||||||
</template>
|
|
||||||
</interface>
|
|
||||||
@@ -2,7 +2,6 @@
|
|||||||
<gresources>
|
<gresources>
|
||||||
<gresource prefix="/de/feschber/LanMouse">
|
<gresource prefix="/de/feschber/LanMouse">
|
||||||
<file compressed="true" preprocess="xml-stripblanks">window.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">window.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">authorization_window.ui</file>
|
|
||||||
<file compressed="true" preprocess="xml-stripblanks">fingerprint_window.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">fingerprint_window.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">client_row.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">client_row.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">key_row.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">key_row.ui</file>
|
||||||
|
|||||||
@@ -1,19 +0,0 @@
|
|||||||
mod imp;
|
|
||||||
|
|
||||||
use glib::Object;
|
|
||||||
use gtk::{gio, glib, subclass::prelude::ObjectSubclassIsExt};
|
|
||||||
|
|
||||||
glib::wrapper! {
|
|
||||||
pub struct AuthorizationWindow(ObjectSubclass<imp::AuthorizationWindow>)
|
|
||||||
@extends adw::Window, gtk::Window, gtk::Widget,
|
|
||||||
@implements gio::ActionGroup, gio::ActionMap, gtk::Accessible, gtk::Buildable,
|
|
||||||
gtk::ConstraintTarget, gtk::Native, gtk::Root, gtk::ShortcutManager;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AuthorizationWindow {
|
|
||||||
pub(crate) fn new(fingerprint: &str) -> Self {
|
|
||||||
let window: Self = Object::builder().build();
|
|
||||||
window.imp().set_fingerprint(fingerprint);
|
|
||||||
window
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,75 +0,0 @@
|
|||||||
use std::sync::OnceLock;
|
|
||||||
|
|
||||||
use adw::prelude::*;
|
|
||||||
use adw::subclass::prelude::*;
|
|
||||||
use glib::subclass::InitializingObject;
|
|
||||||
use gtk::{
|
|
||||||
glib::{self, subclass::Signal},
|
|
||||||
template_callbacks, Button, CompositeTemplate, Label,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[derive(CompositeTemplate, Default)]
|
|
||||||
#[template(resource = "/de/feschber/LanMouse/authorization_window.ui")]
|
|
||||||
pub struct AuthorizationWindow {
|
|
||||||
#[template_child]
|
|
||||||
pub fingerprint: TemplateChild<Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub cancel_button: TemplateChild<Button>,
|
|
||||||
#[template_child]
|
|
||||||
pub confirm_button: TemplateChild<Button>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[glib::object_subclass]
|
|
||||||
impl ObjectSubclass for AuthorizationWindow {
|
|
||||||
const NAME: &'static str = "AuthorizationWindow";
|
|
||||||
const ABSTRACT: bool = false;
|
|
||||||
|
|
||||||
type Type = super::AuthorizationWindow;
|
|
||||||
type ParentType = adw::Window;
|
|
||||||
|
|
||||||
fn class_init(klass: &mut Self::Class) {
|
|
||||||
klass.bind_template();
|
|
||||||
klass.bind_template_callbacks();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn instance_init(obj: &InitializingObject<Self>) {
|
|
||||||
obj.init_template();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[template_callbacks]
|
|
||||||
impl AuthorizationWindow {
|
|
||||||
#[template_callback]
|
|
||||||
fn handle_confirm(&self, _button: Button) {
|
|
||||||
let fp = self.fingerprint.text().as_str().trim().to_owned();
|
|
||||||
self.obj().emit_by_name("confirm-clicked", &[&fp])
|
|
||||||
}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn handle_cancel(&self, _: Button) {
|
|
||||||
self.obj().emit_by_name("cancel-clicked", &[])
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(super) fn set_fingerprint(&self, fingerprint: &str) {
|
|
||||||
self.fingerprint.set_text(fingerprint);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ObjectImpl for AuthorizationWindow {
|
|
||||||
fn signals() -> &'static [Signal] {
|
|
||||||
static SIGNALS: OnceLock<Vec<Signal>> = OnceLock::new();
|
|
||||||
SIGNALS.get_or_init(|| {
|
|
||||||
vec![
|
|
||||||
Signal::builder("confirm-clicked")
|
|
||||||
.param_types([String::static_type()])
|
|
||||||
.build(),
|
|
||||||
Signal::builder("cancel-clicked").build(),
|
|
||||||
]
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl WidgetImpl for AuthorizationWindow {}
|
|
||||||
impl WindowImpl for AuthorizationWindow {}
|
|
||||||
impl ApplicationWindowImpl for AuthorizationWindow {}
|
|
||||||
impl AdwWindowImpl for AuthorizationWindow {}
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
mod imp;
|
mod imp;
|
||||||
|
|
||||||
use glib::Object;
|
use glib::Object;
|
||||||
use gtk::{gio, glib, prelude::ObjectExt, subclass::prelude::ObjectSubclassIsExt};
|
use gtk::{gio, glib};
|
||||||
|
|
||||||
glib::wrapper! {
|
glib::wrapper! {
|
||||||
pub struct FingerprintWindow(ObjectSubclass<imp::FingerprintWindow>)
|
pub struct FingerprintWindow(ObjectSubclass<imp::FingerprintWindow>)
|
||||||
@@ -11,12 +11,8 @@ glib::wrapper! {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl FingerprintWindow {
|
impl FingerprintWindow {
|
||||||
pub(crate) fn new(fingerprint: Option<String>) -> Self {
|
pub(crate) fn new() -> Self {
|
||||||
let window: Self = Object::builder().build();
|
let window: Self = Object::builder().build();
|
||||||
if let Some(fp) = fingerprint {
|
|
||||||
window.imp().fingerprint.set_property("text", fp);
|
|
||||||
window.imp().fingerprint.set_property("editable", false);
|
|
||||||
}
|
|
||||||
window
|
window
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ use super::KeyObject;
|
|||||||
|
|
||||||
glib::wrapper! {
|
glib::wrapper! {
|
||||||
pub struct KeyRow(ObjectSubclass<imp::KeyRow>)
|
pub struct KeyRow(ObjectSubclass<imp::KeyRow>)
|
||||||
@extends gtk::ListBoxRow, gtk::Widget, adw::PreferencesRow, adw::ActionRow,
|
@extends gtk::ListBoxRow, gtk::Widget, adw::PreferencesRow, adw::ExpanderRow,
|
||||||
@implements gtk::Accessible, gtk::Actionable, gtk::Buildable, gtk::ConstraintTarget;
|
@implements gtk::Accessible, gtk::Actionable, gtk::Buildable, gtk::ConstraintTarget;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
mod authorization_window;
|
|
||||||
mod client_object;
|
mod client_object;
|
||||||
mod client_row;
|
mod client_row;
|
||||||
mod fingerprint_window;
|
mod fingerprint_window;
|
||||||
@@ -147,21 +146,8 @@ fn build_ui(app: &Application) {
|
|||||||
FrontendEvent::EmulationStatus(s) => window.set_emulation(s.into()),
|
FrontendEvent::EmulationStatus(s) => window.set_emulation(s.into()),
|
||||||
FrontendEvent::AuthorizedUpdated(keys) => window.set_authorized_keys(keys),
|
FrontendEvent::AuthorizedUpdated(keys) => window.set_authorized_keys(keys),
|
||||||
FrontendEvent::PublicKeyFingerprint(fp) => window.set_pk_fp(&fp),
|
FrontendEvent::PublicKeyFingerprint(fp) => window.set_pk_fp(&fp),
|
||||||
FrontendEvent::ConnectionAttempt { fingerprint } => {
|
FrontendEvent::IncomingConnected(_fingerprint, addr, pos) => {
|
||||||
window.request_authorization(&fingerprint);
|
window.show_toast(format!("device connected: {addr} ({pos})").as_str());
|
||||||
}
|
|
||||||
FrontendEvent::DeviceConnected {
|
|
||||||
fingerprint: _,
|
|
||||||
addr,
|
|
||||||
} => {
|
|
||||||
window.show_toast(format!("device connected: {addr}").as_str());
|
|
||||||
}
|
|
||||||
FrontendEvent::DeviceEntered {
|
|
||||||
fingerprint: _,
|
|
||||||
addr,
|
|
||||||
pos,
|
|
||||||
} => {
|
|
||||||
window.show_toast(format!("device entered: {addr} ({pos})").as_str());
|
|
||||||
}
|
}
|
||||||
FrontendEvent::IncomingDisconnected(addr) => {
|
FrontendEvent::IncomingDisconnected(addr) => {
|
||||||
window.show_toast(format!("{addr} disconnected").as_str());
|
window.show_toast(format!("{addr} disconnected").as_str());
|
||||||
|
|||||||
@@ -16,10 +16,7 @@ use lan_mouse_ipc::{
|
|||||||
DEFAULT_PORT,
|
DEFAULT_PORT,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{fingerprint_window::FingerprintWindow, key_object::KeyObject, key_row::KeyRow};
|
||||||
authorization_window::AuthorizationWindow, fingerprint_window::FingerprintWindow,
|
|
||||||
key_object::KeyObject, key_row::KeyRow,
|
|
||||||
};
|
|
||||||
|
|
||||||
use super::{client_object::ClientObject, client_row::ClientRow};
|
use super::{client_object::ClientObject, client_row::ClientRow};
|
||||||
|
|
||||||
@@ -397,8 +394,8 @@ impl Window {
|
|||||||
self.request(FrontendRequest::Create);
|
self.request(FrontendRequest::Create);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn open_fingerprint_dialog(&self, fp: Option<String>) {
|
fn open_fingerprint_dialog(&self) {
|
||||||
let window = FingerprintWindow::new(fp);
|
let window = FingerprintWindow::new();
|
||||||
window.set_transient_for(Some(self));
|
window.set_transient_for(Some(self));
|
||||||
window.connect_closure(
|
window.connect_closure(
|
||||||
"confirm-clicked",
|
"confirm-clicked",
|
||||||
@@ -472,29 +469,4 @@ impl Window {
|
|||||||
pub(super) fn set_pk_fp(&self, fingerprint: &str) {
|
pub(super) fn set_pk_fp(&self, fingerprint: &str) {
|
||||||
self.imp().fingerprint_row.set_subtitle(fingerprint);
|
self.imp().fingerprint_row.set_subtitle(fingerprint);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn request_authorization(&self, fingerprint: &str) {
|
|
||||||
let window = AuthorizationWindow::new(fingerprint);
|
|
||||||
window.set_transient_for(Some(self));
|
|
||||||
window.connect_closure(
|
|
||||||
"confirm-clicked",
|
|
||||||
false,
|
|
||||||
closure_local!(
|
|
||||||
#[strong(rename_to = parent)]
|
|
||||||
self,
|
|
||||||
move |w: AuthorizationWindow, fp: String| {
|
|
||||||
w.close();
|
|
||||||
parent.open_fingerprint_dialog(Some(fp));
|
|
||||||
}
|
|
||||||
),
|
|
||||||
);
|
|
||||||
window.connect_closure(
|
|
||||||
"cancel-clicked",
|
|
||||||
false,
|
|
||||||
closure_local!(move |w: AuthorizationWindow| {
|
|
||||||
w.close();
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
window.present();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -149,7 +149,7 @@ impl Window {
|
|||||||
|
|
||||||
#[template_callback]
|
#[template_callback]
|
||||||
fn handle_add_cert_fingerprint(&self, _button: &Button) {
|
fn handle_add_cert_fingerprint(&self, _button: &Button) {
|
||||||
self.obj().open_fingerprint_dialog(None);
|
self.obj().open_fingerprint_dialog();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_port(&self, port: u16) {
|
pub fn set_port(&self, port: u16) {
|
||||||
|
|||||||
@@ -59,7 +59,6 @@ pub enum IpcError {
|
|||||||
pub const DEFAULT_PORT: u16 = 4242;
|
pub const DEFAULT_PORT: u16 = 4242;
|
||||||
|
|
||||||
#[derive(Debug, Default, Eq, Hash, PartialEq, Clone, Copy, Serialize, Deserialize)]
|
#[derive(Debug, Default, Eq, Hash, PartialEq, Clone, Copy, Serialize, Deserialize)]
|
||||||
#[serde(rename_all = "lowercase")]
|
|
||||||
pub enum Position {
|
pub enum Position {
|
||||||
#[default]
|
#[default]
|
||||||
Left,
|
Left,
|
||||||
@@ -202,21 +201,10 @@ pub enum FrontendEvent {
|
|||||||
AuthorizedUpdated(HashMap<String, String>),
|
AuthorizedUpdated(HashMap<String, String>),
|
||||||
/// public key fingerprint of this device
|
/// public key fingerprint of this device
|
||||||
PublicKeyFingerprint(String),
|
PublicKeyFingerprint(String),
|
||||||
/// new device connected
|
/// incoming connected
|
||||||
DeviceConnected {
|
IncomingConnected(String, SocketAddr, Position),
|
||||||
addr: SocketAddr,
|
|
||||||
fingerprint: String,
|
|
||||||
},
|
|
||||||
/// incoming device entered the screen
|
|
||||||
DeviceEntered {
|
|
||||||
fingerprint: String,
|
|
||||||
addr: SocketAddr,
|
|
||||||
pos: Position,
|
|
||||||
},
|
|
||||||
/// incoming disconnected
|
/// incoming disconnected
|
||||||
IncomingDisconnected(SocketAddr),
|
IncomingDisconnected(SocketAddr),
|
||||||
/// failed connection attempt (approval for fingerprint required)
|
|
||||||
ConnectionAttempt { fingerprint: String },
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Eq, PartialEq, Clone, Serialize, Deserialize)]
|
#[derive(Debug, Eq, PartialEq, Clone, Serialize, Deserialize)]
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ struct ConfigToml {
|
|||||||
port: Option<u16>,
|
port: Option<u16>,
|
||||||
release_bind: Option<Vec<scancode::Linux>>,
|
release_bind: Option<Vec<scancode::Linux>>,
|
||||||
cert_path: Option<PathBuf>,
|
cert_path: Option<PathBuf>,
|
||||||
clients: Option<Vec<TomlClient>>,
|
clients: Vec<TomlClient>,
|
||||||
authorized_fingerprints: Option<HashMap<String, String>>,
|
authorized_fingerprints: Option<HashMap<String, String>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -61,7 +61,7 @@ struct TomlClient {
|
|||||||
host_name: Option<String>,
|
host_name: Option<String>,
|
||||||
ips: Option<Vec<IpAddr>>,
|
ips: Option<Vec<IpAddr>>,
|
||||||
port: Option<u16>,
|
port: Option<u16>,
|
||||||
position: Option<Position>,
|
pos: Option<Position>,
|
||||||
activate_on_startup: Option<bool>,
|
activate_on_startup: Option<bool>,
|
||||||
enter_hook: Option<String>,
|
enter_hook: Option<String>,
|
||||||
}
|
}
|
||||||
@@ -262,7 +262,7 @@ impl From<TomlClient> for ConfigClient {
|
|||||||
let hostname = toml.hostname;
|
let hostname = toml.hostname;
|
||||||
let ips = HashSet::from_iter(toml.ips.into_iter().flatten());
|
let ips = HashSet::from_iter(toml.ips.into_iter().flatten());
|
||||||
let port = toml.port.unwrap_or(DEFAULT_PORT);
|
let port = toml.port.unwrap_or(DEFAULT_PORT);
|
||||||
let pos = toml.position.unwrap_or_default();
|
let pos = toml.pos.unwrap_or_default();
|
||||||
Self {
|
Self {
|
||||||
ips,
|
ips,
|
||||||
hostname,
|
hostname,
|
||||||
@@ -370,7 +370,6 @@ impl Config {
|
|||||||
self.config_toml
|
self.config_toml
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.map(|c| c.clients.clone())
|
.map(|c| c.clients.clone())
|
||||||
.unwrap_or_default()
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.flatten()
|
.flatten()
|
||||||
.map(From::<TomlClient>::from)
|
.map(From::<TomlClient>::from)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use crate::listen::{LanMouseListener, ListenEvent, ListenerCreationError};
|
use crate::listen::{LanMouseListener, ListenerCreationError};
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use input_emulation::{EmulationHandle, InputEmulation, InputEmulationError};
|
use input_emulation::{EmulationHandle, InputEmulation, InputEmulationError};
|
||||||
use input_event::Event;
|
use input_event::Event;
|
||||||
@@ -24,15 +24,8 @@ pub(crate) struct Emulation {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) enum EmulationEvent {
|
pub(crate) enum EmulationEvent {
|
||||||
Connected {
|
|
||||||
addr: SocketAddr,
|
|
||||||
fingerprint: String,
|
|
||||||
},
|
|
||||||
ConnectionAttempt {
|
|
||||||
fingerprint: String,
|
|
||||||
},
|
|
||||||
/// new connection
|
/// new connection
|
||||||
Entered {
|
Connected {
|
||||||
/// address of the connection
|
/// address of the connection
|
||||||
addr: SocketAddr,
|
addr: SocketAddr,
|
||||||
/// position of the connection
|
/// position of the connection
|
||||||
@@ -41,9 +34,7 @@ pub(crate) enum EmulationEvent {
|
|||||||
fingerprint: String,
|
fingerprint: String,
|
||||||
},
|
},
|
||||||
/// connection closed
|
/// connection closed
|
||||||
Disconnected {
|
Disconnected { addr: SocketAddr },
|
||||||
addr: SocketAddr,
|
|
||||||
},
|
|
||||||
/// the port of the listener has changed
|
/// the port of the listener has changed
|
||||||
PortChanged(Result<u16, ListenerCreationError>),
|
PortChanged(Result<u16, ListenerCreationError>),
|
||||||
/// emulation was disabled
|
/// emulation was disabled
|
||||||
@@ -130,8 +121,11 @@ impl ListenTask {
|
|||||||
let mut last_response = HashMap::new();
|
let mut last_response = HashMap::new();
|
||||||
loop {
|
loop {
|
||||||
select! {
|
select! {
|
||||||
e = self.listener.next() => {match e {
|
e = self.listener.next() => {
|
||||||
Some(ListenEvent::Msg { event, addr }) => {
|
let (event, addr) = match e {
|
||||||
|
Some(e) => e,
|
||||||
|
None => break,
|
||||||
|
};
|
||||||
log::trace!("{event} <-<-<-<-<- {addr}");
|
log::trace!("{event} <-<-<-<-<- {addr}");
|
||||||
last_response.insert(addr, Instant::now());
|
last_response.insert(addr, Instant::now());
|
||||||
match event {
|
match event {
|
||||||
@@ -140,7 +134,7 @@ impl ListenTask {
|
|||||||
log::info!("releasing capture: {addr} entered this device");
|
log::info!("releasing capture: {addr} entered this device");
|
||||||
self.event_tx.send(EmulationEvent::ReleaseNotify).expect("channel closed");
|
self.event_tx.send(EmulationEvent::ReleaseNotify).expect("channel closed");
|
||||||
self.listener.reply(addr, ProtoEvent::Ack(0)).await;
|
self.listener.reply(addr, ProtoEvent::Ack(0)).await;
|
||||||
self.event_tx.send(EmulationEvent::Entered{addr, pos: to_ipc_pos(pos), fingerprint}).expect("channel closed");
|
self.event_tx.send(EmulationEvent::Connected{addr, pos: to_ipc_pos(pos), fingerprint}).expect("channel closed");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ProtoEvent::Leave(_) => {
|
ProtoEvent::Leave(_) => {
|
||||||
@@ -152,14 +146,6 @@ impl ListenTask {
|
|||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Some(ListenEvent::Accept { addr, fingerprint }) => {
|
|
||||||
self.event_tx.send(EmulationEvent::Connected { addr, fingerprint }).expect("channel closed");
|
|
||||||
}
|
|
||||||
Some(ListenEvent::Rejected { fingerprint }) => {
|
|
||||||
self.event_tx.send(EmulationEvent::ConnectionAttempt { fingerprint }).expect("channel closed");
|
|
||||||
}
|
|
||||||
None => break
|
|
||||||
}}
|
|
||||||
event = self.emulation_proxy.event() => {
|
event = self.emulation_proxy.event() => {
|
||||||
self.event_tx.send(event).expect("channel closed");
|
self.event_tx.send(event).expect("channel closed");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,15 +3,15 @@ use lan_mouse_proto::{ProtoEvent, MAX_EVENT_SIZE};
|
|||||||
use local_channel::mpsc::{channel, Receiver, Sender};
|
use local_channel::mpsc::{channel, Receiver, Sender};
|
||||||
use rustls::pki_types::CertificateDer;
|
use rustls::pki_types::CertificateDer;
|
||||||
use std::{
|
use std::{
|
||||||
collections::{HashMap, VecDeque},
|
collections::HashMap,
|
||||||
net::SocketAddr,
|
net::SocketAddr,
|
||||||
rc::Rc,
|
rc::Rc,
|
||||||
sync::{Arc, Mutex, RwLock},
|
sync::{Arc, RwLock},
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
use tokio::{
|
use tokio::{
|
||||||
sync::Mutex as AsyncMutex,
|
sync::Mutex,
|
||||||
task::{spawn_local, JoinHandle},
|
task::{spawn_local, JoinHandle},
|
||||||
};
|
};
|
||||||
use webrtc_dtls::{
|
use webrtc_dtls::{
|
||||||
@@ -34,25 +34,11 @@ pub enum ListenerCreationError {
|
|||||||
|
|
||||||
type ArcConn = Arc<dyn Conn + Send + Sync>;
|
type ArcConn = Arc<dyn Conn + Send + Sync>;
|
||||||
|
|
||||||
pub(crate) enum ListenEvent {
|
|
||||||
Msg {
|
|
||||||
event: ProtoEvent,
|
|
||||||
addr: SocketAddr,
|
|
||||||
},
|
|
||||||
Accept {
|
|
||||||
addr: SocketAddr,
|
|
||||||
fingerprint: String,
|
|
||||||
},
|
|
||||||
Rejected {
|
|
||||||
fingerprint: String,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) struct LanMouseListener {
|
pub(crate) struct LanMouseListener {
|
||||||
listen_rx: Receiver<ListenEvent>,
|
listen_rx: Receiver<(ProtoEvent, SocketAddr)>,
|
||||||
listen_tx: Sender<ListenEvent>,
|
listen_tx: Sender<(ProtoEvent, SocketAddr)>,
|
||||||
listen_task: JoinHandle<()>,
|
listen_task: JoinHandle<()>,
|
||||||
conns: Rc<AsyncMutex<Vec<(SocketAddr, ArcConn)>>>,
|
conns: Rc<Mutex<Vec<(SocketAddr, ArcConn)>>>,
|
||||||
request_port_change: Sender<u16>,
|
request_port_change: Sender<u16>,
|
||||||
port_changed: Receiver<Result<u16, ListenerCreationError>>,
|
port_changed: Receiver<Result<u16, ListenerCreationError>>,
|
||||||
}
|
}
|
||||||
@@ -72,12 +58,9 @@ impl LanMouseListener {
|
|||||||
let (listen_tx, listen_rx) = channel();
|
let (listen_tx, listen_rx) = channel();
|
||||||
let (request_port_change, mut request_port_change_rx) = channel();
|
let (request_port_change, mut request_port_change_rx) = channel();
|
||||||
let (port_changed_tx, port_changed) = channel();
|
let (port_changed_tx, port_changed) = channel();
|
||||||
let connection_attempts: Arc<Mutex<VecDeque<String>>> = Default::default();
|
|
||||||
|
|
||||||
let authorized = authorized_keys.clone();
|
let authorized = authorized_keys.clone();
|
||||||
let verify_peer_certificate: Option<VerifyPeerCertificateFn> = {
|
let verify_peer_certificate: Option<VerifyPeerCertificateFn> = Some(Arc::new(
|
||||||
let connection_attempts = connection_attempts.clone();
|
|
||||||
Some(Arc::new(
|
|
||||||
move |certs: &[Vec<u8>], _chains: &[CertificateDer<'static>]| {
|
move |certs: &[Vec<u8>], _chains: &[CertificateDer<'static>]| {
|
||||||
assert!(certs.len() == 1);
|
assert!(certs.len() == 1);
|
||||||
let fingerprints = certs
|
let fingerprints = certs
|
||||||
@@ -91,16 +74,10 @@ impl LanMouseListener {
|
|||||||
{
|
{
|
||||||
Ok(())
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
let fingerprint = fingerprints.into_iter().next().expect("fingerprint");
|
|
||||||
connection_attempts
|
|
||||||
.lock()
|
|
||||||
.expect("lock")
|
|
||||||
.push_back(fingerprint);
|
|
||||||
Err(webrtc_dtls::Error::ErrVerifyDataMismatch)
|
Err(webrtc_dtls::Error::ErrVerifyDataMismatch)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
))
|
));
|
||||||
};
|
|
||||||
let cfg = Config {
|
let cfg = Config {
|
||||||
certificates: vec![cert.clone()],
|
certificates: vec![cert.clone()],
|
||||||
extended_master_secret: ExtendedMasterSecretType::Require,
|
extended_master_secret: ExtendedMasterSecretType::Require,
|
||||||
@@ -112,14 +89,11 @@ impl LanMouseListener {
|
|||||||
let listen_addr = SocketAddr::new("0.0.0.0".parse().expect("invalid ip"), port);
|
let listen_addr = SocketAddr::new("0.0.0.0".parse().expect("invalid ip"), port);
|
||||||
let mut listener = listen(listen_addr, cfg.clone()).await?;
|
let mut listener = listen(listen_addr, cfg.clone()).await?;
|
||||||
|
|
||||||
let conns: Rc<AsyncMutex<Vec<(SocketAddr, ArcConn)>>> =
|
let conns: Rc<Mutex<Vec<(SocketAddr, ArcConn)>>> = Rc::new(Mutex::new(Vec::new()));
|
||||||
Rc::new(AsyncMutex::new(Vec::new()));
|
|
||||||
|
|
||||||
let conns_clone = conns.clone();
|
let conns_clone = conns.clone();
|
||||||
let listen_task: JoinHandle<()> = {
|
let tx = listen_tx.clone();
|
||||||
let listen_tx = listen_tx.clone();
|
let listen_task: JoinHandle<()> = spawn_local(async move {
|
||||||
let connection_attempts = connection_attempts.clone();
|
|
||||||
spawn_local(async move {
|
|
||||||
loop {
|
loop {
|
||||||
let sleep = tokio::time::sleep(Duration::from_secs(2));
|
let sleep = tokio::time::sleep(Duration::from_secs(2));
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
@@ -130,31 +104,9 @@ impl LanMouseListener {
|
|||||||
log::info!("dtls client connected, ip: {addr}");
|
log::info!("dtls client connected, ip: {addr}");
|
||||||
let mut conns = conns_clone.lock().await;
|
let mut conns = conns_clone.lock().await;
|
||||||
conns.push((addr, conn.clone()));
|
conns.push((addr, conn.clone()));
|
||||||
let dtls_conn: &DTLSConn = conn.as_any().downcast_ref().expect("dtls conn");
|
spawn_local(read_loop(conns_clone.clone(), addr, conn, tx.clone()));
|
||||||
let certs = dtls_conn.connection_state().await.peer_certificates;
|
|
||||||
let cert = certs.first().expect("cert");
|
|
||||||
let fingerprint = crypto::generate_fingerprint(cert);
|
|
||||||
listen_tx.send(ListenEvent::Accept { addr, fingerprint }).expect("channel closed");
|
|
||||||
spawn_local(read_loop(conns_clone.clone(), addr, conn, listen_tx.clone()));
|
|
||||||
},
|
},
|
||||||
Err(e) => {
|
Err(e) => log::warn!("accept: {e}"),
|
||||||
if let Error::Std(ref e) = e {
|
|
||||||
if let Some(e) = e.0.downcast_ref::<webrtc_dtls::Error>() {
|
|
||||||
match e {
|
|
||||||
webrtc_dtls::Error::ErrVerifyDataMismatch => {
|
|
||||||
if let Some(fingerprint) = connection_attempts.lock().expect("lock").pop_front() {
|
|
||||||
listen_tx.send(ListenEvent::Rejected { fingerprint }).expect("channel closed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => log::warn!("accept: {e}"),
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
log::warn!("accept: {e:?}");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
log::warn!("accept: {e:?}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
port = request_port_change_rx.recv() => {
|
port = request_port_change_rx.recv() => {
|
||||||
let port = port.expect("channel closed");
|
let port = port.expect("channel closed");
|
||||||
@@ -173,8 +125,7 @@ impl LanMouseListener {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
};
|
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
conns,
|
conns,
|
||||||
@@ -235,7 +186,7 @@ impl LanMouseListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Stream for LanMouseListener {
|
impl Stream for LanMouseListener {
|
||||||
type Item = ListenEvent;
|
type Item = (ProtoEvent, SocketAddr);
|
||||||
|
|
||||||
fn poll_next(
|
fn poll_next(
|
||||||
mut self: std::pin::Pin<&mut Self>,
|
mut self: std::pin::Pin<&mut Self>,
|
||||||
@@ -246,18 +197,16 @@ impl Stream for LanMouseListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn read_loop(
|
async fn read_loop(
|
||||||
conns: Rc<AsyncMutex<Vec<(SocketAddr, ArcConn)>>>,
|
conns: Rc<Mutex<Vec<(SocketAddr, ArcConn)>>>,
|
||||||
addr: SocketAddr,
|
addr: SocketAddr,
|
||||||
conn: ArcConn,
|
conn: ArcConn,
|
||||||
dtls_tx: Sender<ListenEvent>,
|
dtls_tx: Sender<(ProtoEvent, SocketAddr)>,
|
||||||
) -> Result<(), Error> {
|
) -> Result<(), Error> {
|
||||||
let mut b = [0u8; MAX_EVENT_SIZE];
|
let mut b = [0u8; MAX_EVENT_SIZE];
|
||||||
|
|
||||||
while conn.recv(&mut b).await.is_ok() {
|
while conn.recv(&mut b).await.is_ok() {
|
||||||
match b.try_into() {
|
match b.try_into() {
|
||||||
Ok(event) => dtls_tx
|
Ok(event) => dtls_tx.send((event, addr)).expect("channel closed"),
|
||||||
.send(ListenEvent::Msg { event, addr })
|
|
||||||
.expect("channel closed"),
|
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::warn!("error receiving event: {e}");
|
log::warn!("error receiving event: {e}");
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -211,10 +211,7 @@ impl Service {
|
|||||||
|
|
||||||
fn handle_emulation_event(&mut self, event: EmulationEvent) {
|
fn handle_emulation_event(&mut self, event: EmulationEvent) {
|
||||||
match event {
|
match event {
|
||||||
EmulationEvent::ConnectionAttempt { fingerprint } => {
|
EmulationEvent::Connected {
|
||||||
self.notify_frontend(FrontendEvent::ConnectionAttempt { fingerprint });
|
|
||||||
}
|
|
||||||
EmulationEvent::Entered {
|
|
||||||
addr,
|
addr,
|
||||||
pos,
|
pos,
|
||||||
fingerprint,
|
fingerprint,
|
||||||
@@ -222,11 +219,7 @@ impl Service {
|
|||||||
// check if already registered
|
// check if already registered
|
||||||
if !self.incoming_conns.contains(&addr) {
|
if !self.incoming_conns.contains(&addr) {
|
||||||
self.add_incoming(addr, pos, fingerprint.clone());
|
self.add_incoming(addr, pos, fingerprint.clone());
|
||||||
self.notify_frontend(FrontendEvent::DeviceEntered {
|
self.notify_frontend(FrontendEvent::IncomingConnected(fingerprint, addr, pos));
|
||||||
fingerprint,
|
|
||||||
addr,
|
|
||||||
pos,
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
self.update_incoming(addr, pos, fingerprint);
|
self.update_incoming(addr, pos, fingerprint);
|
||||||
}
|
}
|
||||||
@@ -253,9 +246,6 @@ impl Service {
|
|||||||
self.notify_frontend(FrontendEvent::EmulationStatus(self.emulation_status));
|
self.notify_frontend(FrontendEvent::EmulationStatus(self.emulation_status));
|
||||||
}
|
}
|
||||||
EmulationEvent::ReleaseNotify => self.capture.release(),
|
EmulationEvent::ReleaseNotify => self.capture.release(),
|
||||||
EmulationEvent::Connected { addr, fingerprint } => {
|
|
||||||
self.notify_frontend(FrontendEvent::DeviceConnected { addr, fingerprint });
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -357,11 +347,7 @@ impl Service {
|
|||||||
self.remove_incoming(addr);
|
self.remove_incoming(addr);
|
||||||
self.add_incoming(addr, pos, fingerprint.clone());
|
self.add_incoming(addr, pos, fingerprint.clone());
|
||||||
self.notify_frontend(FrontendEvent::IncomingDisconnected(addr));
|
self.notify_frontend(FrontendEvent::IncomingDisconnected(addr));
|
||||||
self.notify_frontend(FrontendEvent::DeviceEntered {
|
self.notify_frontend(FrontendEvent::IncomingConnected(fingerprint, addr, pos));
|
||||||
fingerprint,
|
|
||||||
addr,
|
|
||||||
pos,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
3
wix/bundle/.gitignore
vendored
Normal file
3
wix/bundle/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
/bin
|
||||||
|
/obj
|
||||||
|
icon.ico
|
||||||
16
wix/bundle/Bundle.wixproj
Normal file
16
wix/bundle/Bundle.wixproj
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<Project Sdk="WixToolset.Sdk/5.0.0">
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Bundle</OutputType>
|
||||||
|
<TargetExt>.exe</TargetExt>
|
||||||
|
<Platforms>x64</Platforms>
|
||||||
|
<InstallerPlatform>x64</InstallerPlatform>
|
||||||
|
</PropertyGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="WixToolset.Heat">
|
||||||
|
<Version>5.0.2</Version>
|
||||||
|
</PackageReference>
|
||||||
|
<PackageReference Include="WixToolset.Bal.wixext">
|
||||||
|
<Version>5.0.2</Version>
|
||||||
|
</PackageReference>
|
||||||
|
</ItemGroup>
|
||||||
|
</Project>
|
||||||
42
wix/bundle/Bundle.wxs
Normal file
42
wix/bundle/Bundle.wxs
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs"
|
||||||
|
xmlns:bal="http://wixtoolset.org/schemas/v4/wxs/bal">
|
||||||
|
|
||||||
|
<Bundle
|
||||||
|
Name="Lan Mouse"
|
||||||
|
Version="0.10.0"
|
||||||
|
UpgradeCode="{39A9744D-9D6E-4CD3-A84F-9E034786A7B1}"
|
||||||
|
Compressed="no"
|
||||||
|
SplashScreenSourceFile="icon.ico">
|
||||||
|
|
||||||
|
<BootstrapperApplication>
|
||||||
|
<bal:WixStandardBootstrapperApplication
|
||||||
|
LicenseUrl=""
|
||||||
|
Theme="hyperlinkLicense" />
|
||||||
|
</BootstrapperApplication>
|
||||||
|
|
||||||
|
<Chain>
|
||||||
|
<!-- Visual C++ 2015-2022 Redistributable (x64) - 14.40.33810 -->
|
||||||
|
<ExePackage
|
||||||
|
Id="VC_REDIST_X64"
|
||||||
|
DisplayName="Microsoft Visual C++ 2015-2022 Redistributable (x64) - 14.40.33810"
|
||||||
|
PerMachine="yes"
|
||||||
|
Permanent="yes"
|
||||||
|
Protocol="burn"
|
||||||
|
InstallCondition="VersionNT64 AND (ARCH_NAME = "AMD64")"
|
||||||
|
DetectCondition="(VCRUNTIME_X64_VER >= VCRUNTIME_VER) AND VersionNT64 AND (ARCH_NAME = "AMD64")"
|
||||||
|
InstallArguments="/install /quiet /norestart"
|
||||||
|
RepairArguments="/repair /quiet /norestart"
|
||||||
|
UninstallArguments="/uninstall /quiet /norestart">
|
||||||
|
<ExePackagePayload
|
||||||
|
Name="VC_redist.x64.exe"
|
||||||
|
ProductName="Microsoft Visual C++ 2015-2022 Redistributable (x64) - 14.40.33810"
|
||||||
|
Description="Microsoft Visual C++ 2015-2022 Redistributable (x64) - 14.40.33810"
|
||||||
|
Hash="5935B69F5138AC3FBC33813C74DA853269BA079F910936AEFA95E230C6092B92F6225BFFB594E5DD35FF29BF260E4B35F91ADEDE90FDF5F062030D8666FD0104"
|
||||||
|
Size="25397512"
|
||||||
|
Version="14.40.33810.0"
|
||||||
|
DownloadUrl="https://download.visualstudio.microsoft.com/download/pr/1754ea58-11a6-44ab-a262-696e194ce543/3642E3F95D50CC193E4B5A0B0FFBF7FE2C08801517758B4C8AEB7105A091208A/VC_redist.x64.exe" />
|
||||||
|
</ExePackage>
|
||||||
|
<MsiPackage SourceFile="..\lan-mouse\bin\Debug\en-US\LanMouse.msi" Compressed="yes"/>
|
||||||
|
</Chain>
|
||||||
|
</Bundle>
|
||||||
|
</Wix>
|
||||||
3
wix/lan-mouse/.gitignore
vendored
Normal file
3
wix/lan-mouse/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
/bin
|
||||||
|
/obj
|
||||||
|
icon.ico
|
||||||
13
wix/lan-mouse/Folders.wxs
Normal file
13
wix/lan-mouse/Folders.wxs
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
|
||||||
|
<Fragment>
|
||||||
|
<StandardDirectory Id="ProgramFiles64Folder">
|
||||||
|
<Directory Id="INSTALLFOLDER" Name="!(bind.Property.Manufacturer) !(bind.Property.ProductName)">
|
||||||
|
<Directory Id="SHARE" Name="share"/>
|
||||||
|
<Directory Id="LIB" Name="lib"/>
|
||||||
|
</Directory>
|
||||||
|
</StandardDirectory>
|
||||||
|
<StandardDirectory Id="ProgramMenuFolder">
|
||||||
|
<Directory Id="ApplicationProgramsFolder" Name="!(bind.Property.ProductName)"/>
|
||||||
|
</StandardDirectory>
|
||||||
|
</Fragment>
|
||||||
|
</Wix>
|
||||||
34
wix/lan-mouse/LanMouse.wixproj
Normal file
34
wix/lan-mouse/LanMouse.wixproj
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<Project Sdk="WixToolset.Sdk/5.0.2">
|
||||||
|
<PropertyGroup>
|
||||||
|
<InstallerPlatform>x64</InstallerPlatform>
|
||||||
|
</PropertyGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="WixToolset.Heat">
|
||||||
|
<Version>5.0.2</Version>
|
||||||
|
</PackageReference>
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<HarvestDirectory Include="C:\gtk-build\gtk\x64\release\bin">
|
||||||
|
<ComponentGroupName>GTKBIN</ComponentGroupName>
|
||||||
|
<DirectoryRefId>INSTALLFOLDER</DirectoryRefId>
|
||||||
|
<SuppressRegistry>true</SuppressRegistry>
|
||||||
|
</HarvestDirectory>
|
||||||
|
<BindPath Include="C:\gtk-build\gtk\x64\release\bin" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<HarvestDirectory Include="C:\gtk-build\gtk\x64\release\share\icons">
|
||||||
|
<ComponentGroupName>GTKICONS</ComponentGroupName>
|
||||||
|
<DirectoryRefId>SHARE</DirectoryRefId>
|
||||||
|
<SuppressRegistry>true</SuppressRegistry>
|
||||||
|
</HarvestDirectory>
|
||||||
|
<BindPath Include="C:\gtk-build\gtk\x64\release\share\icons" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<HarvestDirectory Include="C:\gtk-build\gtk\x64\release\lib\gdk-pixbuf-2.0">
|
||||||
|
<ComponentGroupName>GTKLIBS</ComponentGroupName>
|
||||||
|
<DirectoryRefId>LIB</DirectoryRefId>
|
||||||
|
<SuppressRegistry>true</SuppressRegistry>
|
||||||
|
</HarvestDirectory>
|
||||||
|
<BindPath Include="C:\gtk-build\gtk\x64\release\lib\gdk-pixbuf-2.0" />
|
||||||
|
</ItemGroup>
|
||||||
|
</Project>
|
||||||
32
wix/lan-mouse/LanMouseComponents.wxs
Normal file
32
wix/lan-mouse/LanMouseComponents.wxs
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
|
||||||
|
<Fragment>
|
||||||
|
<ComponentGroup Id="LanMouseComponents" Directory="INSTALLFOLDER" Subdirectory="bin">
|
||||||
|
<Component Guid="{ECB52D3E-28AD-4BEC-B9DF-E01CCAB356BE}">
|
||||||
|
<!-- the main binary -->
|
||||||
|
<File Source="..\..\target\release\lan-mouse.exe"/>
|
||||||
|
|
||||||
|
<!-- visual c runtime dll -->
|
||||||
|
<!--<File Source="C:\windows\system32\VCRUNTIME140.dll"/>-->
|
||||||
|
<!--<File Source="C:\windows\system32\VCRUNTIME140_1.dll"/>-->
|
||||||
|
</Component>
|
||||||
|
<!-- start menu entry-->
|
||||||
|
<Component Id="ApplicationShortcut" Directory="ApplicationProgramsFolder">
|
||||||
|
<Shortcut Id="ApplicationStartMenuShortcut"
|
||||||
|
Name="!(bind.Property.ProductName)"
|
||||||
|
Description ="Mouse and Keyboard sharing Software"
|
||||||
|
Target="[INSTALLFOLDER]bin\lan-mouse.exe"
|
||||||
|
WorkingDirectory="INSTALLFOLDER">
|
||||||
|
<Icon Id="LanMouse" SourceFile=".\icon.ico"/>
|
||||||
|
</Shortcut>
|
||||||
|
<RemoveFolder Id="ApplicationProgramsFolder" On="uninstall"/>
|
||||||
|
<RegistryValue
|
||||||
|
Root="HKCU"
|
||||||
|
Key="Software\Feschber\LanMouse"
|
||||||
|
Name="installed"
|
||||||
|
Type="integer"
|
||||||
|
Value="1"
|
||||||
|
KeyPath="yes"/>
|
||||||
|
</Component>
|
||||||
|
</ComponentGroup>
|
||||||
|
</Fragment>
|
||||||
|
</Wix>
|
||||||
8
wix/lan-mouse/Package.en-us.wxl
Normal file
8
wix/lan-mouse/Package.en-us.wxl
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
<!--
|
||||||
|
This file contains the declaration of all the localizable strings.
|
||||||
|
-->
|
||||||
|
<WixLocalization xmlns="http://wixtoolset.org/schemas/v4/wxl" Culture="en-US">
|
||||||
|
|
||||||
|
<String Id="DowngradeError" Value="A newer version of [ProductName] is already installed." />
|
||||||
|
|
||||||
|
</WixLocalization>
|
||||||
16
wix/lan-mouse/Package.wxs
Normal file
16
wix/lan-mouse/Package.wxs
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
|
||||||
|
<Package Name="Lan Mouse"
|
||||||
|
Manufacturer="Ferdinand Schober"
|
||||||
|
Version="0.10.0.0"
|
||||||
|
UpgradeCode="{a330cd60-4c35-4a54-8bb6-75b3049b46c6}">
|
||||||
|
<MajorUpgrade DowngradeErrorMessage="!(loc.DowngradeError)" />
|
||||||
|
|
||||||
|
<MediaTemplate EmbedCab="yes"/>
|
||||||
|
<Feature Id="Main">
|
||||||
|
<ComponentGroupRef Id="GTKBIN"/>
|
||||||
|
<ComponentGroupRef Id="GTKICONS"/>
|
||||||
|
<ComponentGroupRef Id="GTKLIBS"/>
|
||||||
|
<ComponentGroupRef Id="LanMouseComponents"/>
|
||||||
|
</Feature>
|
||||||
|
</Package>
|
||||||
|
</Wix>
|
||||||
2
wix/lan-mouse/build.ps1
Normal file
2
wix/lan-mouse/build.ps1
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
magick -background none -density 384 ..\lan-mouse-gtk\resources\de.feschber.LanMouse.svg -trim -define icon:auto-resize icon.ico
|
||||||
|
dotnet build
|
||||||
Reference in New Issue
Block a user