diff --git a/.gitignore b/.gitignore
index dbd935f..981373b 100644
--- a/.gitignore
+++ b/.gitignore
@@ -26,6 +26,7 @@ share/python-wheels/
.installed.cfg
*.egg
MANIFEST
+.vscode
# PyInstaller
# Usually these files are written by a python script from a template
diff --git a/src/staticwordpress/core/project.py b/src/staticwordpress/core/project.py
index 01448ca..845399a 100644
--- a/src/staticwordpress/core/project.py
+++ b/src/staticwordpress/core/project.py
@@ -86,7 +86,7 @@ def __init__(self, path_: str = None) -> None:
self["wordpress"] = {"user": "", "api-token": ""}
self["github"] = {"token": "", "repository": ""}
- self["redirects"] = REDIRECTS.REDIRECTION
+ self["redirects"] = REDIRECTS.NONE
self["sitemap"] = "sitemap_index.xml"
self["search"] = "search"
self["404"] = "404-error"
@@ -135,15 +135,17 @@ def is_valid(self) -> bool:
)
def has_github(self) -> bool:
- return self["github"]["token"] != "" or self["github"]["repository"] != ""
+ return self["github"]["token"] != "" and self["github"]["repository"] != ""
def has_wordpress(self) -> bool:
- return self["wordpress"]["api-token"] != "" or self["wordpress"]["user"] != ""
+ return self["source"]["type"] == SOURCE.CRAWL or (
+ self["wordpress"]["api-token"] != "" and self["wordpress"]["user"] != ""
+ )
def can_crawl(self) -> bool:
return all(
[
- self["source"]["type"] != SOURCE.CRAWL,
+ self["source"]["type"] == SOURCE.CRAWL,
self["source"]["url"] != "",
self["destination"]["output"] != "",
]
diff --git a/src/staticwordpress/core/utils.py b/src/staticwordpress/core/utils.py
index 24a624b..22f05fe 100644
--- a/src/staticwordpress/core/utils.py
+++ b/src/staticwordpress/core/utils.py
@@ -125,9 +125,9 @@ def rm_dir_tree(dir_path: str = None, delete_root: bool = False) -> None:
return
for _path in dir_path.glob("**/*"):
- if _path.is_file():
+ if _path.is_file() and _path.stem not in [".gitignore", ".project"]:
_path.unlink()
- elif _path.is_dir():
+ elif _path.is_dir() and _path.stem != "._data":
shutil.rmtree(_path, onerror=rmtree_permission_error)
if delete_root:
@@ -222,3 +222,12 @@ def extract_zip_file(zip_file_path: Path, output_location: Path) -> None:
if output_location.is_dir() and zip_file_path.exists():
with ZipFile(zip_file_path, "r") as zf:
zf.extractall(output_location)
+
+
+def is_url_valid(url_: str) -> bool:
+ url_parsed_ = parse.urlparse(url_)
+
+ if all([url_parsed_.scheme, url_parsed_.netloc]):
+ return get_remote_content(url_parsed_, max_retires=1).status_code < 399
+
+ return False
diff --git a/src/staticwordpress/core/workflow.py b/src/staticwordpress/core/workflow.py
index 9e3cde7..b145d66 100644
--- a/src/staticwordpress/core/workflow.py
+++ b/src/staticwordpress/core/workflow.py
@@ -97,10 +97,14 @@ def create_project(
output_folder_: str = "",
custom_404_: str = "",
custom_search_: str = "",
- src_type_: SOURCE = SOURCE.CRAWL,
+ src_type_: SOURCE = SOURCE.ZIP,
host_type_: HOST = HOST.NETLIFY,
) -> None:
self._project.status = PROJECT.NEW
+
+ if src_type_ == SOURCE.ZIP:
+ self._project.redirects = REDIRECTS.REDIRECTION
+
self._project.name = project_name_
self._project.path = project_path_
self._project._404 = custom_404_
diff --git a/src/staticwordpress/gui/config.py b/src/staticwordpress/gui/config.py
index 2d84732..f0aa358 100644
--- a/src/staticwordpress/gui/config.py
+++ b/src/staticwordpress/gui/config.py
@@ -102,9 +102,9 @@ def paintEvent(self, event: QPaintEvent) -> None:
return
-class ConfigWidget(QDialog):
- def __init__(self):
- super(ConfigWidget, self).__init__()
+class ConfigDialog(QDialog):
+ def __init__(self, parent=None):
+ super(ConfigDialog, self).__init__(parent=parent)
self.appConfigurations = QSettings(
CONFIGS["APPLICATION_NAME"], CONFIGS["APPLICATION_NAME"]
)
diff --git a/src/staticwordpress/gui/mainwindow.py b/src/staticwordpress/gui/mainwindow.py
index ddac545..67d3989 100644
--- a/src/staticwordpress/gui/mainwindow.py
+++ b/src/staticwordpress/gui/mainwindow.py
@@ -30,6 +30,7 @@
import sys
import logging
import os
+import shutil
from pathlib import Path
from datetime import date
@@ -39,21 +40,12 @@
# +++++++++++++++++++++++++++++++++++++++++++++++++++++
from PyQt5.QtWidgets import (
- QLabel,
QLineEdit,
QMainWindow,
- QTextEdit,
- QVBoxLayout,
- QHBoxLayout,
- QToolButton,
QAction,
QApplication,
- QDockWidget,
- QWidget,
QFileDialog,
QMessageBox,
- QComboBox,
- QFormLayout,
QProgressBar,
QMenu,
QToolBar,
@@ -69,13 +61,8 @@
from ..core.constants import (
VERISON,
CONFIGS,
- ENUMS_MAP,
SHARE_FOLDER_PATH,
- HOST,
- REDIRECTS,
- PROJECT,
SOURCE,
- USER_AGENT,
)
from ..core.project import Project
from ..core.utils import (
@@ -85,9 +72,10 @@
)
from .workflow import WorkflowGUI
from ..gui.logger import LoggerWidget
-from ..gui.rawtext import RawTextWidget
-from ..gui.config import ConfigWidget
-from ..gui.utils import logging_decorator, GUI_SETTINGS
+from ..gui.rawtext import RawTextDialog
+from ..gui.config import ConfigDialog
+from ..gui.project import ProjectDialog
+from ..gui.utils import GUI_SETTINGS, logging_decorator
# +++++++++++++++++++++++++++++++++++++++++++++++++++++
# IMPLEMENATIONS
@@ -100,223 +88,19 @@ def __init__(self):
self.appConfigurations = QSettings(
CONFIGS["APPLICATION_NAME"], CONFIGS["APPLICATION_NAME"]
)
- self.appConfigurations.setValue("icon_path", SHARE_FOLDER_PATH)
self._project = Project()
self._bg_thread = QThread(parent=self)
self._bg_worker = WorkflowGUI()
- self.docked_widget_project_properties = QDockWidget("Project Properties", self)
- self.docked_widget_project_properties.setMinimumSize(QSize(400, 100))
-
- widget_project_properties = QWidget()
- form_layout_project_properties = QFormLayout()
-
- self.lineedit_project_name = QLineEdit()
- self.lineedit_project_name.setObjectName("name")
- self.lineedit_project_name.textChanged.connect(self.update_windows_title)
- form_layout_project_properties.addRow(
- QLabel("Project Name:"), self.lineedit_project_name
- )
-
- horizontal_Layout_project_scource = QHBoxLayout()
- self.combobox_source_type = QComboBox()
- self.combobox_source_type.setObjectName("source")
- self.combobox_source_type.currentTextChanged.connect(self.update_windows_title)
- self.combobox_source_type.setMinimumWidth(120)
- self.combobox_source_type.addItems([item.value for item in list(SOURCE)])
- horizontal_Layout_project_scource.addWidget(self.combobox_source_type)
- horizontal_Layout_project_scource.addStretch()
- self.combobox_user_agent = QComboBox()
- self.combobox_user_agent.setObjectName("user-agent")
- self.combobox_user_agent.setMinimumWidth(120)
- self.combobox_user_agent.addItems([item.value for item in list(USER_AGENT)])
- self.combobox_user_agent.currentTextChanged.connect(self.update_windows_title)
- horizontal_Layout_project_scource.addWidget(QLabel("User Agent"))
- horizontal_Layout_project_scource.addWidget(self.combobox_user_agent)
- form_layout_project_properties.addRow(
- QLabel("Project Source"), horizontal_Layout_project_scource
- )
- self.lineedit_src_url = QLineEdit()
- self.lineedit_src_url.setObjectName("src-url")
- self.lineedit_src_url.textChanged.connect(self.update_windows_title)
- form_layout_project_properties.addRow(
- QLabel("Source Url"), self.lineedit_src_url
- )
- self.combobox_project_destination = QComboBox()
- self.combobox_project_destination.setObjectName("host")
- self.combobox_project_destination.currentTextChanged.connect(
- self.update_windows_title
- )
-
- self.combobox_project_destination.addItems([item.value for item in list(HOST)])
- form_layout_project_properties.addRow(
- QLabel("Destination Host"), self.combobox_project_destination
- )
-
- self.lineedit_dest_url = QLineEdit()
- self.lineedit_dest_url.setObjectName("dst-url")
- self.lineedit_dest_url.textChanged.connect(self.update_windows_title)
- form_layout_project_properties.addRow(
- QLabel("Destination Url"), self.lineedit_dest_url
- )
-
- horizontal_Layout_output_directory = QHBoxLayout()
- self.lineedit_output = QLineEdit()
- self.lineedit_output.setObjectName("output")
- self.lineedit_output.textChanged.connect(self.update_windows_title)
- self.toolbutton_output_directory = QToolButton()
- self.toolbutton_output_directory.setIcon(
- QIcon(f"{SHARE_FOLDER_PATH}/icons/three-dots.svg")
- )
- horizontal_Layout_output_directory.addWidget(self.lineedit_output)
- horizontal_Layout_output_directory.addWidget(self.toolbutton_output_directory)
- self.toolbutton_output_directory.clicked.connect(self.get_output_directory)
- form_layout_project_properties.addRow(
- QLabel("Output Directory"), horizontal_Layout_output_directory
- )
-
- vertical_layout_additional_properties = QVBoxLayout()
- vertical_layout_additional_properties.addLayout(form_layout_project_properties)
- widget_project_properties.setLayout(vertical_layout_additional_properties)
- self.docked_widget_project_properties.setWidget(widget_project_properties)
- self.docked_widget_project_properties.setFeatures(
- QDockWidget.NoDockWidgetFeatures
- )
-
- self.docked_widget_project_properties.setMaximumHeight(200)
- self.addDockWidget(Qt.LeftDockWidgetArea, self.docked_widget_project_properties)
-
- # =============================
- # Github Properties dock
- # =============================
- self.docked_widget_github_properties = QDockWidget("Github Setttings", self)
- self.docked_widget_github_properties.setMaximumHeight(100)
-
- widget_github_properties = QWidget()
- form_layout_github_properties = QFormLayout()
-
- self.lineedit_gh_repo = QLineEdit()
- self.lineedit_gh_repo.setObjectName("github-repository")
- self.lineedit_gh_repo.textChanged.connect(self.update_windows_title)
- form_layout_github_properties.addRow(
- QLabel("Repository Name"), self.lineedit_gh_repo
- )
- self.lineedit_gh_token = QLineEdit()
- self.lineedit_gh_token.setObjectName("github-token")
- self.lineedit_gh_token.textChanged.connect(self.update_windows_title)
- form_layout_github_properties.addRow(
- QLabel("GitHub Token"), self.lineedit_gh_token
- )
-
- vertical_layout_github_properties = QVBoxLayout()
- vertical_layout_github_properties.addLayout(form_layout_github_properties)
- widget_github_properties.setLayout(vertical_layout_github_properties)
- self.docked_widget_github_properties.setWidget(widget_github_properties)
- self.docked_widget_github_properties.setFeatures(
- QDockWidget.NoDockWidgetFeatures
- )
- self.addDockWidget(Qt.LeftDockWidgetArea, self.docked_widget_github_properties)
-
- # =============================
- # Crawl Properties dock
- # =============================
- self.docked_widget_crawl_properties = QDockWidget("Crawl Settings", self)
- self.docked_widget_crawl_properties.setMinimumSize(QSize(400, 100))
-
- widget_crawl_properties = QWidget()
- form_layout_crawl_properties = QFormLayout()
-
- horizontal_Layout_wordpress = QHBoxLayout()
- self.lineedit_wp_user = QLineEdit()
- self.lineedit_wp_user.setMaximumWidth(80)
- self.lineedit_wp_user.setObjectName("wordpress-user")
- self.lineedit_wp_user.textChanged.connect(self.update_windows_title)
- horizontal_Layout_wordpress.addWidget(self.lineedit_wp_user)
- self.lineedit_wp_api_token = QLineEdit()
- self.lineedit_wp_api_token.setObjectName("wordpress-api-token")
- self.lineedit_wp_api_token.textChanged.connect(self.update_windows_title)
- horizontal_Layout_wordpress.addWidget(QLabel("API Token"))
- horizontal_Layout_wordpress.addWidget(self.lineedit_wp_api_token)
- form_layout_crawl_properties.addRow(
- QLabel("WordPress User"), horizontal_Layout_wordpress
- )
-
- horizontal_Layout_redirects = QHBoxLayout()
- self.combobox_redirects = QComboBox()
- self.combobox_redirects.setObjectName("redirects")
- self.combobox_redirects.currentTextChanged.connect(self.update_windows_title)
- self.combobox_redirects.addItems([item.value for item in list(REDIRECTS)])
- horizontal_Layout_redirects.addWidget(self.combobox_redirects)
- form_layout_crawl_properties.addRow(
- QLabel("Redirects Source"), horizontal_Layout_redirects
- )
-
- horizontal_Layout_sitemap = QHBoxLayout()
- self.lineedit_sitemap = QLineEdit()
- self.lineedit_sitemap.setObjectName("sitemap")
- self.lineedit_sitemap.textChanged.connect(self.update_windows_title)
- self.toolbutton_output_sitemap = QToolButton()
- self.toolbutton_output_sitemap.setIcon(
- QIcon(f"{SHARE_FOLDER_PATH}/icons/search.svg")
- )
- horizontal_Layout_sitemap.addWidget(self.lineedit_sitemap)
- horizontal_Layout_sitemap.addWidget(self.toolbutton_output_sitemap)
- self.toolbutton_output_sitemap.clicked.connect(self.get_sitemap_location)
- form_layout_crawl_properties.addRow(
- QLabel("Sitemap Location"), horizontal_Layout_sitemap
- )
-
- horizontal_Layout_search_404 = QHBoxLayout()
- self.lineedit_search = QLineEdit()
- self.lineedit_search.setObjectName("search")
- self.lineedit_search.textChanged.connect(self.update_windows_title)
- horizontal_Layout_search_404.addWidget(self.lineedit_search)
- horizontal_Layout_search_404.addWidget(QLabel("404 Page"))
- self.lineedit_404_page = QLineEdit()
- self.lineedit_404_page.setObjectName("404-error")
- self.lineedit_404_page.textChanged.connect(self.update_windows_title)
- horizontal_Layout_search_404.addWidget(self.lineedit_404_page)
- form_layout_crawl_properties.addRow(
- QLabel("Search Page"), horizontal_Layout_search_404
- )
-
- self.lineedit_delay = QLineEdit()
- self.lineedit_delay.setObjectName("delay")
- self.lineedit_delay.textChanged.connect(self.update_windows_title)
- form_layout_crawl_properties.addRow(
- QLabel("Delay (Seconds)"), self.lineedit_delay
- )
-
- form_layout_crawl_properties.addRow(QLabel("Additional Files"))
- self.textedit_additional = QTextEdit()
- self.textedit_additional.setObjectName("additional")
- self.textedit_additional.textChanged.connect(self.update_windows_title)
- form_layout_crawl_properties.addRow(self.textedit_additional)
-
- form_layout_crawl_properties.addRow(QLabel("Exclude Patterns"))
- self.textedit_exclude = QTextEdit()
- self.textedit_exclude.setObjectName("exclude")
- self.textedit_exclude.textChanged.connect(self.update_windows_title)
- form_layout_crawl_properties.addRow(self.textedit_exclude)
-
- vertical_layout_wp_properties = QVBoxLayout()
- vertical_layout_wp_properties.addLayout(form_layout_crawl_properties)
- widget_crawl_properties.setLayout(vertical_layout_wp_properties)
- self.docked_widget_crawl_properties.setWidget(widget_crawl_properties)
- self.docked_widget_crawl_properties.setFeatures(
- QDockWidget.NoDockWidgetFeatures
- )
- self.addDockWidget(Qt.LeftDockWidgetArea, self.docked_widget_crawl_properties)
-
self.text_edit_logging = LoggerWidget(self)
self.text_edit_logging.setFormatter(
logging.Formatter("%(asctime)s - %(levelname)s - %(message)s")
)
logging.getLogger().addHandler(self.text_edit_logging)
logging.getLogger().setLevel(logging.INFO)
- self.setCentralWidget(self.text_edit_logging.plainTextEdit)
+ self.setCentralWidget(self.text_edit_logging.plainTextEdit)
self.statusBar().showMessage(f"{CONFIGS['APPLICATION_NAME']} is Ready")
self.progressBar = QProgressBar()
self.progressBar.setAlignment(Qt.AlignCenter)
@@ -415,30 +199,28 @@ def inner(self):
return inner
- @is_new_project
- @logging_decorator
- def get_output_directory(self):
- """"""
- output_directory = QFileDialog.getExistingDirectory(
- self,
- "Select Output Directory",
- self.appConfigurations.value("output-directory"),
- )
- if output_directory:
- self.lineedit_output.setText(output_directory)
- self.appConfigurations.setValue("output-directory", output_directory)
-
@is_project_open
@logging_decorator
def clean_output_directory(self):
"""Clean Output Directory"""
- reply = QMessageBox.question(
- self,
- "Clean Output Folder Content",
- f"Existing content in Output folder will be delete?\n {self._project.output}",
- QMessageBox.Yes | QMessageBox.No,
+
+ msgBox = QMessageBox(parent=self)
+ msgBox.setWindowTitle("Clean Output Folder Content")
+ msgBox.setText(
+ f"Existing content in Output folder will be delete?
{self._project.output}",
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
)
- if reply == QMessageBox.Yes:
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
+ )
+ msgBox.setWindowIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg"))
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.exec_()
+
+ if msgBox.clickedButton().text() == "&OK":
rm_dir_tree(self._project.output)
logging.info(
f"Content of output folder at {self._project.output} are deleted"
@@ -463,27 +245,23 @@ def get_sitemap_location(self):
def update_sitemap_location(self, sitemap_location):
self._project.sitemap = sitemap_location
logging.info(f"Found Sitemap location: {sitemap_location}")
- self.update_properties_widgets()
+ self.update_widgets()
@is_new_project
@logging_decorator
def extract_url_from_raw_text(self):
- rtp = RawTextWidget(
- src_url=self._project.src_url, dest_url=self._project.dst_url
+ rtp = RawTextDialog(
+ parent=self, src_url=self._project.src_url, dest_url=self._project.dst_url
)
if rtp.exec_():
raw_text = rtp.textedit_raw_text_with_links.toPlainText()
- current_additional_urls = self.textedit_additional.toPlainText().split("\n")
-
if raw_text:
new_additional_links = extract_urls_from_raw_text(
raw_text, rtp.lineedit_dest_url.text(), rtp.linedit_src_url.text()
)
logging.info(f" {len(new_additional_links)} Additional Urls Found")
- current_additional_urls += new_additional_links
- self.textedit_additional.setText(
- "\n".join(set(current_additional_urls))
- )
+ self._project.additional += new_additional_links
+ self._project.save()
@is_new_project
@logging_decorator
@@ -494,14 +272,24 @@ def clear_cache(self):
def closeEvent(self, event):
""" """
- reply = QMessageBox.question(
- self,
- "Exiting static-wordpress",
- "Do you really want to exit?.\nAny unsaved changes will be lost!",
- QMessageBox.Yes | QMessageBox.No,
+ msgBox = QMessageBox(parent=self)
+ msgBox.setWindowTitle(f"Exiting {CONFIGS['APPLICATION_NAME']}")
+ msgBox.setIcon(QMessageBox.Question)
+ msgBox.setText(
+ "Do you really want to exit?.
Any unsaved changes will be lost!",
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
)
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
+ )
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.setWindowIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg"))
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.exec_()
- if reply == QMessageBox.Yes:
+ if msgBox.clickedButton().text() == "&OK":
if self._bg_thread.isRunning():
self._bg_thread.quit()
del self._bg_thread
@@ -547,73 +335,114 @@ def help(self):
@logging_decorator
def show_configs(self):
"""Interface with System Configurations"""
- w = ConfigWidget()
+ w = ConfigDialog(parent=self)
if w.exec_():
logging.info("Saved/Updated Default Configurations")
def about(self):
""" """
- msgBox = QMessageBox()
+ msgBox = QMessageBox(parent=self)
msgBox.setText(
f"Copyright {date.today().year} - SERP Wings"
f"
{CONFIGS['APPLICATION_NAME']} Version - {VERISON}"
"
This work is an opensource project under
GNU General Public License v3 or later (GPLv3+)"
f"
More Information at {CONFIGS['ORGANIZATION_NAME']}"
)
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
msgBox.setWindowIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg"))
msgBox.setTextFormat(Qt.RichText)
msgBox.setWindowTitle("About Us")
- msgBox.setStandardButtons(QMessageBox.Ok)
msgBox.exec()
@logging_decorator
- def create_project(self):
+ def new_project(self):
"""Closing current project will automatically start a new project."""
self.close_project()
- self._project.create()
- self.update_properties_widgets()
+
+ pdialog = ProjectDialog(self, self._project, title_="New Project")
+ if pdialog.exec_():
+ self._project = pdialog._project
+ self.appConfigurations.setValue("last-project", self._project.output)
+
+ if Path(self._project.output).is_dir():
+ self._project.path.parent.mkdir(parents=True, exist_ok=True)
+ src = Path(f"{SHARE_FOLDER_PATH}/_ignore")
+ dst = Path(f"{self._project.output}/.gitignore")
+ if src.exists():
+ shutil.copyfile(src, dst)
+ else:
+ return
+
+ self.update_widgets()
+ logging.info("Saved/Update Project")
+ self._project.save()
@logging_decorator
def open_project(self):
"""Opening static-wordpress Project File"""
self.close_project()
if not self._project.is_open():
- options = QFileDialog.Options()
- project_path, _ = QFileDialog.getOpenFileName(
+ project_folder = QFileDialog.getExistingDirectory(
self,
- "Select Static-WordPress Project File",
- self.appConfigurations.value("last-project"),
- "JSON Files (*.json)",
- options=options,
+ "Select Static-WordPress Project Directory",
+ str(self.appConfigurations.value("last-project")),
+ QFileDialog.ShowDirsOnly,
)
- if project_path:
- self._project.open(Path(project_path))
+ project_path = Path(f"{project_folder}/._data/.project.json")
+ if project_path.exists():
+ self._project.open(project_path)
if self._project.is_open():
+ pdialog = ProjectDialog(
+ self, self._project, title_="Project Properties"
+ )
+
+ if pdialog.exec_():
+ self._project = pdialog._project
+
if not self._project.is_older_version():
logging.warning(
f"Your Project was saved with an older version : {self._project.version}."
)
+
logging.info(f"Open Project {self._project.path} Successfully")
- self.appConfigurations.setValue("last-project", project_path)
+ self.appConfigurations.setValue("last-project", project_folder)
else:
- msgBox = QMessageBox()
- msgBox.setText(f"Project cannot be opened." f"Please try again.")
+ msgBox = QMessageBox(parent=self)
+ msgBox.setText(
+ f"Project cannot be opened or selected path invalid."
+ f"
Please try again with project folder."
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
msgBox.setWindowIcon(
QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg")
)
msgBox.setTextFormat(Qt.RichText)
msgBox.setWindowTitle("Open Project")
- msgBox.setStandardButtons(QMessageBox.Ok)
msgBox.exec()
logging.info(
"No New Project Opened. Unsaved project properties will be lost."
)
- self.update_windows_title()
- self.update_properties_widgets()
+ self.update_widgets()
+ self._project.save()
+
+ @logging_decorator
+ def show_project(self):
+ """showing static-wordpress Project File"""
+ if self._project.is_open():
+ pdialog = ProjectDialog(self, self._project, title_="Current Project")
+ if pdialog.exec_():
+ self._project = pdialog._project
+
+ self._project.save()
+ self.update_widgets()
@is_project_open
@logging_decorator
@@ -621,109 +450,51 @@ def close_project(self):
"""Assign new project and old properties will be lost.
Default is assigned as CLOSED project
"""
- reply = QMessageBox.question(
- self,
- "Close Existing Project",
- "Are you sure to close current project and open new one?.\n All existing project properties will be lost!",
- QMessageBox.Yes | QMessageBox.No,
+ msgBox = QMessageBox(parent=self)
+ msgBox.setWindowTitle("Close Existing Project")
+ msgBox.setText(
+ "Are you sure to close current project and open new one?.
All existing project properties will be lost!",
)
- if reply == QMessageBox.Yes:
- self._project = Project()
- self.update_properties_widgets()
- verifications = {
- "name": None,
- "src-url": None,
- "wordpress-user": None,
- "wordpress-api-token": None,
- "sitemap": None,
- "github-token": None,
- "github-repository": None,
- }
- self.update_expert_mode_widgets(verifications)
-
- @is_project_open
- @logging_decorator
- def save_project(self):
- """Saving Current static-wordpress Project"""
- if self.lineedit_project_name.text():
- new_project_path = self.appConfigurations.value("last-project")
- if self._project.is_new():
- options = QFileDialog.Options()
- new_project_path, _ = QFileDialog.getSaveFileName(
- self,
- "Select StaticWordPress Project File",
- self.appConfigurations.value("last-project"),
- "JSON Files (*.json)",
- options=options,
- )
- if new_project_path:
- self._project.path = Path(new_project_path)
- else:
- return
-
- self.appConfigurations.setValue("last-project", new_project_path)
- self._project.name = self.lineedit_project_name.text()
- self._project.path = Path(new_project_path)
- self._project.src_url = self.lineedit_src_url.text()
- self._project.sitemap = self.lineedit_sitemap.text()
- self._project.wp_user = self.lineedit_wp_user.text()
- self._project.wp_api_token = self.lineedit_wp_api_token.text()
- self._project.search = self.lineedit_search.text()
- self._project._404 = self.lineedit_404_page.text()
- self._project.delay = float(self.lineedit_delay.text())
- self._project.redirects = REDIRECTS[self.combobox_redirects.currentText()]
- self._project.src_type = SOURCE[self.combobox_source_type.currentText()]
- self._project.user_agent = USER_AGENT[
- self.combobox_user_agent.currentText()
- ]
- self._project.host = HOST[self.combobox_project_destination.currentText()]
- self._project.output = Path(self.lineedit_output.text())
- self._project.dst_url = self.lineedit_dest_url.text()
- self._project.gh_token = self.lineedit_gh_token.text()
- self._project.gh_repo = self.lineedit_gh_repo.text()
- self._project.additional = self.textedit_additional.toPlainText().split(
- "\n"
- )
- self._project.exclude = self.textedit_exclude.toPlainText().split("\n")
- if not self._project.is_older_version():
- logging.warning(
- f"Your Project will be saved with new version number : {VERISON}."
- )
- self._project.version = VERISON
-
- # save project
- self._project.save()
- if self._project.is_saved():
- logging.info(f"Project Saved Successfully at {self._project.path}")
- self.update_properties_widgets()
- self.update_windows_title()
-
- def check_project(self) -> None:
- if self._bg_thread.isRunning():
- self._bg_thread.quit()
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
+ )
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.setWindowIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg"))
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.exec_()
- self._bg_thread = QThread(parent=self)
- self._bg_worker = WorkflowGUI()
- self._bg_worker.set_project(project_=self._project)
- self._bg_worker.moveToThread(self._bg_thread)
- self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalVerification.connect(self.update_expert_mode_widgets)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
- self._bg_thread.started.connect(self._bg_worker.verify_project)
- self._bg_thread.start()
+ if msgBox.clickedButton().text() == "&OK":
+ self._project = Project()
+ self.update_widgets()
@is_project_open
def start_batch_process(self):
"""Start Crawling"""
+
if not self._project.output.exists():
- reply = QMessageBox.question(
- self,
- f"Output Folder",
- f"Following Output Folder doesnt not exit?.\n{self._project.output}\nDo You want to create it now?",
- QMessageBox.Yes | QMessageBox.No,
+ msgBox = QMessageBox(parent=self)
+ msgBox.setIcon(QMessageBox.Question)
+ msgBox.setWindowTitle("Output Folder")
+ msgBox.setText(
+ f"Following Output Folder doesnt not exit?.
{self._project.output}
Do You want to create it now?",
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
)
+ msgBox.setWindowIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg")
+ )
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.exec_()
- if reply == QMessageBox.Yes:
+ if msgBox.clickedButton().text() == "&OK":
os.mkdir(self._project.output)
else:
return
@@ -736,35 +507,59 @@ def start_batch_process(self):
if self._project.src_type == SOURCE.ZIP:
if not self._bg_worker._work_flow.verify_simply_static():
- reply = QMessageBox.question(
- self,
- f"ZIP File Missing",
- f"ZIP File not found. Please check your project configurations?",
- QMessageBox.Yes,
+ msgBox = QMessageBox(parent=self)
+ msgBox.setWindowTitle("ZIP File Missing")
+ msgBox.setIcon(QMessageBox.Question)
+ msgBox.setText(
+ "ZIP File not found. Please check your project configurations?",
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
+ )
+ msgBox.setWindowIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg")
)
- if reply == QMessageBox.Yes:
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.exec_()
+
+ if msgBox.clickedButton().text() == "&OK":
return
self._bg_thread = QThread(parent=self)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.batch_processing)
self._bg_thread.start()
@is_project_open
def stop_process(self) -> None:
if self._bg_worker.is_running():
- reply = QMessageBox.question(
- self,
- "Stop Crawling Process",
- "Do you really want to Stop Crawling Thrad?",
- QMessageBox.Yes | QMessageBox.No,
+ msgBox = QMessageBox(parent=self)
+ msgBox.setWindowTitle("Stop Crawling Process")
+ msgBox.setText(
+ "Do you really want to Stop Crawling Thread?",
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
)
+ msgBox.setWindowIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg")
+ )
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.exec_()
- if reply == QMessageBox.Yes:
+ if msgBox.clickedButton().text() == "&OK":
self._bg_worker.stop_calcualations()
- self.update_statusbar_widgets("Stoping Processing", 100)
+ self.update_statusbar("Stoping Processing", 100)
@is_project_open
def crawl_website(self) -> None:
@@ -776,7 +571,7 @@ def crawl_website(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.pre_processing)
self._bg_thread.start()
@@ -790,7 +585,7 @@ def crawl_additional_files(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.crawl_additional_files)
self._bg_thread.start()
@@ -804,7 +599,7 @@ def create_search_index(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.add_search)
self._bg_thread.start()
@@ -818,7 +613,7 @@ def create_404_page(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.add_404_page)
self._bg_thread.start()
@@ -832,7 +627,7 @@ def create_redirects(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.add_redirects)
self._bg_thread.start()
@@ -846,7 +641,7 @@ def create_robots_txt(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.add_robots_txt)
self._bg_thread.start()
@@ -861,21 +656,31 @@ def create_github_repositoy(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.create_github_repositoy)
self._bg_thread.start()
@is_project_open
def delete_github_repository(self) -> None:
""""""
- reply = QMessageBox.question(
- self,
- "Deleting Repository on GitHub",
- f"Do you really want to delete {self._project.gh_repo} on GitHub?\nThis deletion is not reversible.",
- QMessageBox.Yes | QMessageBox.No,
+
+ msgBox = QMessageBox(parent=self)
+ msgBox.setWindowTitle("Deleting Repository on GitHub")
+ msgBox.setText(
+ f"Do you really want to delete {self._project.gh_repo} on GitHub?
This deletion is not reversible.",
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
)
+ msgBox.addButton(QMessageBox.Cancel).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg")
+ )
+ msgBox.setDefaultButton(QMessageBox.Ok)
+ msgBox.setWindowIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg"))
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.exec_()
- if reply == QMessageBox.Yes:
+ if msgBox.clickedButton().text() == "&OK":
if self._bg_thread.isRunning():
self._bg_thread.quit()
@@ -884,7 +689,7 @@ def delete_github_repository(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.delete_github_repositoy)
self._bg_thread.start()
@@ -899,7 +704,7 @@ def initialize_repository(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.init_git_repositoy)
self._bg_thread.start()
@@ -914,7 +719,7 @@ def commit_repository(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.commit_git_repositoy)
self._bg_thread.start()
@@ -929,11 +734,11 @@ def publish_repository(self) -> None:
self._bg_worker.set_project(project_=self._project)
self._bg_worker.moveToThread(self._bg_thread)
self._bg_thread.finished.connect(self._bg_worker.deleteLater)
- self._bg_worker.signalProgress.connect(self.update_statusbar_widgets)
+ self._bg_worker.signalProgress.connect(self.update_statusbar)
self._bg_thread.started.connect(self._bg_worker.publish_github_repositoy)
self._bg_thread.start()
- def update_statusbar_widgets(self, message_, percent_) -> None:
+ def update_statusbar(self, message_, percent_) -> None:
if percent_ >= 0:
self.progressBar.setValue(percent_)
self.statusBar().showMessage(message_)
@@ -943,29 +748,10 @@ def update_statusbar_widgets(self, message_, percent_) -> None:
if percent_ >= 100:
self.progressBar.setFormat(message_)
- def update_properties_widgets(self) -> None:
- self.lineedit_project_name.setText(self._project.name)
- self.lineedit_src_url.setText(self._project.src_url)
- self.lineedit_sitemap.setText(self._project.sitemap)
- self.lineedit_search.setText(self._project.search)
- self.lineedit_404_page.setText(self._project._404)
- self.lineedit_delay.setText(f"{self._project.delay}")
- self.combobox_source_type.setCurrentText(self._project.src_type.value)
- self.combobox_user_agent.setCurrentText(self._project.user_agent.value)
- self.lineedit_output.setText(str(self._project.output))
- self.lineedit_dest_url.setText(self._project.dst_url)
- self.lineedit_wp_user.setText(self._project.wp_user)
- self.lineedit_wp_api_token.setText(self._project.wp_api_token)
- self.lineedit_gh_token.setText(self._project.gh_token)
- self.lineedit_gh_repo.setText(self._project.gh_repo)
- self.textedit_additional.setText("\n".join(self._project.additional))
- self.textedit_exclude.setText("\n".join(self._project.exclude))
- self.combobox_redirects.setCurrentText(self._project.redirects.value)
- self.combobox_redirects.setEnabled(True)
-
+ def update_widgets(self) -> None:
self.findChild(QMenu, "menu_github").setEnabled(self._project.has_github())
self.findChild(QMenu, "menu_wordpress").setEnabled(
- self._project.has_wordpress()
+ self._project.has_wordpress() or self._project.can_crawl()
)
self.findChild(QToolBar, "toolbar_github").setEnabled(
self._project.has_github()
@@ -990,71 +776,9 @@ def update_properties_widgets(self) -> None:
self.findChild(QAction, "action_edit_expert_mode").isChecked()
)
- def update_expert_mode_widgets(self, verifications) -> None:
- for key, value in verifications.items():
- if value is not None:
- bg_color = (
- CONFIGS["COLOR"]["SUCCESS"] if value else CONFIGS["COLOR"]["ERROR"]
- )
- else:
- bg_color = value
-
- self.findChild(QLineEdit, key).setStyleSheet(
- f"background-color: {bg_color}"
- )
-
- def update_windows_title(self) -> None:
- sender = self.sender()
- if (
- type(sender) in [QLineEdit, QComboBox, QTextEdit]
- and self._project.is_open()
- ):
- gui_value = sender.property("text")
-
- if type(sender) == QLineEdit:
- gui_value = sender.property("text")
- if sender.objectName() == "output":
- gui_value = Path(sender.property("text"))
- elif sender.objectName() == "delay":
- try:
- gui_value = float(sender.property("text"))
- except: # mostly value error e.g. 0.^ as input
- pass
-
- elif type(sender) == QComboBox:
- gui_value = ENUMS_MAP[sender.objectName()][
- sender.property("currentText")
- ]
- elif type(sender) == QTextEdit:
- gui_value = [url for url in sender.toPlainText().split("\n") if url]
-
- project_value = self._project.get(sender.objectName())
- project_functions_dict = {
- "github-repository": self._project.gh_repo,
- "github-token": self._project.gh_token,
- "wordpress-user": self._project.wp_user,
- "wordpress-api-token": self._project.wp_api_token,
- "src-url": self._project.src_url,
- "source": self._project.src_type,
- "output": self._project.output,
- "dst-url": self._project.dst_url,
- "404-error": self._project._404,
- }
-
- if sender.objectName() in project_functions_dict:
- project_value = project_functions_dict[sender.objectName()]
-
- if gui_value != project_value and not self._project.is_new():
- self._project.status = PROJECT.UPDATE
-
- status_string = (
- f"{'' if self._project.is_saved() else '*'} {self._project.name}"
- )
new_window_title = (
- f"{status_string} - {CONFIGS['APPLICATION_NAME']} Version - {VERISON}"
+ f"{self._project.name} - {CONFIGS['APPLICATION_NAME']} Version - {VERISON}"
)
- if self._project.status == PROJECT.NOT_FOUND:
- new_window_title = f"{CONFIGS['APPLICATION_NAME']} Version - {VERISON}"
self.setWindowTitle(new_window_title)
diff --git a/src/staticwordpress/gui/project.py b/src/staticwordpress/gui/project.py
new file mode 100644
index 0000000..304ae2a
--- /dev/null
+++ b/src/staticwordpress/gui/project.py
@@ -0,0 +1,468 @@
+# -*- coding: utf-8 -*-
+
+"""
+STATIC WORDPRESS: WordPress as Static Site Generator
+A Python Package for Converting WordPress Installation to a Static Website
+https://github.com/serpwings/staticwordpress
+
+ src\staticwordpress\gui\project.py
+
+ Copyright (C) 2020-2023 Faisal Shahzad
+
+
+The contents of this file are subject to version 3 of the
+GNU General Public License (GPL-3.0). You may not use this file except in
+compliance with the License. You may obtain a copy of the License at
+https://www.gnu.org/licenses/gpl-3.0.txt
+https://github.com/serpwings/staticwordpress/blob/master/LICENSE
+
+
+Software distributed under the License is distributed on an "AS IS" basis,
+WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the
+specific language governing rights and limitations under the License.
+
+"""
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+# STANDARD LIBARY IMPORTS
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+
+import logging
+from pathlib import Path
+
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+# 3rd PARTY LIBRARY IMPORTS
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+
+from PyQt5.QtWidgets import (
+ QTabWidget,
+ QWidget,
+ QDialog,
+ QVBoxLayout,
+ QHBoxLayout,
+ QFormLayout,
+ QLineEdit,
+ QComboBox,
+ QTextEdit,
+ QLabel,
+ QGroupBox,
+ QToolButton,
+ QButtonGroup,
+ QRadioButton,
+ QDoubleSpinBox,
+ QMessageBox,
+ QFileDialog,
+ QPushButton,
+ QAction,
+)
+from PyQt5.QtCore import Qt, QSettings, QSize, QThread
+from PyQt5.QtGui import QIcon
+
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+# INTERNAL IMPORTS
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+
+from ..core.constants import (
+ REDIRECTS,
+ USER_AGENT,
+ CONFIGS,
+ SHARE_FOLDER_PATH,
+ SOURCE,
+ HOST,
+)
+
+from ..core.utils import is_url_valid
+
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+# IMPLEMENATIONS
+# +++++++++++++++++++++++++++++++++++++++++++++++++++++
+
+
+class ProjectDialog(QDialog):
+ def __init__(self, parent, project_, title_="Project Settings"):
+ super(ProjectDialog, self).__init__(parent=parent)
+ self.appConfigurations = QSettings(
+ CONFIGS["APPLICATION_NAME"], CONFIGS["APPLICATION_NAME"]
+ )
+
+ self._project = project_
+ vertical_layout_project = QVBoxLayout()
+ groupbox_general_settings = QGroupBox("General Settings")
+ form_layout_general_settings = QFormLayout()
+
+ self.lineedit_project_name = QLineEdit(self._project.name)
+ self.lineedit_project_name.setObjectName("name")
+ form_layout_general_settings.addRow(
+ QLabel("Project Name"), self.lineedit_project_name
+ )
+
+ self.lineedit_src_url = QLineEdit(self._project.src_url)
+ self.lineedit_src_url.setObjectName("src-url")
+ form_layout_general_settings.addRow(QLabel("Source Url"), self.lineedit_src_url)
+
+ horizontal_Layout_output_directory = QHBoxLayout()
+ self.lineedit_output = QLineEdit(str(self._project.output))
+ self.lineedit_output.setObjectName("output")
+ self.toolbutton_output_directory = QToolButton()
+ self.toolbutton_output_directory.setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/three-dots.svg")
+ )
+ self.toolbutton_output_directory.clicked.connect(self.get_output_directory)
+ horizontal_Layout_output_directory.addWidget(self.lineedit_output)
+ horizontal_Layout_output_directory.addWidget(self.toolbutton_output_directory)
+ form_layout_general_settings.addRow(
+ QLabel("Output Directory"), horizontal_Layout_output_directory
+ )
+
+ horizontal_layout_crawl_delay_user_agent = QHBoxLayout()
+ self.double_spinbox_delay = QDoubleSpinBox()
+ self.double_spinbox_delay.setMinimumWidth(120)
+ self.double_spinbox_delay.setMinimum(0)
+ self.double_spinbox_delay.setSingleStep(0.05)
+ self.double_spinbox_delay.setMaximum(2)
+ self.double_spinbox_delay.setMinimumWidth(120)
+ self.double_spinbox_delay.setValue(self._project.delay)
+ self.double_spinbox_delay.setObjectName("delay")
+ horizontal_layout_crawl_delay_user_agent.addWidget(self.double_spinbox_delay)
+
+ self.combobox_user_agent = QComboBox()
+ self.combobox_user_agent.setObjectName("user-agent")
+ self.combobox_user_agent.setMinimumWidth(120)
+ self.combobox_user_agent.addItems([item.value for item in list(USER_AGENT)])
+ self.combobox_user_agent.setCurrentText(self._project.user_agent.value)
+ horizontal_layout_crawl_delay_user_agent.addWidget(QLabel("User Agent"))
+ horizontal_layout_crawl_delay_user_agent.addWidget(self.combobox_user_agent)
+
+ form_layout_general_settings.addRow(
+ QLabel("Crawl Delay(sec)"), horizontal_layout_crawl_delay_user_agent
+ )
+
+ horitzontal_layout_project_type = QHBoxLayout()
+ button_group_project_type = QButtonGroup()
+
+ self.radiobutton_seo = QRadioButton("SEO", self)
+ self.radiobutton_seo.setObjectName("radio-seo")
+ self.radiobutton_seo.setEnabled(False)
+ self.radiobutton_seo.toggled.connect(self.update_widgets)
+
+ self.radiobutton_static_website = QRadioButton("Static Website", self)
+ self.radiobutton_static_website.setObjectName("radio-static-website")
+ self.radiobutton_static_website.setChecked(True)
+ self.radiobutton_static_website.toggled.connect(self.update_widgets)
+
+ button_group_project_type.addButton(self.radiobutton_seo)
+ button_group_project_type.addButton(self.radiobutton_static_website)
+
+ horitzontal_layout_project_type.addWidget(self.radiobutton_seo)
+ horitzontal_layout_project_type.addWidget(self.radiobutton_static_website)
+ horitzontal_layout_project_type.addStretch()
+ form_layout_general_settings.addRow(
+ QLabel("Project Type"), horitzontal_layout_project_type
+ )
+
+ groupbox_general_settings.setLayout(form_layout_general_settings)
+
+ widget_static_website_tab = QWidget()
+ form_layout_static_website_properties = QFormLayout()
+
+ horizontal_Layout_project_scource = QHBoxLayout()
+ self.combobox_source_type = QComboBox()
+ self.combobox_source_type.setObjectName("source")
+ self.combobox_source_type.setMinimumWidth(120)
+ self.combobox_source_type.addItems([item.value for item in list(SOURCE)])
+ self.combobox_source_type.setCurrentText(self._project.src_type.value)
+ horizontal_Layout_project_scource.addWidget(self.combobox_source_type)
+ horizontal_Layout_project_scource.addStretch()
+
+ form_layout_static_website_properties.addRow(
+ QLabel("Data Source"), horizontal_Layout_project_scource
+ )
+
+ horizontal_Layout_project_redirects = QHBoxLayout()
+ self.combobox_redirects = QComboBox()
+ self.combobox_redirects.setObjectName("redirects")
+ self.combobox_redirects.setMinimumWidth(120)
+ self.combobox_redirects.addItems([item.value for item in list(REDIRECTS)])
+ self.combobox_redirects.setCurrentText(self._project.redirects.value)
+
+ horizontal_Layout_project_redirects.addWidget(self.combobox_redirects)
+ horizontal_Layout_project_redirects.addStretch()
+ form_layout_static_website_properties.addRow(
+ QLabel("Redirects Source"), horizontal_Layout_project_redirects
+ )
+
+ horizontal_Layout_project_destination = QHBoxLayout()
+ self.combobox_project_destination = QComboBox()
+ self.combobox_project_destination.setObjectName("host")
+ self.combobox_project_destination.setMinimumWidth(120)
+ self.combobox_project_destination.addItems([item.value for item in list(HOST)])
+ self.combobox_project_destination.setCurrentText(self._project.host.value)
+
+ horizontal_Layout_project_destination.addWidget(
+ self.combobox_project_destination
+ )
+ horizontal_Layout_project_destination.addStretch()
+ form_layout_static_website_properties.addRow(
+ QLabel("Destination Host"), horizontal_Layout_project_destination
+ )
+
+ self.lineedit_dest_url = QLineEdit(self._project.dst_url)
+ self.lineedit_dest_url.setObjectName("dst-url")
+ form_layout_static_website_properties.addRow(
+ QLabel("Destination Url"), self.lineedit_dest_url
+ )
+
+ self.lineedit_search = QLineEdit(self._project.search)
+ self.lineedit_search.setObjectName("search")
+ form_layout_static_website_properties.addRow(
+ QLabel("Search Page"), self.lineedit_search
+ )
+
+ self.lineedit_404_page = QLineEdit(self._project._404)
+ self.lineedit_404_page.setObjectName("404-error")
+ form_layout_static_website_properties.addRow(
+ QLabel("404 Page"), self.lineedit_404_page
+ )
+ widget_static_website_tab.setLayout(form_layout_static_website_properties)
+
+ horizontal_Layout_sitemap = QHBoxLayout()
+ self.lineedit_sitemap = QLineEdit(self._project.sitemap)
+ self.lineedit_sitemap.setObjectName("sitemap")
+ self.toolbutton_output_sitemap = QToolButton()
+ self.toolbutton_output_sitemap.setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/search.svg")
+ )
+ horizontal_Layout_sitemap.addWidget(self.lineedit_sitemap)
+ horizontal_Layout_sitemap.addWidget(self.toolbutton_output_sitemap)
+ form_layout_static_website_properties.addRow(
+ QLabel("Sitemap Location"), horizontal_Layout_sitemap
+ )
+
+ self.textedit_additional_urls = QTextEdit()
+ self.textedit_additional_urls.setText("\n".join(self._project.additional))
+ self.textedit_additional_urls.setObjectName("additional")
+ self.textedit_exclude_patterns = QTextEdit()
+ self.textedit_exclude_patterns.setText("\n".join(self._project.exclude))
+ self.textedit_exclude_patterns.setObjectName("exclude")
+
+ widget_project_api_tab = QWidget()
+ vertical_layout_project_api = QVBoxLayout()
+
+ groupbox_wp_api = QGroupBox("WordPress")
+ form_layout_wp_api = QFormLayout()
+ self.lineedit_wp_user = QLineEdit(self._project.wp_user)
+ self.lineedit_wp_user.setObjectName("wordpress-user")
+ form_layout_wp_api.addRow(QLabel("User Name"), self.lineedit_wp_user)
+
+ self.lineedit_wp_api_token = QLineEdit(self._project.wp_api_token)
+ self.lineedit_wp_api_token.setEchoMode(QLineEdit.Password)
+ self.lineedit_wp_api_token.setObjectName("wordpress-api-token")
+
+ toolbutton_show_wp_api_token = QToolButton()
+ toolbutton_show_wp_api_token.setObjectName("toolbutton_show_wp_api_token")
+ toolbutton_show_wp_api_token.setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/visibility.svg")
+ )
+ toolbutton_show_wp_api_token.setCheckable(True)
+ toolbutton_show_wp_api_token.clicked.connect(self.change_password_visiblity)
+
+ horizontal_layout_wp_api_token = QHBoxLayout()
+ horizontal_layout_wp_api_token.addWidget(self.lineedit_wp_api_token)
+ horizontal_layout_wp_api_token.addWidget(toolbutton_show_wp_api_token)
+
+ form_layout_wp_api.addRow(QLabel("API Token"), horizontal_layout_wp_api_token)
+ groupbox_wp_api.setLayout(form_layout_wp_api)
+
+ groupbox_gh_api = QGroupBox("GitHub")
+ form_layout_gh_api = QFormLayout()
+
+ self.lineedit_gh_repo = QLineEdit(self._project.gh_repo)
+ self.lineedit_gh_repo.setObjectName("github-repository")
+ form_layout_gh_api.addRow(QLabel("Repository"), self.lineedit_gh_repo)
+
+ self.lineedit_gh_token = QLineEdit(self._project.gh_token)
+ self.lineedit_gh_token.setEchoMode(QLineEdit.Password)
+ self.lineedit_gh_token.setObjectName("github-token")
+ horizontal_layout_gh_token = QHBoxLayout()
+ horizontal_layout_gh_token.addWidget(self.lineedit_gh_token)
+
+ toolbutton_show_gh_token = QToolButton()
+ toolbutton_show_gh_token.setObjectName("toolbutton_show_gh_token")
+ toolbutton_show_gh_token.setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/visibility.svg")
+ )
+ toolbutton_show_gh_token.setCheckable(True)
+ toolbutton_show_gh_token.clicked.connect(self.change_password_visiblity)
+ horizontal_layout_gh_token.addWidget(toolbutton_show_gh_token)
+ form_layout_gh_api.addRow(QLabel("API Token"), horizontal_layout_gh_token)
+
+ groupbox_gh_api.setLayout(form_layout_gh_api)
+ vertical_layout_project_api.addWidget(groupbox_wp_api)
+ vertical_layout_project_api.addWidget(groupbox_gh_api)
+
+ widget_project_api_tab.setLayout(vertical_layout_project_api)
+
+ self.tabwidget_dialog = QTabWidget()
+ self.tabwidget_dialog.addTab(widget_static_website_tab, "Static &Website")
+ self.tabwidget_dialog.addTab(widget_project_api_tab, "&API")
+ self.tabwidget_dialog.addTab(self.textedit_additional_urls, "Additional &Files")
+ self.tabwidget_dialog.addTab(
+ self.textedit_exclude_patterns, "Exclude &Patterns"
+ )
+
+ vertical_layout_project.addWidget(groupbox_general_settings)
+ vertical_layout_project.addWidget(self.tabwidget_dialog)
+ vertical_layout_project.addStretch()
+
+ self.pushbutton_verify = QPushButton("&Verify")
+ self.pushbutton_verify.setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/check_project.svg")
+ )
+ self.pushbutton_verify.clicked.connect(self.check_project)
+
+ self.pushbutton_save = QPushButton("&Save")
+ self.pushbutton_save.setIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg"))
+ self.pushbutton_save.setDefault(True)
+ self.pushbutton_save.clicked.connect(self.accept)
+
+ self.pushbutton_cancel = QPushButton("&Cancel")
+ self.pushbutton_cancel.setIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/cancel.svg"))
+ self.pushbutton_cancel.clicked.connect(self.reject)
+
+ horizontal_layout_buttons = QHBoxLayout()
+ horizontal_layout_buttons.addWidget(self.pushbutton_verify)
+ horizontal_layout_buttons.addStretch()
+ horizontal_layout_buttons.addWidget(self.pushbutton_save)
+ horizontal_layout_buttons.addWidget(self.pushbutton_cancel)
+ vertical_layout_project.addLayout(horizontal_layout_buttons)
+
+ self.setLayout(vertical_layout_project)
+ self.setWindowTitle(title_)
+ self.setWindowIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg"))
+
+ def change_password_visiblity(self):
+ if not self.sender().isChecked():
+ self.sender().setIcon(QIcon(f"{SHARE_FOLDER_PATH}/icons/visibility.svg"))
+ self.sender().setChecked(False)
+ if self.sender().objectName() == "toolbutton_show_gh_token":
+ self.lineedit_gh_token.setEchoMode(QLineEdit.Password)
+ elif self.sender().objectName() == "toolbutton_show_wp_api_token":
+ self.lineedit_wp_api_token.setEchoMode(QLineEdit.Password)
+
+ else:
+ self.sender().setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/visibility_lock.svg")
+ )
+ self.sender().setChecked(True)
+ if self.sender().objectName() == "toolbutton_show_gh_token":
+ self.lineedit_gh_token.setEchoMode(QLineEdit.Normal)
+ elif self.sender().objectName() == "toolbutton_show_wp_api_token":
+ self.lineedit_wp_api_token.setEchoMode(QLineEdit.Normal)
+
+ def update_widgets(self):
+ if self.sender().objectName() == "radio-static-website":
+ self.tabwidget_dialog.setVisible(True)
+ self.setFixedSize(QSize(448, 437))
+ elif self.sender().objectName() == "radio-seo":
+ self.tabwidget_dialog.setVisible(False)
+ self.setFixedSize(QSize(448, 208))
+ else:
+ pass
+
+ def get_output_directory(self):
+ """"""
+ output_directory = QFileDialog.getExistingDirectory(
+ self,
+ "Select Output Directory",
+ str(self.appConfigurations.value("last-project")),
+ )
+ if output_directory:
+ self.lineedit_output.setText(output_directory)
+ self.appConfigurations.setValue("last-project", output_directory)
+
+ def check_project(self):
+ """"""
+ # TODO: Add checks for WP_API and Gh_API and if not present then disable them.
+ # TODO: Move these checks to background thread
+ if not (self.lineedit_wp_api_token.text() and self.lineedit_wp_user.text()):
+ self.combobox_redirects.setCurrentText(REDIRECTS.NONE.value)
+
+ if self.lineedit_project_name.text():
+ self.lineedit_project_name.setStyleSheet(
+ f"background-color: {CONFIGS['COLOR']['SUCCESS']}"
+ )
+ else:
+ self.lineedit_project_name.setStyleSheet(
+ f"background-color: {CONFIGS['COLOR']['ERROR']}"
+ )
+
+ if is_url_valid(self.lineedit_src_url.text()):
+ self.lineedit_src_url.setStyleSheet(
+ f"background-color: {CONFIGS['COLOR']['SUCCESS']}"
+ )
+ else:
+ self.lineedit_src_url.setStyleSheet(
+ f"background-color: {CONFIGS['COLOR']['ERROR']}"
+ )
+
+ if self.lineedit_output.text() and Path(self.lineedit_output.text()).is_dir():
+ self.lineedit_output.setStyleSheet(
+ f"background-color: {CONFIGS['COLOR']['SUCCESS']}"
+ )
+ else:
+ self.lineedit_output.setStyleSheet(
+ f"background-color: {CONFIGS['COLOR']['ERROR']}"
+ )
+
+ def accept(self) -> None:
+ """"""
+ if all(
+ [
+ self.lineedit_project_name.text(),
+ self.lineedit_output.text(),
+ is_url_valid(self.lineedit_src_url.text()),
+ Path(self.lineedit_output.text()).is_dir(),
+ ]
+ ):
+ self._project.create()
+ self._project.output = Path(self.lineedit_output.text())
+ self._project.path = Path(f"{self._project.output}/._data/.project.json")
+ self._project.name = self.lineedit_project_name.text()
+ self._project.src_url = self.lineedit_src_url.text()
+ self._project.sitemap = self.lineedit_sitemap.text()
+ self._project.wp_user = self.lineedit_wp_user.text()
+ self._project.wp_api_token = self.lineedit_wp_api_token.text()
+ self._project.search = self.lineedit_search.text()
+ self._project._404 = self.lineedit_404_page.text()
+ self._project.delay = self.double_spinbox_delay.value()
+ self._project.redirects = REDIRECTS[self.combobox_redirects.currentText()]
+ self._project.src_type = SOURCE[self.combobox_source_type.currentText()]
+ self._project.user_agent = USER_AGENT[
+ self.combobox_user_agent.currentText()
+ ]
+ self._project.host = HOST[self.combobox_project_destination.currentText()]
+ self._project.dst_url = self.lineedit_dest_url.text()
+ self._project.gh_token = self.lineedit_gh_token.text()
+ self._project.gh_repo = self.lineedit_gh_repo.text()
+ self._project.additional = (
+ self.textedit_additional_urls.toPlainText().split("\n")
+ )
+ self._project.exclude = self.textedit_exclude_patterns.toPlainText().split(
+ "\n"
+ )
+ return super().accept()
+ else:
+ msgBox = QMessageBox(parent=self)
+ msgBox.setText(
+ "Cannot start this project.
Please check project settings."
+ )
+ msgBox.addButton(QMessageBox.Ok).setIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/ok.svg")
+ )
+ msgBox.setWindowIcon(
+ QIcon(f"{SHARE_FOLDER_PATH}/icons/static-wordpress.svg")
+ )
+ msgBox.setTextFormat(Qt.RichText)
+ msgBox.setWindowTitle(
+ "Invalid Project Settings",
+ )
+ msgBox.exec()
diff --git a/src/staticwordpress/gui/rawtext.py b/src/staticwordpress/gui/rawtext.py
index f30fe2a..6f422d9 100644
--- a/src/staticwordpress/gui/rawtext.py
+++ b/src/staticwordpress/gui/rawtext.py
@@ -51,9 +51,9 @@
# +++++++++++++++++++++++++++++++++++++++++++++++++++++
-class RawTextWidget(QDialog):
- def __init__(self, src_url: str, dest_url: str):
- super(RawTextWidget, self).__init__()
+class RawTextDialog(QDialog):
+ def __init__(self, parent, src_url: str, dest_url: str):
+ super(RawTextDialog, self).__init__(parent=parent)
self.appConfigurations = QSettings(
CONFIGS["APPLICATION_NAME"], CONFIGS["APPLICATION_NAME"]
)
diff --git a/src/staticwordpress/gui/workflow.py b/src/staticwordpress/gui/workflow.py
index 2bc4daf..fba1b90 100644
--- a/src/staticwordpress/gui/workflow.py
+++ b/src/staticwordpress/gui/workflow.py
@@ -36,7 +36,7 @@
from ..core.project import Project
from ..core.constants import SOURCE
from ..core.workflow import Workflow
-from .utils import logging_decorator, progress_decorator
+from ..gui.utils import logging_decorator, progress_decorator
# +++++++++++++++++++++++++++++++++++++++++++++++++++++
diff --git a/src/staticwordpress/share/_ignore b/src/staticwordpress/share/_ignore
new file mode 100644
index 0000000..e0b03fb
--- /dev/null
+++ b/src/staticwordpress/share/_ignore
@@ -0,0 +1 @@
+._data/
\ No newline at end of file
diff --git a/src/staticwordpress/share/config.json b/src/staticwordpress/share/config.json
index 054d50b..228165d 100644
--- a/src/staticwordpress/share/config.json
+++ b/src/staticwordpress/share/config.json
@@ -44,7 +44,7 @@
"xmlns:image": "http://www.google.com/schemas/sitemap-image/1.1"
},
"COLOR": {
- "SUCCESS": "#005500",
+ "SUCCESS": "#aaff7f",
"WARNING": "#ffff7f",
"ERROR": "#ff557f"
},
diff --git a/src/staticwordpress/share/gui.json b/src/staticwordpress/share/gui.json
index 1645a55..28c5c22 100644
--- a/src/staticwordpress/share/gui.json
+++ b/src/staticwordpress/share/gui.json
@@ -1,8 +1,8 @@
{
"MENUS": [
{
- "name": "menu_file",
- "text": "&File",
+ "name": "menu_project",
+ "text": "&Project",
"enable": true,
"icon": "",
"parent": ""
@@ -56,85 +56,85 @@
"text": "Project",
"enable": true
},
+ {
+ "name": "toolbar_edit",
+ "text": "Edit",
+ "enable": true
+ },
{
"name": "toolbar_wordpres",
"text": "WordPress",
- "enable": true
+ "enable": false
},
{
"name": "toolbar_github",
"text": "GitHub",
- "enable": true
- },
- {
- "name": "toolbar_edit",
- "text": "Edit",
- "enable": true
+ "enable": false
}
],
"ACTIONS": [
{
"icon": "/icons/file-outline.svg",
- "name": "action_file_project_new",
+ "name": "action_project_new",
"visible": true,
- "text": "&New Project",
+ "text": "&New",
"shortcut": "Ctrl+N",
- "tooltip": "New Project (Ctrl+N)",
- "function": "self.create_project",
+ "tooltip": "New (Ctrl+N)",
+ "function": "self.new_project",
"setCheckable": false,
- "menu": "menu_file",
+ "menu": "menu_project",
"seperator": false,
"toolbar": "toolbar_project"
},
{
"icon": "/icons/folder-outline.svg",
- "name": "action_file_project_open",
+ "name": "action_project_open",
"visible": true,
- "text": "&Open Project",
+ "text": "&Open",
"shortcut": "Ctrl+O",
- "tooltip": "Open Project (Ctrl+O)",
+ "tooltip": "Open (Ctrl+O)",
"function": "self.open_project",
"setCheckable": false,
- "menu": "menu_file",
+ "menu": "menu_project",
"seperator": false,
"toolbar": "toolbar_project"
},
{
- "icon": "/icons/content-save-outline.svg",
- "name": "action_file_project_save",
+ "icon": "/icons/project_settings.svg",
+ "name": "action_project_settings",
"visible": true,
- "text": "&Save Project",
- "shortcut": "Ctrl+S",
- "tooltip": "Save Project (Ctrl+S)",
- "function": "self.save_project",
+ "text": "&Settings",
+ "shortcut": "Ctrl+C",
+ "tooltip": "Check (Ctrl+C)",
+ "function": "self.show_project",
"setCheckable": false,
- "menu": "menu_file",
+ "menu": "menu_project",
"seperator": false,
"toolbar": "toolbar_project"
},
{
- "icon": "/icons/file-document-remove-outline.svg",
- "name": "action_file_project_close",
+ "icon": "/icons/project_close.svg",
+ "name": "action_project_close",
"visible": true,
- "text": "&Close Project",
+ "text": "&Close",
"shortcut": "Ctrl+X",
- "tooltip": "Close Project (Ctrl+X)",
+ "tooltip": "Close (Ctrl+X)",
"function": "self.close_project",
"setCheckable": false,
- "menu": "menu_file",
+ "menu": "menu_project",
"seperator": true,
"toolbar": "toolbar_project"
},
{
"icon": "/icons/exit-to-app.svg",
- "name": "action_file_exit",
+ "name": "action_project_exit",
"visible": true,
"text": "&Exit",
"shortcut": "Ctrl+Q",
"tooltip": "Exit (Ctrl+Q)",
"function": "self.close",
"setCheckable": false,
- "menu": "menu_file",
+ "menu": "menu_project",
"seperator": false,
"toolbar": ""
},
@@ -359,19 +359,6 @@
"menu": "menu_tools",
"toolbar": ""
},
- {
- "icon": "/icons/check_project.svg",
- "name": "action_file_check_project",
- "visible": true,
- "text": "&Check Project",
- "shortcut": "Ctrl+F3",
- "tooltip": "Check Project (Ctrl+F3)",
- "function": "self.check_project",
- "setCheckable": false,
- "menu": "menu_tools",
- "seperator": false,
- "toolbar": false
- },
{
"icon": "/icons/text-recognition.svg",
"name": "action_utilities_extract_url_from_raw_text",
diff --git a/src/staticwordpress/share/icons/project_close.svg b/src/staticwordpress/share/icons/project_close.svg
new file mode 100644
index 0000000..d9f6b9c
--- /dev/null
+++ b/src/staticwordpress/share/icons/project_close.svg
@@ -0,0 +1,38 @@
+
+
diff --git a/src/staticwordpress/share/icons/project_settings.svg b/src/staticwordpress/share/icons/project_settings.svg
new file mode 100644
index 0000000..af1a0dc
--- /dev/null
+++ b/src/staticwordpress/share/icons/project_settings.svg
@@ -0,0 +1,38 @@
+
+
diff --git a/src/staticwordpress/share/icons/visibility.svg b/src/staticwordpress/share/icons/visibility.svg
new file mode 100644
index 0000000..f7e15dd
--- /dev/null
+++ b/src/staticwordpress/share/icons/visibility.svg
@@ -0,0 +1,38 @@
+
+
diff --git a/src/staticwordpress/share/icons/visibility_lock.svg b/src/staticwordpress/share/icons/visibility_lock.svg
new file mode 100644
index 0000000..0a6d371
--- /dev/null
+++ b/src/staticwordpress/share/icons/visibility_lock.svg
@@ -0,0 +1,38 @@
+
+
diff --git a/tests/test_project.py b/tests/test_project.py
index 80f757e..c4af89e 100644
--- a/tests/test_project.py
+++ b/tests/test_project.py
@@ -34,5 +34,5 @@
def test_project_create():
p = Project()
assert p.status == PROJECT.NOT_FOUND
- assert p.redirects == REDIRECTS.REDIRECTION
+ assert p.redirects == REDIRECTS.NONE
assert p.host == HOST.NETLIFY