content.py 8.4 KB

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