app.py 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. # @Author: Bachir Soussi Chiadmi <bach>
  4. # @Date: 23-05-2017
  5. # @Email: bachir@figureslibres.io
  6. # @Last modified by: bach
  7. # @Last modified time: 21-04-2017
  8. # @License: GPL-V3
  9. import sys, os, shutil, tempfile
  10. from PyQt5 import QtCore
  11. from PyQt5.QtCore import QCoreApplication, QUrl, pyqtSlot, QSettings
  12. from PyQt5.QtGui import QIcon, QKeySequence, QFont, QSyntaxHighlighter
  13. from PyQt5.QtWidgets import QMainWindow, QAction, QWidget, QApplication, QShortcut, QGridLayout, QLabel, QTabWidget, QStackedWidget, QHBoxLayout, QVBoxLayout, QSplitter, QSplitterHandle, QPlainTextEdit, QInputDialog, QLineEdit, QFileDialog, QMessageBox, QPushButton
  14. # from PyQt5.QtNetwork import QNetworkProxyFactory, QNetworkRequest
  15. from PyQt5.QtWebKit import QWebSettings
  16. from PyQt5.QtWebKitWidgets import QWebPage, QWebView, QWebInspector
  17. import json
  18. import git
  19. from classes import server, compiler, design, content
  20. class Core():
  21. def __init__(self, parent=None):
  22. # restore previous preferences
  23. self.restorePreferences()
  24. self._mw = False
  25. self.temp = tempfile.mkdtemp()
  26. # print(self.temp)
  27. self.tempcwd = False
  28. if(self.cwd == None or not os.path.isdir(self.cwd)):
  29. self.cwd = os.path.join(self.temp, 'cwd')
  30. self.tempcwd = True
  31. self.initnewproject()
  32. self.server = server.Server(self.temp)
  33. self.compiler = compiler.Compiler(self.temp)
  34. @property
  35. def mainwindow(self):
  36. return self.mainwindow
  37. @mainwindow.setter
  38. def mainwindow(self, mw):
  39. if not self._mw:
  40. self._mw = mw
  41. if not self.tempcwd:
  42. self._mw.setWindowTitle("Cascade – "+self.cwd)
  43. def restorePreferences(self):
  44. # print("restorePreferences")
  45. settings = QSettings('FiguresLibres', 'Cascade')
  46. # settings.clear()
  47. # print(settings.allKeys())
  48. self.cwd = settings.value('core/cwd', None)
  49. self.mw_size = settings.value('mainwindow/size', QtCore.QSize(1024, 768))
  50. self.mw_pos = settings.value('mainwindow/pos', QtCore.QPoint(0, 0))
  51. self.mw_curstack = int(settings.value('mainwindow/curstack', 0))
  52. def savePreferences(self):
  53. # print("savePreferences")
  54. settings = QSettings('FiguresLibres', 'Cascade')
  55. # print(settings.allKeys())
  56. if not self.tempcwd:
  57. settings.setValue('core/cwd', self.cwd)
  58. settings.setValue('mainwindow/size', self._mw.size())
  59. settings.setValue('mainwindow/pos', self._mw.pos())
  60. settings.setValue('mainwindow/curstack', self._mw.mainstack.currentIndex())
  61. def initnewproject(self, cwd = None):
  62. if cwd == None :
  63. cwd = self.cwd
  64. else :
  65. self.cwd = cwd
  66. shutil.copytree('templates/newproject', cwd)
  67. self.prefs = json.loads(open(os.path.join(cwd,'.config/prefs.json')).read())
  68. self.summary = json.loads(open(os.path.join(cwd,'.config/summary.json')).read())
  69. self.repository = git.Repo.init(cwd)
  70. self.repository.index.add(['assets','contents','.config'])
  71. self.repository.index.commit("initial commit")
  72. # TODO: set mdtohtml compiler from project md to app index.html
  73. # TODO: embed project styles.scss to app scss frame work
  74. def saveproject(self, cwd = None):
  75. if not cwd == None and self.tempcwd:
  76. shutil.copytree(self.cwd, cwd)
  77. self.cwd = cwd
  78. self.tempcwd = False
  79. self._mw.setWindowTitle("Cascade – "+self.cwd)
  80. def quit(self):
  81. self.savePreferences()
  82. shutil.rmtree(self.temp, ignore_errors=True)
  83. QCoreApplication.instance().quit()
  84. class MainWindow(QMainWindow):
  85. def __init__(self, core):
  86. super(MainWindow, self).__init__()
  87. # load core class
  88. self.core = core
  89. self.setWindowTitle("Cascade")
  90. self.setWindowIcon(QIcon('assets/images/icon.png'))
  91. self.resize(self.core.mw_size)
  92. self.move(self.core.mw_pos)
  93. self.initMenuBar()
  94. self.initMainStack()
  95. self.show()
  96. def initMenuBar(self):
  97. # menu bar
  98. bar = self.menuBar()
  99. file = bar.addMenu("&File")
  100. new = QAction("&New Project",self)
  101. new.setShortcut("Ctrl+n")
  102. file.addAction(new)
  103. open = QAction("&Open",self)
  104. open.setShortcut("Ctrl+o")
  105. file.addAction(open)
  106. self.save_action = QAction("&Save Project as",self)
  107. self.save_action.setShortcut("Ctrl+Shift+s")
  108. file.addAction(self.save_action)
  109. # if not self.core.tempcwd:
  110. # self.save_action.setEnabled(False)
  111. quit = QAction("&Quit",self)
  112. quit.setShortcut("Ctrl+q")
  113. file.addAction(quit)
  114. file.triggered[QAction].connect(self.onfilemenutrigger)
  115. # edit menu
  116. edit = bar.addMenu("&Edit")
  117. edit.addAction("&copy")
  118. edit.addAction("&paste")
  119. edit.addAction("&build")
  120. edit.addAction("&reload")
  121. edit.addAction("&preferences")
  122. # view menu
  123. view = bar.addMenu("&View")
  124. designview = QAction("&Design",self)
  125. designview.setShortcut("F1")
  126. view.addAction(designview)
  127. contentview = QAction("&Content",self)
  128. contentview.setShortcut("F2")
  129. view.addAction(contentview)
  130. versionview = QAction("&Version",self)
  131. versionview.setShortcut("F3")
  132. view.addAction(versionview)
  133. view.triggered[QAction].connect(self.onviewmenutrigger)
  134. # about menu
  135. about = bar.addMenu("About")
  136. about.addAction("&Website")
  137. def onfilemenutrigger(self, q):
  138. print(q.text()+" is triggered")
  139. if q.text() == "&New Project":
  140. self.newprojectdialogue()
  141. elif q.text() == "&Open":
  142. self.openprojectdialogue()
  143. elif q.text() == "&Save Project as":
  144. self.saveprojectdialogue()
  145. elif q.text() == "&Quit":
  146. self.quit()
  147. def openprojectdialogue(self):
  148. print("open")
  149. def newprojectdialogue(self):
  150. projectname = QFileDialog.getSaveFileName(self, 'New Project', '')
  151. # TODO: no file type
  152. try:
  153. if not os.path.isdir(projectname[0]):
  154. self.core.initnewproject(projectname[0])
  155. else:
  156. print("folder already exists")
  157. # TODO: check if is cascade folder
  158. except Exception as e:
  159. pass
  160. def saveprojectdialogue(self, quit=False):
  161. projectname = QFileDialog.getSaveFileName(self, 'Save Project', '')
  162. # TODO: no file type
  163. try:
  164. if not os.path.isdir(projectname[0]):
  165. self.core.saveproject(projectname[0])
  166. # self.save_action.setEnabled(False)
  167. if quit:
  168. self.quit()
  169. else:
  170. print("folder already exists")
  171. # TODO: check if is cascade folder
  172. except Exception as e:
  173. pass
  174. def quit(self):
  175. print("Quit")
  176. if self.core.tempcwd:
  177. buttonReply = QMessageBox.question(self, 'Project Not Saved', "Do you want to save your current project before quiting?", QMessageBox.Yes | QMessageBox.No | QMessageBox.Cancel, QMessageBox.Cancel)
  178. if buttonReply == QMessageBox.Yes:
  179. self.saveprojectdialogue(quit=True)
  180. if buttonReply == QMessageBox.No:
  181. self.core.quit()
  182. else:
  183. self.core.quit()
  184. def onviewmenutrigger(self, q):
  185. print(q.text()+" is triggered")
  186. if q.text() == "&Design":
  187. self.mainstack.setCurrentIndex(0)
  188. elif q.text() == "&Content":
  189. self.mainstack.setCurrentIndex(1)
  190. elif q.text() == "&Version":
  191. self.mainstack.setCurrentIndex(2)
  192. def initMainStack(self):
  193. self.mainstack = QStackedWidget()
  194. self.designstack = design.DesignStack(self.core)
  195. self.contentstack = content.ContentStack(self.core)
  196. self.versionstack = QLabel("Version (git).")
  197. self.mainstack.addWidget(self.designstack)
  198. self.mainstack.addWidget(self.contentstack)
  199. self.mainstack.addWidget(self.versionstack)
  200. self.mainstack.setCurrentIndex(self.core.mw_curstack)
  201. self.setCentralWidget(self.mainstack)
  202. def main():
  203. app = QApplication(sys.argv)
  204. app.setOrganizationName('figli')
  205. app.setApplicationName('Cascade')
  206. core = Core()
  207. mainappwindow = MainWindow(core)
  208. core.mainwindow = mainappwindow
  209. sys.exit(app.exec_())
  210. if __name__ == "__main__":
  211. main()