highlighter.py 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. # @Author: Bachir Soussi Chiadmi <bach>
  4. # @Date: 02-06-2017
  5. # @Email: bachir@figureslibres.io
  6. # @Filename: highlighter.py
  7. # @Last modified by: bach
  8. # @Last modified time: 03-06-2017
  9. # @License: GPL-V3
  10. # TODO add link to the source
  11. import sys
  12. import re
  13. from PyQt5 import QtCore, QtGui
  14. from pygments import highlight
  15. from pygments.lexers import *
  16. from pygments.formatter import Formatter
  17. from pygments.styles import get_all_styles, get_style_by_name
  18. import time
  19. # Copyright (C) 2008 Christophe Kibleur <kib2@free.fr>
  20. #
  21. # This file is part of WikiParser (http://thewikiblog.appspot.com/).
  22. #
  23. def hex2QColor(c):
  24. r=int(c[0:2],16)
  25. g=int(c[2:4],16)
  26. b=int(c[4:6],16)
  27. return QtGui.QColor(r,g,b)
  28. class QFormatter(Formatter):
  29. def __init__(self, linenos=True, style="default"): #, style
  30. Formatter.__init__(self)
  31. self.data=[]
  32. self.style = get_style_by_name(style)
  33. # self.linenos = linenos
  34. # styles = list(get_all_styles())
  35. # print(styles)
  36. # Create a dictionary of text styles, indexed
  37. # by pygments token names, containing QTextCharFormat
  38. # instances according to pygments' description
  39. # of each style
  40. self.styles={}
  41. for token, style in self.style:
  42. qtf=QtGui.QTextCharFormat()
  43. if style['color']:
  44. qtf.setForeground(hex2QColor(style['color']))
  45. if style['bgcolor']:
  46. qtf.setBackground(hex2QColor(style['bgcolor']))
  47. if style['bold']:
  48. qtf.setFontWeight(QtGui.QFont.Bold)
  49. if style['italic']:
  50. qtf.setFontItalic(True)
  51. if style['underline']:
  52. qtf.setFontUnderline(True)
  53. self.styles[str(token)]=qtf
  54. def format(self, tokensource, outfile):
  55. global styles
  56. # We ignore outfile, keep output in a buffer
  57. self.data=[]
  58. # Just store a list of styles, one for each character
  59. # in the input. Obviously a smarter thing with
  60. # offsets and lengths is a good idea!
  61. print(tokensource)
  62. for ttype, value in tokensource:
  63. l=len(value)
  64. t=str(ttype)
  65. self.data.extend([self.styles[t],]*l)
  66. class Highlighter(QtGui.QSyntaxHighlighter):
  67. def __init__(self, parent, mode):
  68. QtGui.QSyntaxHighlighter.__init__(self, parent)
  69. self.tstamp=time.time()
  70. # Keep the formatter and lexer, initializing them
  71. # may be costly.
  72. if not mode == "md":
  73. self.formatter=QFormatter(linenos=True, style="monokai-hcb")
  74. else:
  75. self.formatter=QFormatter(linenos=False, style="github")
  76. self.lexer=get_lexer_by_name(mode)
  77. def highlightBlock(self, text):
  78. """Takes a block, applies format to the document.
  79. according to what's in it.
  80. """
  81. # I need to know where in the document we are,
  82. # because our formatting info is global to
  83. # the document
  84. cb = self.currentBlock()
  85. p = cb.position()
  86. # The \n is not really needed, but sometimes
  87. # you are in an empty last block, so your position is
  88. # **after** the end of the document.
  89. text=str(self.document().toPlainText())+'\n'
  90. # Yes, re-highlight the whole document.
  91. # There **must** be some optimizacion possibilities
  92. # but it seems fast enough.
  93. highlight(text,self.lexer,self.formatter)
  94. # Just apply the formatting to this block.
  95. # For titles, it may be necessary to backtrack
  96. # and format a couple of blocks **earlier**.
  97. for i in range(len(str(text))):
  98. try:
  99. self.setFormat(i,1,self.formatter.data[p+i])
  100. except IndexError:
  101. pass
  102. # I may need to do something about this being called
  103. # too quickly.
  104. self.tstamp=time.time()