Skip to content

Commit

Permalink
ui/about: Use Model-View-Presenter pattern for testability
Browse files Browse the repository at this point in the history
Split model and view, and enable view mocks for unit tests without GDK.
  • Loading branch information
MattHag committed Sep 27, 2024
1 parent 2705f57 commit 65d738f
Show file tree
Hide file tree
Showing 9 changed files with 309 additions and 104 deletions.
102 changes: 0 additions & 102 deletions lib/solaar/ui/about.py

This file was deleted.

Empty file added lib/solaar/ui/about/__init__.py
Empty file.
36 changes: 36 additions & 0 deletions lib/solaar/ui/about/about.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
## Copyright (C) Solaar Contributors
##
## This program is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation; either version 2 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 General Public License for more details.
##
## You should have received a copy of the GNU General Public License along
## with this program; if not, write to the Free Software Foundation, Inc.,
## 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.

from solaar.ui.about.model import AboutModel
from solaar.ui.about.presenter import Presenter
from solaar.ui.about.view import AboutView


def show(model=None, view=None):
"""Opens the About dialog."""
if model is None:
model = AboutModel()
if view is None:
view = AboutView()
presenter = Presenter(model, view)
presenter.run()


if __name__ == "__main__":
from gi.repository import Gtk

show()
Gtk.main()
82 changes: 82 additions & 0 deletions lib/solaar/ui/about/model.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
## Copyright (C) Solaar Contributors
##
## This program is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation; either version 2 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 General Public License for more details.
##
## You should have received a copy of the GNU General Public License along
## with this program; if not, write to the Free Software Foundation, Inc.,
## 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.

from __future__ import annotations

from datetime import datetime
from typing import List
from typing import Tuple

from solaar import __version__
from solaar.i18n import _


def _get_current_year() -> int:
return datetime.now().year


class AboutModel:
def get_version(self) -> str:
return __version__

def get_description(self) -> str:
return _("Manages Logitech receivers,\nkeyboards, mice, and tablets.")

def get_copyright(self) -> str:
return f"© 2012-{_get_current_year()} Daniel Pavel and contributors to the Solaar project"

def get_authors(self) -> List[str]:
return [
"Daniel Pavel http://github.com/pwr",
]

def get_translators(self) -> List[str]:
return [
"gogo (croatian)",
"Papoteur, David Geiger, Damien Lallement (français)",
"Michele Olivo (italiano)",
"Adrian Piotrowicz (polski)",
"Drovetto, JrBenito (Portuguese-BR)",
"Daniel Pavel (română)",
"Daniel Zippert, Emelie Snecker (svensk)",
"Dimitriy Ryazantcev (Russian)",
"El Jinete Sin Cabeza (Español)",
"Ferdina Kusumah (Indonesia)",
]

def get_credit_sections(self) -> List[Tuple[str, List[str]]]:
return [
(_("Additional Programming"), ["Filipe Laíns", "Peter F. Patel-Schneider"]),
(_("GUI design"), ["Julien Gascard", "Daniel Pavel"]),
(
_("Testing"),
[
"Douglas Wagner",
"Julien Gascard",
"Peter Wu http://www.lekensteyn.nl/logitech-unifying.html",
],
),
(
_("Logitech documentation"),
[
"Julien Danjou http://julien.danjou.info/blog/2012/logitech-unifying-upower",
"Nestor Lopez Casado http://drive.google.com/folderview?id=0BxbRzx7vEV7eWmgwazJ3NUFfQ28",
],
),
]

def get_website(self):
return "https://pwr-solaar.github.io/Solaar"
95 changes: 95 additions & 0 deletions lib/solaar/ui/about/presenter.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
## Copyright (C) Solaar Contributors
##
## This program is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation; either version 2 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 General Public License for more details.
##
## You should have received a copy of the GNU General Public License along
## with this program; if not, write to the Free Software Foundation, Inc.,
## 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.

from __future__ import annotations

from typing_extensions import Protocol

from solaar.ui.about.model import AboutModel


class AboutViewProtocol(Protocol):
def init_ui(self) -> None:
...

def update_version_info(self, version: str) -> None:
...

def update_description(self, comments: str) -> None:
...

def update_copyright(self, copyright):
...

def update_authors(self, authors: list[str]) -> None:
...

def update_translators(self, translators: list[str]) -> None:
...

def update_website(self, website):
...

def update_credits(self, credit_sections: list[tuple[str, list[str]]]) -> None:
...

def show(self) -> None:
...


class Presenter:
def __init__(self, model: AboutModel, view: AboutViewProtocol) -> None:
self.model = model
self.view = view

def update_version_info(self) -> None:
version = self.model.get_version()
self.view.update_version_info(version)

def update_credits(self) -> None:
credit_sections = self.model.get_credit_sections()
self.view.update_credits(credit_sections)

def update_description(self) -> None:
comments = self.model.get_description()
self.view.update_description(comments)

def update_copyright(self) -> None:
copyright = self.model.get_copyright()
self.view.update_copyright(copyright)

def update_authors(self) -> None:
authors = self.model.get_authors()
self.view.update_authors(authors)

def update_translators(self) -> None:
translators = self.model.get_translators()
self.view.update_translators(translators)

def update_website(self) -> None:
website = self.model.get_website()
self.view.update_website(website)

def run(self) -> None:
self.view.init_ui()
self.update_version_info()
self.update_description()
self.update_website()
self.update_copyright()
self.update_authors()
self.update_credits()
self.update_translators()
self.view.show()
67 changes: 67 additions & 0 deletions lib/solaar/ui/about/view.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
## Copyright (C) Solaar Contributors
##
## This program is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation; either version 2 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 General Public License for more details.
##
## You should have received a copy of the GNU General Public License along
## with this program; if not, write to the Free Software Foundation, Inc.,
## 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.


from typing import List
from typing import Tuple
from typing import Union

from gi.repository import Gtk

from solaar import NAME


class AboutView:
def __init__(self) -> None:
self.view: Union[Gtk.AboutDialog, None] = None

def init_ui(self) -> None:
self.view = Gtk.AboutDialog()
self.view.set_program_name(NAME)
self.view.set_icon_name(NAME.lower())
self.view.set_license_type(Gtk.License.GPL_2_0)

self.view.connect("response", lambda x, y: self.handle_close(x))

def update_version_info(self, version: str) -> None:
self.view.set_version(version)

def update_description(self, comments: str) -> None:
self.view.set_comments(comments)

def update_copyright(self, copyright_text: str):
self.view.set_copyright(copyright_text)

def update_authors(self, authors: List[str]) -> None:
self.view.set_authors(authors)

def update_credits(self, credit_sections: List[Tuple[str, List[str]]]) -> None:
for section_name, people in credit_sections:
self.view.add_credit_section(section_name, people)

def update_translators(self, translators: List[str]) -> None:
translator_credits = "\n".join(translators)
self.view.set_translator_credits(translator_credits)

def update_website(self, website):
self.view.set_website_label(NAME)
self.view.set_website(website)

def show(self) -> None:
self.view.present()

def handle_close(self, event) -> None:
event.hide()
2 changes: 1 addition & 1 deletion lib/solaar/ui/tray.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ def _create_menu(quit_handler):
menu.append(no_receiver)
menu.append(Gtk.SeparatorMenuItem.new())

menu.append(action.make_image_menu_item(_("About %s") % NAME, "help-about", about.show_window))
menu.append(action.make_image_menu_item(_("About %s") % NAME, "help-about", about.show))
menu.append(action.make_image_menu_item(_("Quit %s") % NAME, "application-exit", quit_handler))

menu.show_all()
Expand Down
Loading

0 comments on commit 65d738f

Please sign in to comment.