Sideband/sbapp/kivymd/tools/patterns/create_project.py
2022-07-07 22:16:10 +02:00

1066 lines
33 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

"""
Script creates a project with the MVC pattern
=============================================
.. versionadded:: 1.0.0
.. seealso::
`MVC pattern <https://en.wikipedia.org/wiki/Modelviewcontroller>`_
.. rubric:: Use a clean architecture for your applications.
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/preview-mvc.png
:align: center
Use a clean architecture for your applications. KivyMD allows you to quickly
create a project template with the MVC pattern. So far, this is the only
pattern that this utility offers. You can also include database support in
your project. At the moment, support for the Firebase database
(the basic implementation of the real time database) and RestDB
(the full implementation) is available.
Project creation
----------------
Template command::
python -m kivymd.tools.patterns.create_project \\
name_pattern \\
path_to_project \\
name_project \\
python_version \\
kivy_version
Example command::
python -m kivymd.tools.patterns.create_project \\
MVC \\
/Users/macbookair/Projects \\
MyMVCProject \\
python3.10 \\
2.1.0
This command will by default create a project with an MVC pattern.
Also, the project will create a virtual environment with Python 3.10,
Kivy version 2.1.0 and KivyMD master version.
.. note::
Please note that the Python version you specified must be installed on your
computer.
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/mvc-base.png
:align: center
Creating a project using a database
-----------------------------------
.. note::
Note that in the following command, you can use one of two database names:
'firebase' or 'restdb'.
Template command::
python -m kivymd.tools.patterns.create_project \\
name_pattern \\
path_to_project \\
name_project \\
python_version \\
kivy_version \\
--name_database
Example command::
python -m kivymd.tools.patterns.create_project \\
MVC \\
/Users/macbookair/Projects \\
MyMVCProject \\
python3.10 \\
2.1.0 \\
--name_database restdb
This command will create a project with an MVC template by default.
The project will also create a virtual environment with Python 3.10,
Kivy version 2.1.0, KivyMD master version and a wrapper for working with
the database restdb.io.
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/mvc-database.png
:align: center
.. code-block:: python
class DataBase:
def __init__(self):
database_url = "https://restdbio-5498.restdb.io"
api_key = "7ce258d66f919d3a891d1166558765f0b4dbd"
.. note::
Please note that `database.py` the shell in the `DataBase` class uses the
`database_url` and `api_key` parameters on the test database (works only in read mode),
so you should use your data for the database.
Preview of the basic MVC template using the RestDB database
-----------------------------------------------------------
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/mvc-restbd-preview.png
:align: center
Create project with hot reload
------------------------------
Template command::
python -m kivymd.tools.patterns.create_project \\
name_pattern \\
path_to_project \\
name_project \\
python_version \\
kivy_version \\
--use_hotreload
Example command::
python -m kivymd.tools.patterns.create_project \\
MVC \\
/Users/macbookair/Projects \\
MyMVCProject \\
python3.10 \\
2.1.0 \\
--use_hotreload yes
After creating the project, open the file `main.py`, there is a lot of useful
information. Also, the necessary information is in other modules of the project
in the form of comments. So do not forget to look at the source files of the
created project.
Others command line arguments
======================
Required Arguments
------------------
- pattern
- the name of the pattern with which the project will be created
- directory
- directory in which the project will be created
- name
- project name
- python_version
- the version of Python (specify as `python3.9` or `python3.8`) with
- which the virtual environment will be created
- kivy_version
- version of Kivy (specify as `2.1.0` or `master`) that will be used in the project
Optional arguments
------------------
- name_screen
- the name of the class which be used when creating the project pattern
- name_database
- provides a basic template for working with the 'firebase' library
- or a complete implementation for working with a database 'restdb.io'
- use_hotreload
- creates a hot reload entry point to the application
-use_localization
- creates application localization files
.. warning:: On Windows, hot reloading of Python files may not work.
But, for example, there is no such problem in macOS. If you fix this,
please report it to the KivyMD community.
"""
import os
import re
import shutil
from typing import Union
from kivy import Logger, platform
from kivymd import path as kivymd_path
from kivymd.tools.argument_parser import ArgumentParserWithHelp
_database_model = '''import multitasking
from Model.base_model import BaseScreenModel
multitasking.set_max_threads(10)
class {name_screen}Model(BaseScreenModel):
"""
Implements the logic of the
:class:`~View.{name_screen}.{module_name}.{name_screen}View` class.
"""
def __init__(self, database):
self.database = database
# Dict:
# 'login': 'KivyMD'
# 'password': '1111'
self.user_data = dict()
self._data_validation_status = None
@property
def data_validation_status(self):
return self._data_validation_status
@data_validation_status.setter
def data_validation_status(self, value):
self._data_validation_status = value
# We notify the View -
# :class:`~View.{name_screen}.{module_name}.{name_screen}View` about the
# changes that have occurred in the data model.
self.notify_observers({notify_name_screen})
@multitasking.task
def check_data(self):
"""
Get data from the database and compares this data with the data entered
by the user. This method is completely asynchronous.
It does not return any value.
"""
data = self.database.get_data_from_collection(self.database.USER_DATA)
data_validation_status = False
if data:
if self.database.name == "RestDB":
data = data[0]
else:
data = list(data.values())[0]
if (
data["login"] == self.user_data["login"]
and data["password"] == self.user_data["password"]
):
data_validation_status = True
self.data_validation_status = data_validation_status
def set_user_data(self, key: str, value: str) -> None:
"""Sets a dictionary of data that the user enters."""
self.user_data[key] = value
def reset_data_validation_status(self) -> None:
self.data_validation_status = None
'''
_without_database_model = '''from Model.base_model import BaseScreenModel
class {name_screen}Model(BaseScreenModel):
"""
Implements the logic of the
:class:`~View.{module_name}.{name_screen}.{name_screen}View` class.
"""'''
_database_controller = '''
{import_module}
class {name_screen}Controller:
"""
The `{name_screen}Controller` class represents a controller implementation.
Coordinates work of the view with the model.
The controller implements the strategy pattern. The controller connects to
the view to control its actions.
"""
def __init__(self, model):
self.model = model # Model.{module_name}.{name_screen}Model object
self.view = {name_view}(
controller=self, model=self.model
)
def set_user_data(self, key: str, value: str) -> None:
"""Called every time the user enters text into the text fields."""
self.model.set_user_data(key, value)
def on_tap_button_login(self) -> None:
"""Called when the `LOGIN` button is pressed."""
self.view.show_dialog_wait()
self.model.check_data()
def reset_data_validation_status(self, *args) -> None:
self.model.reset_data_validation_status()
'''
_without_database_controller = '''
{import_module}
class {name_screen}Controller:
"""
The `{name_screen}Controller` class represents a controller implementation.
Coordinates work of the view with the model.
The controller implements the strategy pattern. The controller connects to
the view to control its actions.
"""
def __init__(self, model):
self.model = model # Model.{module_name}.{name_screen}Model
self.view = {name_view}(controller=self, model=self.model)
def on_tap_button_login(self) -> None:
"""Called when the `LOGIN` button is pressed."""
def set_user_data(self, key, value) -> None:
"""Called every time the user enters text into the text fields."""
'''
_database_view_import = """from typing import Union
from kivy.clock import Clock
from kivymd.uix.dialog import MDDialog
from kivymd.uix.snackbar import Snackbar
"""
_without_database_view_import = """
"""
_database_view_methods = ''' def __init__(self, **kwargs):
super().__init__(**kwargs)
self.dialog = MDDialog()
self.dialog.bind(on_dismiss=self.controller.reset_data_validation_status)
def show_dialog_wait(self) -> None:
"""Displays a wait dialog while the model is processing data."""
self.dialog.auto_dismiss = False
self.dialog.text = "Data validation..."
self.dialog.open()
def show_toast(self, interval: Union[int, float]) -> None:
Snackbar(
text="You have passed the verification successfully!",
snackbar_x="10dp",
snackbar_y="10dp",
size_hint_x=.8,
bg_color=self.theme_cls.primary_color,
).open()
'''
_database_view_model_is_changed_method = """if self.model.data_validation_status:
self.dialog.dismiss()
Clock.schedule_once(self.show_toast, 1)
if self.model.data_validation_status is False:
self.dialog.text = "Wrong data!"
self.dialog.auto_dismiss = True
"""
_firebase_requirements = """kivy==2.1.0
kivymd==1.0.0
multitasking
firebase
firebase-admin
python_jwt
gcloud
sseclient
pycryptodome==3.4.3
requests_toolbelt
"""
_without_firebase_requirements = """kivy==2.1.0
kivymd==1.0.0
"""
_hot_reload_main = '''
"""
Script for managing hot reloading of the project.
For more details see the documentation page -
https://kivymd.readthedocs.io/en/latest/api/kivymd/tools/patterns/create_project/
To run the application in hot boot mode, execute the command in the console:
DEBUG=1 python main.py
"""
import importlib
import os
from kivy import Config
from kivy.uix.screenmanager import ScreenManager
from PIL import ImageGrab
# TODO: You may know an easier way to get the size of a computer display.
resolution = ImageGrab.grab().size
# Change the values of the application window size as you need.
Config.set("graphics", "height", resolution[1])
Config.set("graphics", "width", "400")
from kivy.core.window import Window%s
# Place the application window on the right side of the computer screen.
Window.top = 0
Window.left = resolution[0] - Window.width
from kivymd.tools.hotreload.app import MDApp
%s%s
class %s(MDApp):
KV_FILES = {
os.path.join(
os.getcwd(),
"View",
"%s",
"%s.kv",
),
}%s
def build_app(self) -> ScreenManager:
"""
In this method, you don't need to change anything other than the
application theme.
"""
import View.screens
self.theme_cls.primary_palette = "Orange"
self.manager_screens = ScreenManager()%s%s
Window.bind(on_key_down=self.on_keyboard_down)
importlib.reload(View.screens)
screens = View.screens.screens
for i, name_screen in enumerate(screens.keys()):
model = screens[name_screen]["model"](%s)
controller = screens[name_screen]["controller"](model)
view = controller.get_view()
view.manager_screens = self.manager_screens
view.name = name_screen
self.manager_screens.add_widget(view)
return self.manager_screens
def on_keyboard_down(self, window, keyboard, keycode, text, modifiers) -> None:
"""
The method handles keyboard events.
By default, a forced restart of an application is tied to the
`CTRL+R` key on Windows OS and `COMMAND+R` on Mac OS.
"""
if "meta" in modifiers or "ctrl" in modifiers and text == "r":
self.rebuild()%s%s
%s().run()
# After you finish the project, remove the above code and uncomment the below
# code to test the application normally without hot reloading.
'''
available_patterns = ["MVC"]
available_databases = ["firebase", "restdb"]
def main():
parser = create_argument_parser()
args = parser.parse_args()
pattern_name = args.pattern
project_directory = args.directory
project_name = "".join(args.name.split(" "))
kivy_version = args.kivy_version
python_version = args.python_version
if "3" not in python_version:
parser.error("Python must be at least version 3")
name_screen = "".join(args.name_screen.split(" "))
path_to_project = os.path.join(project_directory, project_name)
name_database = args.name_database
if name_database != "no" and name_database not in available_databases:
parser.error(
f"The database name must be one of the {available_databases} list"
)
use_hotreload = args.use_hotreload
use_localization = args.use_localization
# Check arguments.
if name_screen[-6:] != "Screen":
parser.error(
"Name of the screen must contain the word 'Screen' at the end. "
"\nFor example - '... --name_screen MyFirstScreen'"
)
module_name = chek_camel_case_name_project(name_screen)
if not module_name:
parser.error(
"The name of the screen should be written in camel case style. "
"\nFor example - 'MyFirstScreen'"
)
module_name = "_".join([name.lower() for name in module_name])
if not os.path.exists(
os.path.join(kivymd_path, "tools", "patterns", pattern_name)
):
parser.error(
f"There is no {pattern_name} pattern.\n"
f"Only {available_patterns} template is available."
)
# Call the functions of creating a project.
if not os.path.exists(path_to_project):
shutil.copytree(
os.path.join(kivymd_path, "tools", "patterns", pattern_name),
path_to_project,
)
create_main(
name_database, use_localization, path_to_project, project_name
)
create_model(name_database, name_screen, module_name, path_to_project)
create_controller(
name_database,
use_hotreload,
name_screen,
module_name,
path_to_project,
)
create_view(
name_database,
use_localization,
name_screen,
module_name,
path_to_project,
)
create_requirements(name_database, path_to_project)
os.makedirs(os.path.join(path_to_project, "assets", "images"))
os.mkdir(os.path.join(path_to_project, "assets", "fonts"))
rename_ext_py_tmp_to_py(path_to_project)
move_init(path_to_project, name_screen)
if name_database != "no":
check_databases(name_database, path_to_project)
if use_hotreload == "yes":
create_main_with_hotreload(
path_to_project,
project_name,
name_screen,
module_name,
name_database,
use_localization,
)
with open(
os.path.join(path_to_project, "requirements.txt"),
"a",
encoding="utf-8",
) as requirements:
requirements.write("watchdog")
if use_localization == "yes":
Logger.info("KivyMD: Create localization files...")
create_makefile(
path_to_project, project_name, module_name, name_screen
)
localization_po_file(path_to_project)
create_mofile(path_to_project)
else:
os.remove(os.path.join(path_to_project, "messages.pot"))
os.remove(os.path.join(path_to_project, "libs", "translation.py"))
shutil.rmtree(os.path.join(path_to_project, "data"))
Logger.info(f"KivyMD: Project '{path_to_project}' created")
Logger.info(
f"KivyMD: Create a virtual environment for '{path_to_project}' project..."
)
create_virtual_environment(python_version, path_to_project)
Logger.info(
f"KivyMD: Install requirements for '{path_to_project}' project..."
)
install_requirements(path_to_project, kivy_version, name_database)
else:
parser.error(f"The {path_to_project} project already exists")
def create_main_with_hotreload(
path_to_project: str,
project_name: str,
name_screen: str,
module_name: str,
name_database: str,
use_localization: str,
) -> None:
with open(
os.path.join(path_to_project, "main.py"), encoding="utf-8"
) as main_file:
main_code = ""
for string in main_file.readlines():
main_code += f"# {string}"
with open(
os.path.join(path_to_project, "main.py"), "w", encoding="utf-8"
) as main_file:
main_file.write(f"{_hot_reload_main}\n{main_code}")
replace_in_file(
os.path.join(path_to_project, "main.py"),
(
"\nfrom kivy.properties import StringProperty"
if use_localization == "yes"
else "",
"\nfrom Model.database import DataBase"
if name_database != "no"
else "",
"\nfrom libs.translation import Translation\n"
if use_localization == "yes"
else "",
project_name,
name_screen,
module_name,
'\n lang = StringProperty("en")\n'
if use_localization == "yes"
else "",
"\n self.base = DataBase()\n"
if name_database != "no"
else "",
"\n self.translation = Translation(\n"
' self.lang, "%s", f"{self.directory}/data/locales"'
"\n )" % project_name
if use_localization == "yes"
else "",
"self.database" if name_database != "no" else "",
"\n\n def on_lang(self, instance_app, lang_value: str) -> None:\n"
" self.translation.switch_lang(lang_value)\n"
if use_localization == "yes"
else "",
"\n def switch_lang(self) -> None:\n"
' """Switch lang."""\n\n'
' self.lang = "ru" if self.lang == "en" else "en"'
if use_localization == "yes"
else "",
project_name,
),
)
def create_main(
name_database: str,
use_localization: str,
path_to_project: str,
project_name: str,
) -> None:
replace_in_file(
os.path.join(path_to_project, "main.py_tmp"),
(
"\nfrom kivy.properties import StringProperty"
if use_localization == "yes"
else "",
"\nfrom libs.translation import Translation"
if use_localization == "yes"
else "",
"from Model.database import DataBase\n"
if name_database != "no"
else "",
project_name,
'\n lang = StringProperty("en")\n'
if use_localization == "yes"
else "",
"\n self.translation = Translation(\n"
' self.lang, "%s", f"{self.directory}/data/locales"'
"\n )" % project_name
if use_localization == "yes"
else "",
"self.database = DataBase()\n" if name_database != "no" else "",
"self.database" if name_database != "no" else "",
"\n def on_lang(self, instance_app, lang_value: str) -> None:\n"
" self.translation.switch_lang(lang_value)\n"
if use_localization == "yes"
else "",
"\n def switch_lang(self) -> None:\n"
' """Switch lang."""\n\n'
' self.lang = "ru" if self.lang == "en" else "en"\n'
if use_localization == "yes"
else "",
project_name,
),
)
def create_model(
name_database: str, name_screen: str, module_name: str, path_to_project: str
) -> None:
if name_database != "no":
database_model = _database_model.format(
name_screen=name_screen,
module_name=module_name,
notify_name_screen=f'"{" ".join(module_name.split("_"))}"',
)
replace_in_file(
os.path.join(path_to_project, "Model", "first_screen.py_tmp"),
(database_model),
)
else:
without_database_model = _without_database_model.format(
module_name=module_name, name_screen=name_screen
)
replace_in_file(
os.path.join(path_to_project, "Model", "first_screen.py_tmp"),
(without_database_model),
)
os.remove(
os.path.join(path_to_project, "Model", "database_firebase.py")
)
os.remove(os.path.join(path_to_project, "Model", "database_restdb.py"))
os.rename(
os.path.join(path_to_project, "Model", "first_screen.py_tmp"),
os.path.join(path_to_project, "Model", f"{module_name}.py_tmp"),
)
def create_controller(
name_database: str,
use_hotreload: str,
name_screen: str,
module_name: str,
path_to_project: str,
) -> None:
if name_database != "no":
database_controller = _database_controller
else:
database_controller = _without_database_controller
name_view = (
f"View.{name_screen}.{module_name}.{name_screen}View"
if use_hotreload == "yes"
else f"{name_screen}View"
)
database_controller = database_controller.format(
name_screen=name_screen,
module_name=module_name,
import_module=""
f"import importlib\n\n"
f"import View.{name_screen}.{module_name}\n\n"
f"# We have to manually reload the view module in order to apply the\n"
f"# changes made to the code on a subsequent hot reload.\n"
f"# If you no longer need a hot reload, you can delete this instruction.\n"
f"importlib.reload(View.{name_screen}.{module_name})"
if use_hotreload == "yes"
else f"\nfrom View.{name_screen}.{module_name} import {name_screen}View",
name_view=name_view,
)
replace_in_file(
os.path.join(path_to_project, "Controller", "first_screen.py_tmp"),
(database_controller, name_view),
)
os.rename(
os.path.join(path_to_project, "Controller", "first_screen.py_tmp"),
os.path.join(path_to_project, "Controller", f"{module_name}.py_tmp"),
)
def create_view(
name_database: str,
use_localization: str,
name_screen: str,
module_name: str,
path_to_project: str,
) -> None:
replace_in_file(
os.path.join(path_to_project, "View", "screens.py_tmp"),
(
module_name,
f"{name_screen}Model",
module_name,
f"{name_screen}Controller",
f'"{" ".join(module_name.split("_"))}"',
f"{name_screen}Model",
f"{name_screen}Controller",
),
)
replace_in_file(
os.path.join(path_to_project, "View", "FirstScreen", "first_screen.kv"),
(
f"{name_screen}View",
name_screen,
"app.switch_lang()" if use_localization == "yes" else "x",
'app.translation._("To log in, enter your personal data:")'
if use_localization == "yes"
else '"To log in, enter your personal data:"',
'app.translation._("Login")'
if use_localization == "yes"
else '"Login"',
'app.translation._("Password")'
if use_localization == "yes"
else '"Password"',
'app.translation._("LOGIN")'
if use_localization == "yes"
else '"LOGIN"',
),
)
replace_in_file(
os.path.join(
path_to_project, "View", "FirstScreen", "first_screen.py_tmp"
),
(
_database_view_import
if name_database != "no"
else _without_database_view_import,
f"{name_screen}View",
_database_view_methods if name_database != "no" else "",
_database_view_model_is_changed_method
if name_database != "no"
else "",
),
)
replace_in_file(
os.path.join(path_to_project, "View", "base_screen.py_tmp"),
(
module_name,
f"{name_screen}Model",
module_name,
f"{name_screen}Controller",
module_name,
f"{name_screen}Model",
),
)
os.rename(
os.path.join(path_to_project, "View", "base_screen.py_tmp"),
os.path.join(path_to_project, "View", "base_screen.py"),
)
os.rename(
os.path.join(path_to_project, "View", "FirstScreen", "first_screen.kv"),
os.path.join(
path_to_project, "View", "FirstScreen", f"{module_name}.kv"
),
)
os.rename(
os.path.join(
path_to_project, "View", "FirstScreen", "first_screen.py_tmp"
),
os.path.join(
path_to_project, "View", "FirstScreen", f"{module_name}.py_tmp"
),
)
os.rename(
os.path.join(path_to_project, "View", "FirstScreen"),
os.path.join(path_to_project, "View", name_screen),
)
def create_requirements(name_database: str, path_to_project: str) -> None:
with open(
os.path.join(path_to_project, "requirements.txt"), "w", encoding="utf-8"
) as requirements:
requirements.write(
_firebase_requirements
if name_database != "no"
else _without_firebase_requirements
)
def create_makefile(
path_to_project: str, project_name: str, module_name: str, name_screen: str
) -> None:
path_to_makefile_tmp = os.path.join(path_to_project, "Makefile.tmp")
replace_in_file(
path_to_makefile_tmp,
(
name_screen,
module_name,
name_screen,
module_name,
project_name,
project_name,
),
)
os.rename(path_to_makefile_tmp, os.path.splitext(path_to_makefile_tmp)[0])
os.chdir(path_to_project)
os.system("make po")
def create_mofile(path_to_project: str) -> None:
os.chdir(path_to_project)
os.system("make mo")
def create_virtual_environment(
python_version: str, path_to_project: str
) -> None:
os.system(f"{python_version} -m pip install virtualenv")
os.system(
f"virtualenv -p {python_version} {os.path.join(path_to_project, 'venv')}"
)
def localization_po_file(path_to_project: str) -> None:
path_to_file_po = os.path.join(
path_to_project, "data", "locales", "po", "ru.po"
)
with open(path_to_file_po, "rt", encoding="utf-8") as file_po:
file_po_content = (
file_po.read()
.replace(
'msgid "To log in, enter your personal data:"\nmsgstr ""',
'msgid "To log in, enter your personal data:"\nmsgstr "Для входа введите свои личные данные"',
)
.replace(
'msgid "Login"\nmsgstr ""', 'msgid "Login"\nmsgstr "Логин"'
)
.replace(
'msgid "Password"\nmsgstr ""',
'msgid "Password"\nmsgstr "Пароль"',
)
.replace(
'msgid "LOGIN"\nmsgstr ""', 'msgid "LOGIN"\nmsgstr "ЛОГИН"'
)
)
with open(path_to_file_po, "wt", encoding="utf-8") as file_po:
file_po.write(file_po_content)
def install_requirements(
path_to_project: str, kivy_version: str, name_database: str
) -> None:
python = os.path.join(path_to_project, "venv", "bin", "python3")
if kivy_version == "master":
if platform == "macosx":
os.system(
f"{python} -m pip install 'kivy[base] @ https://github.com/kivy/kivy/archive/master.zip'"
)
else:
os.system(
f"{python} -m pip install https://github.com/kivy/kivy/archive/master.zip"
)
elif kivy_version == "stable":
os.system(f"{python} -m pip install kivy")
else:
os.system(f"{python} -m pip install kivy=={kivy_version}")
os.system(
f"{python} -m pip install https://github.com/kivymd/KivyMD/archive/master.zip"
)
os.system(f"{python} -m pip install watchdog")
if name_database == "firebase":
os.system(
f"{python} -m pip install "
f"multitasking "
f"firebase "
f"firebase-admin "
f"python_jwt "
f"gcloud "
f"sseclient "
f"pycryptodome==3.4.3 "
f"requests_toolbelt "
f"watchdog "
)
os.system(
f"{os.path.join(path_to_project, 'venv', 'bin', 'python3')} -m pip list"
)
def rename_ext_py_tmp_to_py(path_to_project: str) -> None:
for path_to_dir, dirs, files in os.walk(path_to_project):
for name_file in files:
if os.path.splitext(name_file)[1] == ".py_tmp":
os.rename(
os.path.join(path_to_dir, name_file),
os.path.join(
path_to_dir, f"{os.path.splitext(name_file)[0]}.py"
),
)
def move_init(path_to_project: str, name_screen: str) -> None:
path_to_init_file = __file__.replace("create_project", "__init__")
for name_dir in ("Controller", "Model", "Utility", "View"):
shutil.copy(
path_to_init_file,
os.path.join(path_to_project, name_dir, "__init__.py"),
)
shutil.copy(
path_to_init_file,
os.path.join(path_to_project, "View", name_screen, "__init__.py"),
)
path_to_components = os.path.join(
path_to_project, "View", name_screen, "components"
)
os.mkdir(path_to_components)
shutil.copy(
path_to_init_file, os.path.join(path_to_components, "__init__.py")
)
def check_databases(name_database: str, path_to_project: str) -> None:
databases = {"firebase": "restdb", "restdb": "firebase"}
os.remove(
os.path.join(
path_to_project, "Model", f"database_{databases[name_database]}.py"
)
)
os.rename(
os.path.join(path_to_project, "Model", f"database_{name_database}.py"),
os.path.join(path_to_project, "Model", "database.py"),
)
def chek_camel_case_name_project(name_project) -> Union[bool, list]:
result = re.findall("[A-Z][a-z]*", name_project)
if len(result) == 1:
return False
return result
def replace_in_file(path_to_file: str, args) -> None:
with open(path_to_file, "rt", encoding="utf-8") as file_content:
new_file_content = file_content.read() % (args)
with open(path_to_file, "wt", encoding="utf-8") as original_file:
original_file.write(new_file_content)
def create_argument_parser() -> ArgumentParserWithHelp:
parser = ArgumentParserWithHelp(
prog="create_project.py",
allow_abbrev=False,
)
parser.add_argument(
"pattern",
help="the name of the pattern with which the project will be created.",
)
parser.add_argument(
"directory",
help="directory in which the project will be created.",
)
parser.add_argument(
"name",
help="project name.",
)
parser.add_argument(
"python_version",
help="the version of Python (specify as `python3.9` or `python3.8`) "
"with which the virtual environment will be created.",
)
parser.add_argument(
"kivy_version",
help="version of Kivy (specify as `2.1.0` or `master`) that will be "
"used in the project.",
)
parser.add_argument(
"--name_screen",
default="MainScreen",
help="the name of the class wich be used when creating the project pattern.",
)
parser.add_argument(
"--name_database",
default="no",
help="name of the database provider ('firebase' or 'restdb').",
)
parser.add_argument(
"--use_hotreload",
default="no",
help="creates a hot reload entry point to the application.",
)
parser.add_argument(
"--use_localization",
default="no",
help="creates application localization files.",
)
return parser
if __name__ == "__main__":
main()