content.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. import os, re
  4. from PyQt5 import QtCore
  5. from PyQt5.QtCore import QSettings
  6. from PyQt5.QtGui import QKeySequence
  7. from PyQt5.QtWidgets import QWidget, QLabel, QHBoxLayout, QVBoxLayout, QSplitter, QListWidget, QListWidgetItem, QAbstractItemView, QPushButton, QInputDialog, QPlainTextEdit, QTextEdit, QShortcut
  8. import markdown2
  9. import json
  10. # _____
  11. # / ___/__ ______ ___ ____ ___ ____ ________ __
  12. # \__ \/ / / / __ `__ \/ __ `__ \/ __ `/ ___/ / / /
  13. # ___/ / /_/ / / / / / / / / / / / /_/ / / / /_/ /
  14. # /____/\__,_/_/ /_/ /_/_/ /_/ /_/\__,_/_/ \__, /
  15. # /____/
  16. class Summary(QWidget):
  17. def __init__(self, parent):
  18. super(Summary, self).__init__(parent)
  19. self.parent = parent
  20. jsonfilepath = os.path.join(self.parent.core.cwd,'.config/summary.json')
  21. sum_json = open(jsonfilepath).read()
  22. self.sum = json.loads(sum_json)
  23. vbox = QVBoxLayout()
  24. vbox.setContentsMargins(0,0,0,0)
  25. self.list = SummaryList(self)
  26. vbox.addWidget(self.list)
  27. self.actions = SummaryActions(self)
  28. vbox.addWidget(self.actions)
  29. self.setLayout(vbox)
  30. def addItem(self, text):
  31. # file
  32. filename = re.sub(r'\W', "_", text)+".md"
  33. # TODO: check if file does not already exists
  34. filepath = os.path.join(self.parent.core.cwd,'contents',filename)
  35. with open(filepath, 'w') as fp:
  36. fp.write('#'+text)
  37. # json
  38. item = {"title":text,"file":filename}
  39. self.sum.append(item)
  40. jsonfilepath = os.path.join(self.parent.core.cwd,'.config/summary.json')
  41. with open(jsonfilepath, "w") as fp:
  42. json.dump(self.sum, fp, ensure_ascii=False, indent="\t")
  43. # refresh list
  44. self.list.addNewItem(item)
  45. # reload content compiler
  46. self.parent.core.contentcompiler.reload()
  47. def recordNewList(self):
  48. newdata = []
  49. for i in range(0,self.list.count()):
  50. # print(self.item(i).item['title'])
  51. newdata.append(self.list.item(i).data)
  52. # print(newdata)
  53. self.sum = newdata
  54. jsonfilepath = os.path.join(self.parent.core.cwd,'.config/summary.json')
  55. with open(jsonfilepath, "w") as fp:
  56. json.dump(newdata, fp, ensure_ascii=False, indent="\t")
  57. # reload content compiler
  58. self.parent.core.contentcompiler.reload()
  59. class SummaryList(QListWidget):
  60. def __init__(self, parent):
  61. super(SummaryList, self).__init__(parent)
  62. self.parent = parent
  63. # self.sum = sum
  64. # print(self.sum)
  65. # self.setSortingEnabled(True)
  66. self.setDragEnabled(True)
  67. self.setSelectionMode(QAbstractItemView.SingleSelection)
  68. self.setAcceptDrops(True)
  69. self.setDropIndicatorShown(True)
  70. self.setDragDropMode(QAbstractItemView.InternalMove)
  71. self.model().rowsMoved.connect(self.onRowsMoved)
  72. # print(self.model())
  73. self.itemActivated.connect(self.onItemActivated)
  74. # add markdown files to the list
  75. for itemdata in self.parent.sum:
  76. self.addNewItem(itemdata)
  77. # self.setCurrentRow(0)
  78. # self.setCurrentIndex()
  79. # self.setCurrentItem()
  80. self.item(0).setSelected(True)
  81. # TODO: activate first item by default as it will open it with editor
  82. # TODO: show activated item on the list
  83. # TODO: show modifed item on the list
  84. def onRowsMoved(self, model, start, end, dest):
  85. # print("onRowsMoved")
  86. self.parent.recordNewList()
  87. def addNewItem(self, item):
  88. self.addItem(SummaryListWidgetItem(self,item))
  89. def onItemActivated(self, item):
  90. # print('onItemActivated', item.data)
  91. self.parent.parent.editor.openFile()
  92. class SummaryListWidgetItem(QListWidgetItem):
  93. def __init__(self,parent,data):
  94. super(SummaryListWidgetItem, self).__init__(parent)
  95. self.parent = parent
  96. self.data = data
  97. self.setText(data['title'])
  98. self.setToolTip(data['file'])
  99. class SummaryActions(QWidget):
  100. def __init__(self,parent):
  101. super(SummaryActions, self).__init__(parent)
  102. self.parent = parent
  103. self.hbox = QHBoxLayout()
  104. self.hbox.setContentsMargins(0,0,0,0)
  105. new = QPushButton("New Page", self)
  106. new.setShortcut('Ctrl+Shift+n')
  107. # new.setIcon(Icon(ico)))
  108. new.clicked.connect(self.onAddPage)
  109. self.hbox.addWidget(new)
  110. delete = QPushButton("Delete Page", self)
  111. delete.setShortcut('Ctrl+Shift+sup')
  112. # delete.setIcon(Icon(ico)))
  113. delete.clicked.connect(self.onDeletePage)
  114. self.hbox.addWidget(delete)
  115. self.setLayout(self.hbox)
  116. def onAddPage(self):
  117. text, ok = QInputDialog.getText(self, 'Input Dialog', 'Page Name:')
  118. if ok:
  119. self.parent.addItem(text)
  120. def onDeletePage(self):
  121. print("onDeletePage")
  122. # TODO: get the current selected page
  123. # TODO: ask for confirmation for deleting the current selecred page
  124. # TODO: call for summary widget to delete the page
  125. # ______ ___ __
  126. # / ____/___/ (_) /_____ _____
  127. # / __/ / __ / / __/ __ \/ ___/
  128. # / /___/ /_/ / / /_/ /_/ / /
  129. # /_____/\__,_/_/\__/\____/_/
  130. class MarkdownEditor(QWidget):
  131. def __init__(self,parent):
  132. super(MarkdownEditor, self).__init__(parent)
  133. self.parent = parent
  134. self.changed = False
  135. self.hbox = QHBoxLayout()
  136. self.hbox.setContentsMargins(0,0,0,0)
  137. self.styles = """
  138. background-color:white;
  139. color:black;
  140. padding:20px;
  141. """
  142. self.editor = QPlainTextEdit(self)
  143. self.editor.setStyleSheet(self.styles)
  144. self.hbox.addWidget(self.editor)
  145. self.viewer = QTextEdit(self)
  146. self.viewer.setReadOnly(True)
  147. self.viewer.setStyleSheet(self.styles)
  148. # TODO: show all html blocks on viewer
  149. self.hbox.addWidget(self.viewer)
  150. self.setLayout(self.hbox)
  151. self.editor.textChanged.connect(self.onTextChanged)
  152. self.openFile()
  153. self.shortcut = QShortcut(QKeySequence("Ctrl+s"), self)
  154. self.shortcut.activated.connect(self.save)
  155. self.refreshViewer()
  156. def openFile(self):
  157. # print("openFile")
  158. sumlist = self.parent.summary.list
  159. currentitem = sumlist.currentItem()
  160. if currentitem:
  161. if not self.changed:
  162. self.editor.textChanged.disconnect(self.onTextChanged)
  163. filename = currentitem.data['file']
  164. self.file = os.path.join(self.parent.core.cwd,'contents',filename)
  165. self.editor.clear()
  166. self.editor.insertPlainText(open(self.file, 'r').read())
  167. self.refreshViewer()
  168. self.editor.textChanged.connect(self.onTextChanged)
  169. else:
  170. print("Can't changed file, current id modified, please save first")
  171. # TODO: ask for saving current file
  172. def onTextChanged(self):
  173. self.refreshViewer()
  174. if not self.changed:
  175. self.changed = True
  176. # TODO: show in list that content needs to be saved
  177. # i = self.tabs.currentIndex()
  178. # self.tabs.setTabText(i, "* "+self.tabs.tabText(i))
  179. def refreshViewer(self):
  180. markdown = self.editor.toPlainText()
  181. html = markdown2.markdown(markdown)
  182. self.viewer.setHtml(html)
  183. def save(self):
  184. if self.changed:
  185. open(self.file, 'w').write(self.editor.toPlainText())
  186. self.changed = False
  187. # i = self.tabs.currentIndex()
  188. # self.tabs.setTabText(i, re.sub(r'^\*\s', '', self.tabs.tabText(i)))
  189. # TODO: how to combine file save and project save
  190. # __ ___ _
  191. # / |/ /___ _(_)___
  192. # / /|_/ / __ `/ / __ \
  193. # / / / / /_/ / / / / /
  194. # /_/ /_/\__,_/_/_/ /_/
  195. class ContentStack(QWidget):
  196. def __init__(self, core):
  197. super(ContentStack, self).__init__()
  198. self.core = core
  199. hbox = QHBoxLayout()
  200. hbox.setContentsMargins(0,0,0,0)
  201. self.setLayout(hbox)
  202. self.hsplitter = QSplitter(QtCore.Qt.Horizontal)
  203. self.summary = Summary(self)
  204. # TODO: detect external changes (file changed or new file)
  205. self.hsplitter.addWidget(self.summary)
  206. self.editor = MarkdownEditor(self)
  207. self.hsplitter.addWidget(self.editor)
  208. self.hsplitter.splitterMoved.connect(self.movedSplitter)
  209. hbox.addWidget(self.hsplitter)
  210. self.restorePrefs()
  211. def restorePrefs(self):
  212. settings = QSettings('FiguresLibres', 'Cascade')
  213. vals = settings.value('content/hsplitter/sizes', None)
  214. if vals:
  215. sizes = []
  216. for size in vals: sizes.append(int(size))
  217. self.hsplitter.setSizes(sizes)
  218. def movedSplitter(self):
  219. settings = QSettings('FiguresLibres', 'Cascade')
  220. # print(self.hsplitter.sizes())
  221. settings.setValue('content/hsplitter/sizes', self.hsplitter.sizes())