view libervia/desktop_kivy/core/profile_manager.py @ 494:a4a5565e7026

doc (README): move to README.md to use Markdown, and update content
author Goffi <goffi@goffi.org>
date Mon, 28 Aug 2023 17:09:15 +0200
parents b3cedbee561d
children
line wrap: on
line source

#!/usr/bin/env python3


#Libervia Desktop-Kivy
# Copyright (C) 2016-2021 Jérôme Poisson (goffi@goffi.org)

# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.

# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU Affero General Public License for more details.

# You should have received a copy of the GNU Affero General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.


from libervia.backend.core import log as logging
log = logging.getLogger(__name__)
from .constants import Const as C
from libervia.frontends.quick_frontend.quick_profile_manager import QuickProfileManager
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.togglebutton import ToggleButton
from kivy.uix.screenmanager import ScreenManager, Screen
from kivy.metrics import sp
from kivy import properties
from libervia.desktop_kivy import G


class ProfileItem(ToggleButton):
    ps = properties.ObjectProperty()
    index = properties.NumericProperty(0)


class NewProfileScreen(Screen):
    profile_name = properties.ObjectProperty(None)
    jid = properties.ObjectProperty(None)
    password = properties.ObjectProperty(None)
    error_msg = properties.StringProperty('')

    def __init__(self, pm):
        super(NewProfileScreen, self).__init__(name='new_profile')
        self.pm = pm

    def on_creation_failure(self, failure):
        msg = [l for l in str(failure).split('\n') if l][-1]
        self.error_msg = str(msg)

    def on_creation_success(self, profile):
        self.pm.profiles_screen.reload()
        G.host.bridge.profile_start_session(
            self.password.text, profile,
            callback=lambda __: self._session_started(profile),
            errback=self.on_creation_failure)

    def _session_started(self, profile):
        jid = self.jid.text.strip()
        G.host.bridge.param_set("JabberID", jid, "Connection", -1, profile)
        G.host.bridge.param_set("Password", self.password.text, "Connection", -1, profile)
        self.pm.screen_manager.transition.direction = 'right'
        self.pm.screen_manager.current = 'profiles'

    def doCreate(self):
        name = self.profile_name.text.strip()
        # XXX: we use XMPP password for profile password to simplify
        #      if user want to change profile password, he can do it in preferences
        G.host.bridge.profile_create(
            name, self.password.text, '',
            callback=lambda: self.on_creation_success(name),
            errback=self.on_creation_failure)


class DeleteProfilesScreen(Screen):

    def __init__(self, pm):
        self.pm = pm
        super(DeleteProfilesScreen, self).__init__(name='delete_profiles')

    def do_delete(self):
        """This method will delete *ALL* selected profiles"""
        to_delete = self.pm.get_profiles()
        deleted = [0]

        def delete_inc():
            deleted[0] += 1
            if deleted[0] == len(to_delete):
                self.pm.profiles_screen.reload()
                self.pm.screen_manager.transition.direction = 'right'
                self.pm.screen_manager.current = 'profiles'

        for profile in to_delete:
            log.info("Deleteing profile [{}]".format(profile))
            G.host.bridge.profile_delete_async(
                profile, callback=delete_inc, errback=delete_inc)


class ProfilesScreen(Screen):
    layout = properties.ObjectProperty(None)
    profiles = properties.ListProperty()

    def __init__(self, pm):
        self.pm = pm
        super(ProfilesScreen, self).__init__(name='profiles')
        self.reload()

    def _profiles_list_get_cb(self, profiles):
        profiles.sort()
        self.profiles = profiles
        for idx, profile in enumerate(profiles):
            item = ProfileItem(ps=self, index=idx, text=profile, group='profiles')
            self.layout.add_widget(item)

    def converter(self, row_idx, obj):
        return {'text': obj,
                'size_hint_y': None,
                'height': sp(40)}

    def reload(self):
        """Reload profiles list"""
        self.layout.clear_widgets()
        G.host.bridge.profiles_list_get(callback=self._profiles_list_get_cb)


class ProfileManager(QuickProfileManager, BoxLayout):
    selected = properties.ObjectProperty(None)

    def __init__(self, autoconnect=None):
        QuickProfileManager.__init__(self, G.host, autoconnect)
        BoxLayout.__init__(self, orientation="vertical")
        self.screen_manager = ScreenManager()
        self.profiles_screen = ProfilesScreen(self)
        self.new_profile_screen = NewProfileScreen(self)
        self.delete_profiles_screen = DeleteProfilesScreen(self)
        self.xmlui_screen = Screen(name='xmlui')
        self.screen_manager.add_widget(self.profiles_screen)
        self.screen_manager.add_widget(self.xmlui_screen)
        self.screen_manager.add_widget(self.new_profile_screen)
        self.screen_manager.add_widget(self.delete_profiles_screen)
        self.add_widget(self.screen_manager)

    def close_ui(self, xmlui, reason=None):
        self.screen_manager.transition.direction = 'right'
        self.screen_manager.current = 'profiles'

    def show_ui(self, xmlui):
        xmlui.set_close_cb(self.close_ui)
        if xmlui.type == 'popup':
            xmlui.bind(on_touch_up=lambda obj, value: self.close_ui(xmlui))
        self.xmlui_screen.clear_widgets()
        self.xmlui_screen.add_widget(xmlui)
        self.screen_manager.transition.direction = 'left'
        self.screen_manager.current = 'xmlui'

    def select_profile(self, profile_item):
        if not profile_item.selected:
            return
        def authenticate_cb(data, cb_id, profile):
            if not C.bool(data.pop('validated', C.BOOL_FALSE)):
                # profile didn't validate, we unselect it
                profile_item.state = 'normal'
                self.selected = ''
            else:
                # state may have been modified so we need to be sure it's down
                profile_item.state = 'down'
                self.selected = profile_item
            G.host.action_manager(data, callback=authenticate_cb, ui_show_cb=self.show_ui,
                                 profile=profile)

        G.host.action_launch(C.AUTHENTICATE_PROFILE_ID, callback=authenticate_cb,
                            profile=profile_item.text)

    def get_profiles(self):
        # for now we restrict to a single profile in LiberviaDesktopKivy
        # TODO: handle multi-profiles
        return [self.selected.text] if self.selected else []