import subprocess
import logging
import os
import gi

gi.require_version("Gtk", "3.0")
from gi.repository import Gtk, GLib, Pango
from ks_includes.screen_panel import ScreenPanel
from ks_includes.sdbus_nm import SdbusNm


class Panel(ScreenPanel):

    def __init__(self, screen, title):
        title = title or _("Network")
        super().__init__(screen, title)
        self.show_add = False
        try:
            self.sdbus_nm = SdbusNm(self.popup_callback)
        except Exception as e:
            logging.exception("Failed to initialize")
            self.sdbus_nm = None
            self.error_box = Gtk.Box(
                orientation=Gtk.Orientation.VERTICAL,
                hexpand=True,
                vexpand=True
            )
            message = (
                _("Failed to initialize") + "\n"
                + "This panel needs NetworkManager installed into the system\n"
                + "And the apropriate permissions, without them it will not function.\n"
                + f"\n{e}\n"
            )
            self.error_box.add(
                Gtk.Label(
                    label=message,
                    wrap=True,
                    wrap_mode=Pango.WrapMode.WORD_CHAR,
                )
            )
            self.error_box.set_valign(Gtk.Align.CENTER)
            self.content.add(self.error_box)
            self._screen.panels_reinit.append(self._screen._cur_panels[-1])
            return
        self.reload_timer_id = None
        self.monitor_timer_id = None
        self.delay_reload_timer_id = None
        self.init_status = False
        self.reload = False
        self.last_state = None
        self.network_list = Gtk.ListBox(selection_mode=Gtk.SelectionMode.NONE)
        self.network_rows = {}
        self.networks = {}
        self.wifi_signal_icons = {
            "excellent": self._gtk.PixbufFromIcon(
                "wifi_excellent", width=self._gtk.content_width * 0.06, height=self._gtk.content_height * 0.06
            ),
            "good": self._gtk.PixbufFromIcon(
                "wifi_good", width=self._gtk.content_width * 0.06, height=self._gtk.content_height * 0.06
            ),
            "fair": self._gtk.PixbufFromIcon(
                "wifi_fair", width=self._gtk.content_width * 0.06, height=self._gtk.content_height * 0.06
            ),
            "weak": self._gtk.PixbufFromIcon(
                "wifi_weak", width=self._gtk.content_width * 0.06, height=self._gtk.content_height * 0.06
            ),
        }

        self.network_interfaces = self.sdbus_nm.get_interfaces()
        logging.info(f"Network interfaces: {self.network_interfaces}")

        self.wireless_interfaces = [iface.interface for iface in self.sdbus_nm.get_wireless_interfaces()]
        logging.info(f"Wireless interfaces: {self.wireless_interfaces}")

        self.interface = self.sdbus_nm.get_primary_interface()
        logging.info(f"Primary interface: {self.interface}")

        self.labels['interface'] = Gtk.Label(hexpand=True)
        self.labels['ip'] = Gtk.Label(hexpand=True)
        self.network_interface_refresh()

        self.reload_button = self._gtk.Button("refresh", None, "custom-icon-button", self.bts)
        self.reload_button.set_no_show_all(True)
        self.reload_button.connect("clicked", self.reload_networks)
        self.reload_button.set_hexpand(False)

        self.wifi_toggle = Gtk.Switch(
            width_request=round(self._gtk.font_size * 2),
            height_request=round(self._gtk.font_size),
            active=self.sdbus_nm.is_wifi_enabled()
        )
        self.wifi_toggle.connect("notify::active", self.toggle_wifi)

        sbox = Gtk.Box(hexpand=True, vexpand=False)
        sbox.add(self.labels['interface'])
        sbox.add(self.labels['ip'])
        sbox.add(self.reload_button)
        sbox.add(self.wifi_toggle)

        scroll = self._gtk.ScrolledWindow()
        self.labels['main_box'] = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True)

        if self.sdbus_nm.wifi:
            self.labels['main_box'].pack_start(sbox, False, False, 5)
            scroll.add(self.network_list)
            if self.sdbus_nm.nm.wireless_enabled:
                self.reload_button.show()
                self.sdbus_nm.enable_monitoring(True)
                self.monitor_timer_id = GLib.timeout_add(500, self.sdbus_nm.monitor_connection_status)
        else:
            self._screen.show_popup_message(_("No wireless interface has been found"), level=2)
            self.labels["networkinfo"] = Gtk.Label()
            scroll.add(self.labels["networkinfo"])

        self.labels["main_box"].pack_start(scroll, True, True, 0)
        self.content.add(self.labels["main_box"])
        self.network_list.connect("row-activated", self.handle_wifi_selection)

    def popup_callback(self, msg, level=3):
        self.network_interface_refresh()
        if not self.refresh_status(msg):
            for item in self.network_rows:
                if self.network_rows[item]["label_state"] is not None:
                    self.network_rows[item]["label_state"].set_no_show_all(True)
                    self.network_rows[item]["label_state"].hide()
            self._screen.show_popup_message(msg, level)

    def network_interface_refresh(self):
        if self.interface is not None:
            self.interface = self.sdbus_nm.get_primary_interface()
            self.labels['interface'].set_text(_("Interface") + f': {self.interface}')
            self.labels['ip'].set_text(f"IP: {self.sdbus_nm.get_ip_address()}")

    def handle_wifi_selection(self, list_box, row):
        index = row.get_index()
        logging.info(f"clicked SSID is {self.networks[index]['SSID']}")
        self.connect_network(list_box, self.networks[index]["SSID"])

    def add_network_item(self, ssid, lock, known, signal):
        self.network_rows[ssid] = {}
        self.network_rows[ssid]["row"] = Gtk.ListBoxRow()
        self.network_rows[ssid]["hbox"] = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=15)
        self.network_rows[ssid]["row"].add(self.network_rows[ssid]["hbox"])
        self.network_rows[ssid]["row"].get_style_context().add_class("frame-item")

        self.network_rows[ssid]["signal_ico"] = self._gtk.Image()
        self.network_rows[ssid]["signal_ico"].set_from_pixbuf(self.get_signal_strength_icon(signal))
        self.network_rows[ssid]["signal_ico"].set_margin_start(50)
        self.network_rows[ssid]["hbox"].pack_start(self.network_rows[ssid]["signal_ico"], False, True, 20)

        self.network_rows[ssid]["label_box"] = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=0)
        self.network_rows[ssid]["label_ssid"] = Gtk.Label(label=ssid, xalign=0)
        self.network_rows[ssid]["label_state"] = Gtk.Label(label="", xalign=0)
        self.network_rows[ssid]["label_state"].set_ellipsize(Pango.EllipsizeMode.END)
        self.network_rows[ssid]["label_state"].set_no_show_all(True)
        self.network_rows[ssid]["label_box"].pack_start(self.network_rows[ssid]["label_ssid"], True, False, 0)
        self.network_rows[ssid]["label_box"].pack_start(self.network_rows[ssid]["label_state"], True, False, 0)
        self.network_rows[ssid]["hbox"].pack_start(self.network_rows[ssid]["label_box"], False, True, 10)

        if known:
            self.network_rows[ssid]["delete"] = self._gtk.Button("delete", None, "custom-icon-button", self.bts)
            self.network_rows[ssid]["delete"].connect("clicked", self.remove_confirm_dialog, ssid, ssid)
            self.network_rows[ssid]["delete"].set_hexpand(False)
            self.network_rows[ssid]["delete"].set_halign(Gtk.Align.END)
            self.network_rows[ssid]["hbox"].pack_end(self.network_rows[ssid]["delete"], False, True, 10)

        if "Open" not in lock:
            self.network_rows[ssid]["lock_image"] = self._gtk.Image(
                "lock", self._gtk.content_width * 0.04, self._gtk.content_height * 0.04
            )
            self.network_rows[ssid]["lock_image"].set_hexpand(False)
            self.network_rows[ssid]["lock_image"].set_halign(Gtk.Align.END)
            self.network_rows[ssid]["hbox"].pack_end(self.network_rows[ssid]["lock_image"], False, True, 10)

        self.network_rows[ssid]["hbox"].set_margin_end(50)
        self.network_list.add(self.network_rows[ssid]["row"])

    def move_network_to_front(self, networks, target_name):
        if not target_name or (networks and networks[0].get("SSID") == target_name):
            return networks
        target_index = next((i for i, network in enumerate(networks) if network.get("SSID") == target_name), None)
        if target_index is not None:
            target_network = networks.pop(target_index)
            networks.insert(0, target_network)
        return networks

    def load_networks(self):
        self.connected_ap = self.sdbus_nm.get_connected_ap()
        ap_ssid = None
        if self.connected_ap:
            ap_ssid = self.connected_ap.ssid.decode("utf-8")
        self.networks = self.move_network_to_front(self.sdbus_nm.get_networks(), ap_ssid)
        if self.last_state != self.sdbus_nm.wifi_state:
            self.last_state = self.sdbus_nm.wifi_state
            self.sdbus_nm.wifi_state = -1
            self.sdbus_nm.monitor_connection_status()

        for item in self.networks:
            ssid = item.get("SSID")
            if ssid:
                self.add_network_item(
                    ssid,
                    item.get("security", "unknown"),
                    item.get("known", False),
                    item.get("signal_level", 0),
                )
        self.network_list.show_all()

    def remove_confirm_dialog(self, widget, ssid, bssid):

        label = Gtk.Label(wrap=True, vexpand=True)
        label.set_markup(_("Do you want to forget or disconnect %s?") % ssid)
        buttons = [
            {"name": _("Forget"), "response": Gtk.ResponseType.OK, "style": 'dialog-warning'},
            {"name": _("Cancel"), "response": Gtk.ResponseType.CANCEL, "style": 'dialog-error'},
        ]
        if bssid == self.sdbus_nm.get_connected_bssid():
            buttons.insert(0, {"name": _("Disconnect"), "response": Gtk.ResponseType.APPLY, "style": 'dialog-info'})
        self._gtk.Dialog(_("Remove network"), buttons, label, self.confirm_removal, ssid)

    def confirm_removal(self, dialog, response_id, ssid):
        self._gtk.remove_dialog(dialog)
        if response_id == Gtk.ResponseType.CANCEL:
            return
        if response_id == Gtk.ResponseType.OK:
            logging.info(f"Deleting {ssid}")
            self.sdbus_nm.delete_network(ssid)
        if response_id == Gtk.ResponseType.APPLY:
            logging.info(f"Disconnecting {ssid}")
            self.sdbus_nm.disconnect_network()
        self.delay_reload_networks(100)

    def add_new_network(self, widget, ssid):
        self._screen.remove_keyboard()
        psk = self.labels["network_psk"].get_text()
        identity = self.labels["network_identity"].get_text()
        eap_method = self.get_dropdown_value(self.labels["network_eap_method"])
        phase2 = self.get_dropdown_value(self.labels["network_phase2"])
        logging.debug(f"{phase2=}")
        logging.debug(f"{eap_method=}")
        result = self.sdbus_nm.add_network(ssid, psk, eap_method, identity, phase2)
        if "error" in result:
            self._screen.show_popup_message(result["message"])
            if result["error"] == "psk_invalid":
                return
        else:
            self.connect_network(widget, ssid, showadd=False)
        self.close_add_network()

    def get_dropdown_value(self, dropdown, default=None):
        tree_iter = dropdown.get_active_iter()
        model = dropdown.get_model()
        result = model[tree_iter][0]
        return result if result != "disabled" else None

    def back(self):
        if self.show_add:
            self.close_add_network()
            return True
        return False

    def close_add_network(self):
        if not self.show_add:
            return

        for child in self.content.get_children():
            self.content.remove(child)
        self.content.add(self.labels['main_box'])
        self.content.show()
        for i in ['add_network', 'network_psk', 'network_identity']:
            if i in self.labels:
                del self.labels[i]
        self.show_add = False

    def get_network_index(self, ssid):
        for index, network in enumerate(self.networks):
            if network.get("SSID") == ssid:
                return index
        return -1

    def connect_network(self, widget, ssid, showadd=True):
        index = self.get_network_index(ssid)
        ssid = self.networks[index]["SSID"]
        if showadd and not self.networks[index]["known"]:  # self.sdbus_nm.is_known(ssid):
            sec_type = self.networks[index]["security"]  # self.sdbus_nm.get_security_type(ssid)
            if sec_type == "Open" or "OWE" in sec_type:
                logging.debug("Network is Open do not show psk")
                result = self.sdbus_nm.add_network(ssid, "", "")
                self.sdbus_nm.connect(ssid)
                if "error" in result:
                    self._screen.show_popup_message(result["message"])
            else:
                self.show_add_network(widget, ssid)
            return
        self.sdbus_nm.connect(ssid)
        self.delay_reload_networks(1000)
        if self.monitor_timer_id is None:
            self.sdbus_nm.enable_monitoring(True)
            self.monitor_timer_id = GLib.timeout_add(500, self.sdbus_nm.monitor_connection_status)

    def remove_network_list(self):
        for row in self.network_list.get_children():
            self.network_list.remove(row)
            row.destroy()
        self.network_rows.clear()

    def show_add_network(self, widget, ssid):
        if self.show_add:
            return

        for child in self.content.get_children():
            self.content.remove(child)
        if "add_network" in self.labels:
            del self.labels["add_network"]
        eap_method = Gtk.ComboBoxText(hexpand=True)
        for method in ("peap", "ttls", "pwd", "leap", "md5"):
            eap_method.append(method, method.upper())
        self.labels["network_eap_method"] = eap_method
        eap_method.set_active(0)

        phase2 = Gtk.ComboBoxText(hexpand=True)
        for method in ("mschapv2", "gtc", "pap", "chap", "mschap", "disabled"):
            phase2.append(method, method.upper())
        self.labels["network_phase2"] = phase2
        phase2.set_active(0)

        self.labels["network_identity"] = Gtk.Entry(hexpand=True, no_show_all=True)
        self.labels["network_identity"].connect("focus-in-event", self._screen.show_keyboard)

        self.labels["network_psk"] = Gtk.Entry(hexpand=True)
        self.labels["network_psk"].set_visibility(False)
        self.labels["network_psk"].connect("activate", self.add_new_network, ssid)
        self.labels["network_psk"].connect("focus-in-event", self._screen.show_keyboard)

        save = self._gtk.Button("load", style="color3")
        save.set_hexpand(False)
        save.connect("clicked", self.add_new_network, ssid)

        user_label = Gtk.Label(label=_("User"), hexpand=False, no_show_all=True)
        auth_grid = Gtk.Grid()
        auth_grid.attach(user_label, 0, 0, 1, 1)
        auth_grid.attach(self.labels["network_identity"], 1, 0, 1, 1)
        auth_grid.attach(Gtk.Label(label=_("Password"), hexpand=False), 0, 1, 1, 1)
        auth_grid.attach(self.labels["network_psk"], 1, 1, 1, 1)
        auth_grid.attach(save, 2, 0, 1, 2)

        self.labels["add_network"] = Gtk.Box(
            orientation=Gtk.Orientation.VERTICAL, spacing=5, valign=Gtk.Align.CENTER, hexpand=True, vexpand=True
        )
        self.labels["add_network"].add(Gtk.Label(label=_("Connecting to %s") % ssid))
        self.labels["add_network"].add(auth_grid)
        scroll = self._gtk.ScrolledWindow()
        scroll.add(self.labels["add_network"])
        self.content.add(scroll)
        self.labels["network_psk"].grab_focus_without_selecting()
        self.content.show_all()
        self.show_add = True

    def get_signal_strength_icon(self, signal_level):
        # networkmanager uses percentage not dbm
        if signal_level > 75:
            return self.wifi_signal_icons["excellent"]
        elif signal_level > 60:
            return self.wifi_signal_icons["good"]
        elif signal_level > 30:
            return self.wifi_signal_icons["fair"]
        else:
            return self.wifi_signal_icons["weak"]

    def refresh_status(self, state=None):
        if state:
            self.connected_ap = self.sdbus_nm.get_connected_ap()
            if self.connected_ap:
                ap_ssid = self.connected_ap.ssid.decode("utf-8")
                self.set_connect_state(ap_ssid, state)
                return True
            else:
                return False

    def set_connect_state(self, ssid, state):
        for item in self.network_rows:
            if self.network_rows[item]["label_state"] is not None:
                self.network_rows[item]["label_state"].set_no_show_all(True)
                self.network_rows[item]["label_state"].hide()
        if ssid and state and ssid in self.network_rows:
            if self.network_rows[ssid]["label_state"] is not None:
                self.network_rows[ssid]["label_state"].set_markup(f'<span color="gray" size="small">{state}</span>')
                self.network_rows[ssid]["label_state"].set_no_show_all(False)
        self.network_list.show_all()

    def reload_networks(self, widget=None):
        if self.reload:
            return
        self.reload = True
        self.remove_network_list()
        del self.network_rows
        self.network_rows = {}
        if self.sdbus_nm is not None and self.sdbus_nm.wifi:
            self._gtk.Button_busy(self.reload_button, True)
            if not self.init_status:
                self.sdbus_nm.rescan()
            else:
                self.init_status = False
            self.load_networks()
        if self.sdbus_nm.get_is_connected():
            self.refresh_status(_('Network connected'))

        self._gtk.Button_busy(self.reload_button, False)
        self.reload = False

        if self.delay_reload_timer_id:
            GLib.source_remove(self.delay_reload_timer_id)
            self.delay_reload_timer_id = None
        self.network_interface_refresh()

        return self.sdbus_nm.nm.wireless_enabled

    def delay_reload_networks(self, s):
        self._gtk.Button_busy(self.reload_button, True)
        if not self.delay_reload_timer_id:
            self.delay_reload_timer_id = GLib.timeout_add(s, self.reload_networks)

    def start_refresh_timer(self):
        if not self.sdbus_nm.monitor_connection_status():
            self.sdbus_nm.enable_monitoring(True)
        if self.monitor_timer_id is None:
            self.monitor_timer_id = GLib.timeout_add(600, self.sdbus_nm.monitor_connection_status)
        if self.reload_timer_id is None:
            self.reload_timer_id = GLib.timeout_add_seconds(10, self.reload_networks)
        return False

    def stop_refresh_timer(self):
        if self.monitor_timer_id is not None:
            self.sdbus_nm.enable_monitoring(False)
            GLib.source_remove(self.monitor_timer_id)
            self.monitor_timer_id = None
        if self.reload_timer_id is not None:
            GLib.source_remove(self.reload_timer_id)
            self.reload_timer_id = None

    def activate(self):
        if self.sdbus_nm is None:
            return
        if self.sdbus_nm.wifi:
            if self.sdbus_nm.is_wifi_enabled():
                self.delay_reload_networks(2000)
                self.start_refresh_timer()

    def deactivate(self):
        if self.sdbus_nm is None:
            return
        self.stop_refresh_timer()

    def toggle_wifi(self, switch, gparams):
        enable = switch.get_active()
        logging.info(f"WiFi {enable}")
        self.sdbus_nm.toggle_wifi(enable)
        if enable:
            self.reload_button.set_no_show_all(False)
            self.reload_button.show()
            self.network_list.set_no_show_all(False)
            self.network_list.show()
            self.init_status = True
            self.delay_reload_networks(3000)
            self.start_refresh_timer()
        else:
            self.stop_refresh_timer()
            self.reload_button.set_no_show_all(True)
            self.reload_button.hide()
            self.network_list.set_no_show_all(True)
            self.network_list.hide()
        self.network_interface_refresh()