diff --git a/dist/mp-server-v2.exe b/dist/mp-server-v2.exe new file mode 100644 index 0000000..81bc92d Binary files /dev/null and b/dist/mp-server-v2.exe differ diff --git a/mp-server-v2.py b/mp-server-v2.py index f6ab0ea..fab2845 100644 --- a/mp-server-v2.py +++ b/mp-server-v2.py @@ -12,7 +12,9 @@ from flask import Flask, render_template_string, request, jsonify, send_file import webbrowser from waitress import serve import logging -import netifaces +import socket +import qrcode +import sys class MacroPadServer: def __init__(self, root): @@ -22,9 +24,13 @@ class MacroPadServer: self.configure_styles() # Set up directories - base_dir = os.path.dirname(os.path.abspath(__file__)) + if getattr(sys, 'frozen', False): + base_dir = os.path.dirname(sys.executable) + else: + base_dir = os.path.dirname(os.path.abspath(__file__)) self.data_file = os.path.join(base_dir, "macros.json") self.images_dir = os.path.join(base_dir, "macro_images") + self.app_dir = base_dir os.makedirs(self.images_dir, exist_ok=True) self.macros = {} @@ -178,7 +184,7 @@ class MacroPadServer: if macro["image_path"] in self.image_cache: button_image = self.image_cache[macro["image_path"]] else: - img_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), macro["image_path"]) + img_path = os.path.join(self.app_dir, macro["image_path"]) img = Image.open(img_path) img = img.resize((32, 32)) button_image = ImageTk.PhotoImage(img) @@ -213,19 +219,19 @@ class MacroPadServer: self.flask_thread.start() self.server_button.config(text="Stop Web Server") - # Get all IP addresses to display - ip_addresses = self.get_ip_addresses() - if ip_addresses: + # Get the systems internal IP address + ip_address = self.get_ip_addresses() + if ip_address: # Set the URL display - urls = [f"http://{ip}:{self.server_port}" for ip in ip_addresses] - url_text = "Web UI available at:\n" + "\n".join(urls) + url = f"http://{ip_address}:{self.server_port}" + url_text = "Web UI available at:\n" + "\n" + url self.url_var.set(url_text) # Enable browser button self.browser_button.config(state=tk.NORMAL) # Generate and display QR code for the first IP - self.generate_qr_code(urls[0]) + self.generate_qr_code(url) else: self.url_var.set("No network interfaces found") except Exception as e: @@ -246,30 +252,34 @@ class MacroPadServer: # The Flask server will be stopped on the next request def get_ip_addresses(self): - ip_addresses = [] + """Get the primary internal IPv4 address of the machine.""" try: - # Get all network interfaces - interfaces = netifaces.interfaces() - for interface in interfaces: - # Skip loopback interface - if interface.startswith('lo'): - continue - - # Get addresses for this interface - addresses = netifaces.ifaddresses(interface) - if netifaces.AF_INET in addresses: - for address in addresses[netifaces.AF_INET]: - ip = address.get('addr') - if ip and not ip.startswith('127.'): - ip_addresses.append(ip) - - # Always include localhost - ip_addresses.append('localhost') - - return ip_addresses + # Create a socket to connect to an external server + # This helps determine which network interface is used for outbound connections + s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) + # We don't need to actually send data - just configure the socket + s.connect(("8.8.8.8", 80)) + # Get the IP address that would be used for this connection + ip = s.getsockname()[0] + s.close() + return ip except Exception as e: - print(f"Error getting IP addresses: {e}") - return ['localhost'] + print(f"Error getting IP address: {e}") + # Fallback method if the above doesn't work + try: + hostname = socket.gethostname() + ip = socket.gethostbyname(hostname) + # Don't return localhost address + if ip.startswith("127."): + for addr_info in socket.getaddrinfo(hostname, None): + potential_ip = addr_info[4][0] + # If IPv4 and not localhost + if '.' in potential_ip and not potential_ip.startswith("127."): + return potential_ip + else: + return ip + except: + return "127.0.0.1" # Last resort fallback def generate_qr_code(self, url): try: @@ -316,170 +326,196 @@ class MacroPadServer: # Define HTML templates index_html = ''' - -
- - -