915 lines
39 KiB
Python
915 lines
39 KiB
Python
import sublime
|
|
import sublime_plugin
|
|
from os import path
|
|
import tempfile
|
|
import sys
|
|
import time
|
|
import webbrowser
|
|
import re
|
|
from HtmlAnnotations import get_annotations
|
|
import ExportHtmlLib.desktop as desktop
|
|
import json
|
|
|
|
PACKAGE_SETTINGS = "ExportHtml.sublime-settings"
|
|
JS_DIR = path.join(sublime.packages_path(), 'ExportHtml', "js")
|
|
CSS_DIR = path.join(sublime.packages_path(), 'ExportHtml', "css")
|
|
|
|
if sublime.platform() == "linux":
|
|
# Try and load Linux Python2.6 lib. Default path is for Ubuntu.
|
|
linux_lib = sublime.load_settings(PACKAGE_SETTINGS).get("linux_python2.6_lib", "/usr/lib/python2.6/lib-dynload")
|
|
if not linux_lib in sys.path and path.exists(linux_lib):
|
|
sys.path.append(linux_lib)
|
|
from plistlib import readPlist
|
|
from ExportHtmlLib.rgba.rgba import RGBA
|
|
|
|
FILTER_MATCH = re.compile(r'^(?:(brightness|saturation|hue|colorize)\((-?[\d]+|[\d]*\.[\d]+)\)|(sepia|grayscale|invert))$')
|
|
|
|
# HTML Code
|
|
HTML_HEADER = \
|
|
'''<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
|
|
<html>
|
|
<head>
|
|
<title>%(title)s</title>
|
|
<meta http-equiv="Cache-Control" content="no-cache, no-store, must-revalidate" />
|
|
<meta http-equiv="Pragma" content="no-cache" />
|
|
<meta http-equiv="Expires" content="0" />
|
|
<style type="text/css">
|
|
%(css)s
|
|
</style>
|
|
%(js)s
|
|
</head>
|
|
'''
|
|
|
|
TOOL_GUTTER = '''<img onclick="toggle_gutter();" alt="" title="Toggle Gutter" src="" />'''
|
|
|
|
TOOL_PLAIN_TEXT = '''<img onclick="toggle_plain_text();" alt="" title="Toggle Plain" src="" />'''
|
|
|
|
TOOL_PRINT = '''<img onclick="page_print();" alt="" title="Print" src="" />'''
|
|
|
|
TOOL_ANNOTATION = '''<img onclick="toggle_annotations();" alt="" title="Toggle Annotations" src="" />'''
|
|
|
|
TOOL_DUMP_THEME = '''<img onclick="dump_theme();" alt="" title="Download" src="" />'''
|
|
|
|
TOOL_WRAPPING = '''<img onclick="toggle_wrapping();" alt="" title="Toggle Wrapping" src="" />'''
|
|
|
|
TOOLBAR = '''<div id="toolbarhide"><div id="toolbar">%(options)s</div></div>'''
|
|
|
|
ANNOTATE_OPEN = '''<span onclick="toggle_annotations();" class="tooltip_hotspot" onmouseover="tooltip.show(%(comment)s);" onmouseout="tooltip.hide();">%(code)s'''
|
|
|
|
ANNOTATE_CLOSE = '''</span>'''
|
|
|
|
BODY_START = '''<body class="code_page code_text"><pre class="code_page">'''
|
|
|
|
FILE_INFO = '''<tr><td colspan="2" style="background: %(bgcolor)s"><div id="file_info"><span style="color: %(color)s">%(date_time)s %(file)s\n\n</span></div></td></tr>'''
|
|
|
|
TABLE_START = '''<table cellspacing="0" cellpadding="0" class="code_page">'''
|
|
|
|
LINE = (
|
|
'<tr>' +
|
|
'<td valign="top" id="L_%(table)d_%(line_id)d" class="code_text code_gutter" style="background: %(bgcolor)s">' +
|
|
'<span style="color: %(color)s;">%(line)s </span>' +
|
|
'</td>' +
|
|
'<td valign="top" class="code_text code_line" style="background-color: %(pad_color)s;">' +
|
|
'<div id="C_%(table)d_%(code_id)d">%(code)s\n</div>' +
|
|
'</td>' +
|
|
'</tr>'
|
|
)
|
|
|
|
CODE = '''<span class="%(class)s" style="background-color: %(highlight)s; color: %(color)s;">%(content)s</span>'''
|
|
ANNOTATION_CODE = '''<span style="background-color: %(highlight)s;"><a href="javascript:void();" class="annotation"><span class="%(class)s annotation" style="color: %(color)s;">%(content)s</span></a></span>'''
|
|
|
|
TABLE_END = '''</table>'''
|
|
|
|
ROW_START = '''<tr><td>'''
|
|
|
|
ROW_END = '''</td></tr>'''
|
|
|
|
DIVIDER = '''<span style="color: %(color)s">\n...\n\n</span>'''
|
|
|
|
ANNOTATION_TBL_START = (
|
|
'<div id="comment_list" style="display:none"><div id="comment_wrapper">' +
|
|
'<table id="comment_table">' +
|
|
'<tr><th>Line/Col</th><th>Comments' +
|
|
'<a href="javascript:void(0)" class="table_close" onclick="toggle_annotations();return false;">(close)</a>'
|
|
'</th></tr>'
|
|
)
|
|
|
|
ANNOTATION_TBL_END = '''</table></div></div>'''
|
|
|
|
ANNOTATION_ROW = (
|
|
'<tr>' +
|
|
'<td class="annotation_link">' +
|
|
'<a href="javascript:void(0)" onclick="scroll_to_line(\'C_%(table)d_%(row)d\');return false;">%(link)s</a>' +
|
|
'</td>' +
|
|
'<td class="annotation_comment"><div class="annotation_comment">%(comment)s</div></td>' +
|
|
'<tr>'
|
|
)
|
|
|
|
ANNOTATION_FOOTER = (
|
|
'<tr><td colspan=2>' +
|
|
'<div class="table_footer"><label>Position </label>' +
|
|
'<select id="dock" size="1" onchange="dock_table();">' +
|
|
'<option value="0" selected="selected">center</option>' +
|
|
'<option value="1">top</option>' +
|
|
'<option value="2">bottom</option>' +
|
|
'<option value="3">left</option>' +
|
|
'<option value="4">right</option>' +
|
|
'<option value="5">top left</option>' +
|
|
'<option value="6">top right</option>' +
|
|
'<option value="7">bottom left</option>' +
|
|
'<option value="8">bottom right</option>' +
|
|
'</select>' +
|
|
'</div>' +
|
|
'</td></tr>'
|
|
)
|
|
|
|
BODY_END = '''</pre>%(toolbar)s\n%(js)s\n</body>\n</html>\n'''
|
|
|
|
INCLUDE_THEME = \
|
|
'''
|
|
<script type="text/javascript">
|
|
%(jscode)s
|
|
plist.color_scheme = %(theme)s;
|
|
|
|
function dump_theme() {
|
|
extract_theme('%(name)s');
|
|
}
|
|
</script>
|
|
'''
|
|
|
|
TOGGLE_LINE_OPTIONS = \
|
|
'''
|
|
<script type="text/javascript">
|
|
%(jscode)s
|
|
|
|
page_line_info.wrap = false;
|
|
page_line_info.ranges = [%(ranges)s];
|
|
page_line_info.wrap_size = %(wrap_size)d;
|
|
page_line_info.tables = %(tables)s;
|
|
page_line_info.header = %(header)s;
|
|
page_line_info.gutter = %(gutter)s;
|
|
</script>
|
|
'''
|
|
|
|
AUTO_PRINT = \
|
|
'''
|
|
<script type="text/javascript">
|
|
document.getElementsByTagName('body')[0].onload = function (e) { page_print(); self.onload = null; };
|
|
</script>
|
|
'''
|
|
|
|
WRAP = \
|
|
'''
|
|
<script type="text/javascript">
|
|
toggle_wrapping();
|
|
</script>
|
|
'''
|
|
|
|
HTML_JS_WRAP = \
|
|
'''
|
|
<script type="text/javascript">
|
|
%(jscode)s
|
|
</script>
|
|
'''
|
|
|
|
|
|
def getjs(file_name):
|
|
code = ""
|
|
try:
|
|
with open(path.join(JS_DIR, file_name), "r") as f:
|
|
code = f.read()
|
|
except:
|
|
pass
|
|
return code
|
|
|
|
|
|
def getcss(file_name, options):
|
|
code = ""
|
|
final_code = ""
|
|
last_pt = 0
|
|
keys = '|'.join(options.keys())
|
|
replace = re.compile("/\\* *%(" + keys + ")% * \\*/")
|
|
|
|
try:
|
|
with open(path.join(CSS_DIR, file_name), "r") as f:
|
|
code = f.read()
|
|
for m in replace.finditer(code):
|
|
final_code += code[last_pt:m.start()] + options[m.group(1)]
|
|
last_pt = m.end()
|
|
final_code += code[last_pt:]
|
|
except:
|
|
pass
|
|
|
|
return final_code
|
|
|
|
|
|
class ExportHtmlPanelCommand(sublime_plugin.WindowCommand):
|
|
def execute(self, value):
|
|
if value >= 0:
|
|
view = self.window.active_view()
|
|
if view != None:
|
|
ExportHtml(view).run(**self.args[value])
|
|
|
|
def run(self):
|
|
options = sublime.load_settings(PACKAGE_SETTINGS).get("html_panel", {})
|
|
menu = []
|
|
self.args = []
|
|
for opt in options:
|
|
k, v = opt.items()[0]
|
|
menu.append(k)
|
|
self.args.append(v)
|
|
|
|
if len(menu):
|
|
self.window.show_quick_panel(
|
|
menu,
|
|
self.execute
|
|
)
|
|
|
|
|
|
class ExportHtmlCommand(sublime_plugin.WindowCommand):
|
|
def run(self, **kwargs):
|
|
view = self.window.active_view()
|
|
if view != None:
|
|
ExportHtml(view).run(**kwargs)
|
|
|
|
|
|
class ExportHtml(object):
|
|
def __init__(self, view):
|
|
self.view = view
|
|
|
|
def process_inputs(self, **kwargs):
|
|
return {
|
|
"numbers": bool(kwargs.get("numbers", False)),
|
|
"highlight_selections": bool(kwargs.get("highlight_selections", False)),
|
|
"browser_print": bool(kwargs.get("browser_print", False)),
|
|
"color_scheme": kwargs.get("color_scheme", None),
|
|
"wrap": kwargs.get("wrap", None),
|
|
"multi_select": bool(kwargs.get("multi_select", False)),
|
|
"style_gutter": bool(kwargs.get("style_gutter", True)),
|
|
"no_header": bool(kwargs.get("no_header", False)),
|
|
"date_time_format": kwargs.get("date_time_format", "%m/%d/%y %I:%M:%S"),
|
|
"show_full_path": bool(kwargs.get("show_full_path", True)),
|
|
"toolbar": kwargs.get("toolbar", ["plain_text", "gutter", "wrapping", "print", "annotation", "theme"]),
|
|
"save_location": kwargs.get("save_location", None),
|
|
"time_stamp": kwargs.get("time_stamp", "_%m%d%y%H%M%S"),
|
|
"clipboard_copy": bool(kwargs.get("clipboard_copy", False)),
|
|
"view_open": bool(kwargs.get("view_open", False)),
|
|
"shift_brightness": bool(kwargs.get("shift_brightness", False)),
|
|
"filter": kwargs.get("filter", "")
|
|
}
|
|
|
|
def setup(self, **kwargs):
|
|
path_packages = sublime.packages_path()
|
|
|
|
# Get get general document preferences from sublime preferences
|
|
eh_settings = sublime.load_settings(PACKAGE_SETTINGS)
|
|
settings = sublime.load_settings('Preferences.sublime-settings')
|
|
alternate_font_size = eh_settings.get("alternate_font_size", False)
|
|
alternate_font_face = eh_settings.get("alternate_font_face", False)
|
|
self.font_size = settings.get('font_size', 10) if alternate_font_size == False else alternate_font_size
|
|
self.font_face = settings.get('font_face', 'Consolas') if alternate_font_face == False else alternate_font_face
|
|
self.tab_size = settings.get('tab_size', 4)
|
|
self.padd_top = settings.get('line_padding_top', 0)
|
|
self.padd_bottom = settings.get('line_padding_bottom', 0)
|
|
self.char_limit = int(eh_settings.get("valid_selection_size", 4))
|
|
self.bground = ''
|
|
self.fground = ''
|
|
self.gbground = ''
|
|
self.gfground = ''
|
|
self.sbground = ''
|
|
self.sfground = ''
|
|
self.numbers = kwargs["numbers"]
|
|
self.date_time_format = kwargs["date_time_format"]
|
|
self.time = time.localtime()
|
|
self.show_full_path = kwargs["show_full_path"]
|
|
self.highlight_selections = kwargs["highlight_selections"]
|
|
self.browser_print = kwargs["browser_print"]
|
|
self.auto_wrap = kwargs["wrap"] != None and int(kwargs["wrap"]) > 0
|
|
self.wrap = 900 if not self.auto_wrap else int(kwargs["wrap"])
|
|
self.hl_continue = None
|
|
self.curr_hl = None
|
|
self.sels = []
|
|
self.multi_select = self.check_sel() if kwargs["multi_select"] and not kwargs["highlight_selections"] else False
|
|
self.size = self.view.size()
|
|
self.pt = 0
|
|
self.end = 0
|
|
self.curr_row = 0
|
|
self.tables = 0
|
|
self.curr_annot = None
|
|
self.curr_comment = None
|
|
self.annotations = self.get_annotations()
|
|
self.annot_num = -1
|
|
self.new_annot = False
|
|
self.open_annot = False
|
|
self.no_header = kwargs["no_header"]
|
|
self.annot_tbl = []
|
|
self.toolbar = kwargs["toolbar"]
|
|
self.toolbar_orientation = "block" if eh_settings.get("toolbar_orientation", "horizontal") == "vertical" else "inline-block"
|
|
self.matched = {}
|
|
self.ebground = self.bground
|
|
self.dark_lumens = None
|
|
self.lumens_limit = float(eh_settings.get("bg_min_lumen_threshold", 62))
|
|
self.filter = []
|
|
for f in kwargs["filter"].split(";"):
|
|
m = FILTER_MATCH.match(f)
|
|
if m:
|
|
if m.group(1):
|
|
self.filter.append((m.group(1), float(m.group(2))))
|
|
else:
|
|
self.filter.append((m.group(3), 0.0))
|
|
|
|
fname = self.view.file_name()
|
|
if fname == None or not path.exists(fname):
|
|
fname = "Untitled"
|
|
self.file_name = fname
|
|
|
|
# Get color scheme
|
|
if kwargs["color_scheme"] != None:
|
|
alt_scheme = kwargs["color_scheme"]
|
|
else:
|
|
alt_scheme = eh_settings.get("alternate_scheme", False)
|
|
scheme_file = settings.get('color_scheme') if alt_scheme == False else alt_scheme
|
|
colour_scheme = path.normpath(scheme_file)
|
|
self.scheme_file = path.basename(colour_scheme)
|
|
self.plist_file = self.apply_filters(readPlist(path_packages + colour_scheme.replace('Packages', '')))
|
|
colour_settings = self.plist_file["settings"][0]["settings"]
|
|
|
|
# Get general theme colors from color scheme file
|
|
self.bground = self.strip_transparency(colour_settings.get("background", '#FFFFFF'), True, True)
|
|
self.fground = self.strip_transparency(colour_settings.get("foreground", '#000000'))
|
|
self.sbground = self.strip_transparency(colour_settings.get("selection", self.fground), True)
|
|
self.sfground = self.strip_transparency(colour_settings.get("selectionForeground", None))
|
|
self.gbground = self.strip_transparency(colour_settings.get("gutter", self.bground)) if kwargs["style_gutter"] else self.bground
|
|
self.gfground = self.strip_transparency(colour_settings.get("gutterForeground", self.fground), True) if kwargs["style_gutter"] else self.fground
|
|
|
|
self.highlights = []
|
|
if self.highlight_selections:
|
|
for sel in self.view.sel():
|
|
if not sel.empty():
|
|
self.highlights.append(sel)
|
|
|
|
# Create scope colors mapping from color scheme file
|
|
self.colours = {self.view.scope_name(self.end).split(' ')[0]: {"color": self.fground, "bgcolor": None, "style": None}}
|
|
for item in self.plist_file["settings"]:
|
|
scope = item.get('scope', None)
|
|
colour = None
|
|
style = []
|
|
if 'scope' in item:
|
|
scope = item['scope']
|
|
if 'settings' in item:
|
|
colour = item['settings'].get('foreground', None)
|
|
bgcolour = item['settings'].get('background', None)
|
|
if 'fontStyle' in item['settings']:
|
|
for s in item['settings']['fontStyle'].split(' '):
|
|
if s == "bold" or s == "italic": # or s == "underline":
|
|
style.append(s)
|
|
|
|
if scope != None and (colour != None or bgcolour != None):
|
|
self.colours[scope] = {
|
|
"color": self.strip_transparency(colour),
|
|
"bgcolor": self.strip_transparency(bgcolour, True),
|
|
"style": style
|
|
}
|
|
|
|
self.shift_brightness = kwargs["shift_brightness"] and self.dark_lumens is not None and self.dark_lumens < self.lumens_limit
|
|
if self.shift_brightness:
|
|
self.color_adjust()
|
|
|
|
def apply_filters(self, tmtheme):
|
|
def filter_color(color):
|
|
rgba = RGBA(color)
|
|
for f in self.filter:
|
|
name = f[0]
|
|
value = f[1]
|
|
if name == "grayscale":
|
|
rgba.grayscale()
|
|
elif name == "sepia":
|
|
rgba.sepia()
|
|
elif name == "saturation":
|
|
rgba.saturation(value)
|
|
elif name == "invert":
|
|
rgba.invert()
|
|
elif name == "brightness":
|
|
rgba.brightness(value)
|
|
elif name == "hue":
|
|
rgba.hue(value)
|
|
elif name == "colorize":
|
|
rgba.colorize(value)
|
|
return rgba.get_rgba()
|
|
|
|
if len(self.filter):
|
|
general_settings_read = False
|
|
for settings in tmtheme["settings"]:
|
|
if not general_settings_read:
|
|
for k, v in settings["settings"].items():
|
|
try:
|
|
settings["settings"][k] = filter_color(v)
|
|
except:
|
|
pass
|
|
general_settings_read = True
|
|
continue
|
|
|
|
try:
|
|
settings["settings"]["foreground"] = filter_color(settings["settings"]["foreground"])
|
|
except:
|
|
pass
|
|
try:
|
|
settings["settings"]["background"] = filter_color(settings["settings"]["background"])
|
|
except:
|
|
pass
|
|
return tmtheme
|
|
|
|
def color_adjust(self):
|
|
factor = 1 + ((self.lumens_limit - self.dark_lumens) / 255.0) if self.shift_brightness else None
|
|
for k, v in self.colours.items():
|
|
fg, bg = v["color"], v["bgcolor"]
|
|
if v["color"] is not None:
|
|
self.colours[k]["color"] = self.apply_color_change(v["color"], factor)
|
|
if v["bgcolor"] is not None:
|
|
self.colours[k]["bgcolor"] = self.apply_color_change(v["bgcolor"], factor)
|
|
self.bground = self.apply_color_change(self.bground, factor)
|
|
self.fground = self.apply_color_change(self.fground, factor)
|
|
self.sbground = self.apply_color_change(self.sbground, factor)
|
|
if self.sfground is not None:
|
|
self.sfground = self.apply_color_change(self.sfground, factor)
|
|
self.gbground = self.apply_color_change(self.gbground, factor)
|
|
self.gfground = self.apply_color_change(self.gfground, factor)
|
|
|
|
def apply_color_change(self, color, shift_factor):
|
|
rgba = RGBA(color)
|
|
if shift_factor is not None:
|
|
rgba.brightness(shift_factor)
|
|
return rgba.get_rgb()
|
|
|
|
def get_tools(self, tools, use_annotation, use_wrapping):
|
|
toolbar_options = {
|
|
"gutter": TOOL_GUTTER,
|
|
"print": TOOL_PRINT,
|
|
"plain_text": TOOL_PLAIN_TEXT,
|
|
"annotation": TOOL_ANNOTATION if use_annotation else "",
|
|
"theme": TOOL_DUMP_THEME,
|
|
"wrapping": TOOL_WRAPPING if use_wrapping else ""
|
|
}
|
|
t_opt = ""
|
|
toolbar_element = ""
|
|
|
|
if len(tools):
|
|
for t in tools:
|
|
if t in toolbar_options:
|
|
t_opt += toolbar_options[t]
|
|
toolbar_element = TOOLBAR % {"options": t_opt}
|
|
return toolbar_element
|
|
|
|
def strip_transparency(self, color, track_darkness=False, simple_strip=False):
|
|
if color is None:
|
|
return color
|
|
rgba = RGBA(color.replace(" ", ""))
|
|
if not simple_strip:
|
|
rgba.apply_alpha(self.bground if self.bground != "" else "#FFFFFF")
|
|
if track_darkness:
|
|
lumens = rgba.luminance()
|
|
if self.dark_lumens is None or lumens < self.dark_lumens:
|
|
self.dark_lumens = lumens
|
|
return rgba.get_rgb()
|
|
|
|
def setup_print_block(self, curr_sel, multi=False):
|
|
# Determine start and end points and whether to parse whole file or selection
|
|
if not multi and (curr_sel.empty() or self.highlight_selections or curr_sel.size() <= self.char_limit):
|
|
self.size = self.view.size()
|
|
self.pt = 0
|
|
self.end = 1
|
|
self.curr_row = 1
|
|
else:
|
|
self.size = curr_sel.end()
|
|
self.pt = curr_sel.begin()
|
|
self.end = self.pt + 1
|
|
self.curr_row = self.view.rowcol(self.pt)[0] + 1
|
|
self.start_line = self.curr_row
|
|
|
|
self.gutter_pad = len(str(self.view.rowcol(self.size)[0])) + 1
|
|
|
|
def check_sel(self):
|
|
multi = False
|
|
for sel in self.view.sel():
|
|
if not sel.empty() and sel.size() >= self.char_limit:
|
|
multi = True
|
|
self.sels.append(sel)
|
|
return multi
|
|
|
|
def print_line(self, line, num):
|
|
html_line = LINE % {
|
|
"line_id": num,
|
|
"color": self.gfground,
|
|
"bgcolor": self.gbground,
|
|
"line": str(num).rjust(self.gutter_pad).replace(" ", ' '),
|
|
"code_id": num,
|
|
"code": line,
|
|
"table": self.tables,
|
|
"pad_color": self.ebground or self.bground
|
|
}
|
|
|
|
return html_line
|
|
|
|
def guess_colour(self, pt, the_key):
|
|
the_colour = self.fground
|
|
the_bgcolour = None
|
|
the_style = set([])
|
|
if the_key in self.matched:
|
|
the_colour = self.matched[the_key]["color"]
|
|
the_style = self.matched[the_key]["style"]
|
|
the_bgcolour = self.matched[the_key]["bgcolor"]
|
|
else:
|
|
best_match_bg = 0
|
|
best_match_fg = 0
|
|
best_match_style = 0
|
|
for key in self.colours:
|
|
match = self.view.score_selector(pt, key)
|
|
if self.colours[key]["color"] is not None and match > best_match_fg:
|
|
best_match_fg = match
|
|
the_colour = self.colours[key]["color"]
|
|
if self.colours[key]["style"] is not None and match > best_match_style:
|
|
best_match_style = match
|
|
for s in self.colours[key]["style"]:
|
|
the_style.add(s)
|
|
if self.colours[key]["bgcolor"] is not None and match > best_match_bg:
|
|
best_match_bg = match
|
|
the_bgcolour = self.colours[key]["bgcolor"]
|
|
self.matched[the_key] = {"color": the_colour, "bgcolor": the_bgcolour, "style": the_style}
|
|
if len(the_style) == 0:
|
|
the_style = "normal"
|
|
else:
|
|
the_style = ' '.join(the_style)
|
|
return the_colour, the_style, the_bgcolour
|
|
|
|
def write_header(self, the_html):
|
|
header = HTML_HEADER % {
|
|
"title": path.basename(self.file_name),
|
|
"css": getcss(
|
|
'export.css',
|
|
{
|
|
"font_size": str(self.font_size),
|
|
"font_face": '"' + self.font_face + '"',
|
|
"page_bg": self.bground,
|
|
"gutter_bg": self.gbground,
|
|
"body_fg": self.fground,
|
|
"display_mode": 'table-cell' if self.numbers else 'none',
|
|
"dot_color": self.fground,
|
|
"toolbar_orientation": self.toolbar_orientation
|
|
}
|
|
),
|
|
"js": INCLUDE_THEME % {
|
|
"jscode": getjs('plist.js'),
|
|
"theme": json.dumps(self.plist_file, sort_keys=True, indent=4, separators=(',', ': ')).encode('raw_unicode_escape'),
|
|
"name": self.scheme_file,
|
|
}
|
|
}
|
|
the_html.write(header)
|
|
|
|
def convert_view_to_html(self, the_html):
|
|
for line in self.view.split_by_newlines(sublime.Region(self.pt, self.size)):
|
|
self.size = line.end()
|
|
empty = not bool(line.size())
|
|
line = self.convert_line_to_html(the_html, empty)
|
|
the_html.write(self.print_line(line, self.curr_row))
|
|
self.curr_row += 1
|
|
|
|
def html_encode(self, text):
|
|
# Format text to HTML
|
|
encode_table = {
|
|
'&': '&',
|
|
'>': '>',
|
|
'<': '<',
|
|
'\t': ' ' * self.tab_size,
|
|
'\n': ''
|
|
}
|
|
|
|
return re.sub(
|
|
r'(?!\s($|\S))\s',
|
|
' ',
|
|
''.join(
|
|
encode_table.get(c, c) for c in text
|
|
).encode('ascii', 'xmlcharrefreplace')
|
|
)
|
|
|
|
def get_annotations(self):
|
|
annotations = get_annotations(self.view)
|
|
comments = []
|
|
for x in range(0, int(annotations["count"])):
|
|
region = annotations["annotations"]["html_annotation_%d" % x]["region"]
|
|
comments.append((region, annotations["annotations"]["html_annotation_%d" % x]["comment"]))
|
|
comments.sort()
|
|
return comments
|
|
|
|
def annotate_text(self, line, the_colour, the_bgcolour, the_style, empty):
|
|
pre_text = None
|
|
annot_text = None
|
|
post_text = None
|
|
start = None
|
|
|
|
# Pretext Check
|
|
if self.pt >= self.curr_annot.begin():
|
|
# Region starts with an annotation
|
|
start = self.pt
|
|
else:
|
|
# Region has text before annoation
|
|
pre_text = self.html_encode(self.view.substr(sublime.Region(self.pt, self.curr_annot.begin())))
|
|
start = self.curr_annot.begin()
|
|
|
|
if self.end == self.curr_annot.end():
|
|
# Region ends annotation
|
|
annot_text = self.html_encode(self.view.substr(sublime.Region(start, self.end)))
|
|
self.curr_annot = None
|
|
elif self.end > self.curr_annot.end():
|
|
# Region has text following annotation
|
|
annot_text = self.html_encode(self.view.substr(sublime.Region(start, self.curr_annot.end())))
|
|
post_text = self.html_encode(self.view.substr(sublime.Region(self.curr_annot.end(), self.end)))
|
|
self.curr_annot = None
|
|
else:
|
|
# Region ends but annotation is not finished
|
|
annot_text = self.html_encode(self.view.substr(sublime.Region(start, self.end)))
|
|
self.curr_annot = sublime.Region(self.end, self.curr_annot.end())
|
|
|
|
# Print the separate parts pre text, annotation, post text
|
|
if pre_text != None:
|
|
self.format_text(line, pre_text, the_colour, the_bgcolour, the_style, empty)
|
|
if annot_text != None:
|
|
self.format_text(line, annot_text, the_colour, the_bgcolour, the_style, empty, annotate=True)
|
|
if self.curr_annot == None:
|
|
self.curr_comment = None
|
|
if post_text != None:
|
|
self.format_text(line, post_text, the_colour, the_bgcolour, the_style, empty)
|
|
|
|
def add_annotation_table_entry(self):
|
|
row, col = self.view.rowcol(self.annot_pt)
|
|
self.annot_tbl.append(
|
|
(
|
|
self.tables, self.curr_row, "Line %d Col %d" % (row + 1, col + 1),
|
|
self.curr_comment.encode('ascii', 'xmlcharrefreplace')
|
|
)
|
|
)
|
|
self.annot_pt = None
|
|
|
|
def format_text(self, line, text, the_colour, the_bgcolour, the_style, empty, annotate=False):
|
|
if empty:
|
|
text = ' '
|
|
else:
|
|
the_style += " real_text"
|
|
|
|
if the_bgcolour is None:
|
|
the_bgcolour = self.bground
|
|
|
|
if annotate:
|
|
code = ANNOTATION_CODE % {"highlight": the_bgcolour, "color": the_colour, "content": text, "class": the_style}
|
|
else:
|
|
code = CODE % {"highlight": the_bgcolour, "color": the_colour, "content": text, "class": the_style}
|
|
|
|
if annotate:
|
|
if self.curr_annot != None and not self.open_annot:
|
|
# Open an annotation
|
|
if self.annot_pt != None:
|
|
self.add_annotation_table_entry()
|
|
if self.new_annot:
|
|
self.annot_num += 1
|
|
self.new_annot = False
|
|
code = ANNOTATE_OPEN % {"code": code, "comment": str(self.annot_num)}
|
|
self.open_annot = True
|
|
elif self.curr_annot == None:
|
|
if self.open_annot:
|
|
# Close an annotation
|
|
code += ANNOTATE_CLOSE
|
|
self.open_annot = False
|
|
else:
|
|
# Do a complete annotation
|
|
if self.annot_pt != None:
|
|
self.add_annotation_table_entry()
|
|
if self.new_annot:
|
|
self.annot_num += 1
|
|
self.new_annot = False
|
|
code = (
|
|
ANNOTATE_OPEN % {"code": code, "comment": str(self.annot_num)} +
|
|
ANNOTATE_CLOSE
|
|
)
|
|
line.append(code)
|
|
|
|
def convert_line_to_html(self, the_html, empty):
|
|
line = []
|
|
hl_done = False
|
|
|
|
# Continue highlight form last line
|
|
if self.hl_continue != None:
|
|
self.curr_hl = self.hl_continue
|
|
self.hl_continue = None
|
|
|
|
while self.end <= self.size:
|
|
# Get next highlight region
|
|
if self.highlight_selections and self.curr_hl == None and len(self.highlights) > 0:
|
|
self.curr_hl = self.highlights.pop(0)
|
|
|
|
# See if we are starting a highlight region
|
|
if self.curr_hl != None and self.pt == self.curr_hl.begin():
|
|
# Get text of like scope up to a highlight
|
|
scope_name = self.view.scope_name(self.pt)
|
|
while self.view.scope_name(self.end) == scope_name and self.end < self.size:
|
|
# Kick out if we hit a highlight region
|
|
if self.end == self.curr_hl.end():
|
|
break
|
|
self.end += 1
|
|
if self.end < self.curr_hl.end():
|
|
if self.end >= self.size:
|
|
self.hl_continue = sublime.Region(self.end, self.curr_hl.end())
|
|
else:
|
|
self.curr_hl = sublime.Region(self.end, self.curr_hl.end())
|
|
else:
|
|
hl_done = True
|
|
if hl_done and empty:
|
|
the_colour, the_style, the_bgcolour = self.guess_colour(self.pt, scope_name)
|
|
elif self.sfground is None:
|
|
the_colour, the_style, _ = self.guess_colour(self.pt, scope_name)
|
|
the_bgcolour = self.sbground
|
|
else:
|
|
the_colour, the_style = self.sfground, "normal"
|
|
the_bgcolour = self.sbground
|
|
else:
|
|
# Get text of like scope up to a highlight
|
|
scope_name = self.view.scope_name(self.pt)
|
|
while self.view.scope_name(self.end) == scope_name and self.end < self.size:
|
|
# Kick out if we hit a highlight region
|
|
if self.curr_hl != None and self.end == self.curr_hl.begin():
|
|
break
|
|
self.end += 1
|
|
the_colour, the_style, the_bgcolour = self.guess_colour(self.pt, scope_name)
|
|
|
|
# Get new annotation
|
|
if (self.curr_annot == None or self.curr_annot.end() < self.pt) and len(self.annotations):
|
|
self.curr_annot, self.curr_comment = self.annotations.pop(0)
|
|
self.annot_pt = self.curr_annot[0]
|
|
while self.pt > self.curr_annot[1]:
|
|
if len(self.annotations):
|
|
self.curr_annot, self.curr_comment = self.annotations.pop(0)
|
|
self.annot_pt = self.curr_annot[0]
|
|
else:
|
|
self.curr_annot = None
|
|
self.curr_comment = None
|
|
break
|
|
self.new_annot = True
|
|
self.curr_annot = sublime.Region(self.curr_annot[0], self.curr_annot[1])
|
|
|
|
region = sublime.Region(self.pt, self.end)
|
|
if self.curr_annot != None and region.intersects(self.curr_annot):
|
|
# Apply annotation within the text and format the text
|
|
self.annotate_text(line, the_colour, the_bgcolour, the_style, empty)
|
|
else:
|
|
# Normal text formatting
|
|
tidied_text = self.html_encode(self.view.substr(region))
|
|
self.format_text(line, tidied_text, the_colour, the_bgcolour, the_style, empty)
|
|
|
|
if hl_done:
|
|
# Clear highlight flags and variables
|
|
hl_done = False
|
|
self.curr_hl = None
|
|
|
|
# Continue walking through line
|
|
self.pt = self.end
|
|
self.end = self.pt + 1
|
|
|
|
# Close annotation if open at end of line
|
|
if self.open_annot:
|
|
line.append(ANNOTATE_CLOSE % {"comment": self.curr_comment})
|
|
self.open_annot = False
|
|
|
|
# Get the color for the space at the end of a line
|
|
if self.end < self.view.size():
|
|
end_key = self.view.scope_name(self.pt)
|
|
_, _, self.ebground = self.guess_colour(self.pt, end_key)
|
|
|
|
# Join line segments
|
|
return ''.join(line)
|
|
|
|
def write_body(self, the_html):
|
|
processed_rows = ""
|
|
the_html.write(BODY_START)
|
|
|
|
the_html.write(TABLE_START)
|
|
if not self.no_header:
|
|
# Write file name
|
|
date_time = time.strftime(self.date_time_format, self.time)
|
|
the_html.write(
|
|
FILE_INFO % {
|
|
"bgcolor": self.bground,
|
|
"color": self.fground,
|
|
"date_time": date_time,
|
|
"file": self.file_name if self.show_full_path else path.basename(self.file_name)
|
|
}
|
|
)
|
|
|
|
the_html.write(ROW_START)
|
|
the_html.write(TABLE_START)
|
|
# Convert view to HTML
|
|
if self.multi_select:
|
|
count = 0
|
|
total = len(self.sels)
|
|
for sel in self.sels:
|
|
self.setup_print_block(sel, multi=True)
|
|
processed_rows += "[" + str(self.curr_row) + ","
|
|
self.convert_view_to_html(the_html)
|
|
count += 1
|
|
self.tables = count
|
|
processed_rows += str(self.curr_row) + "],"
|
|
|
|
if count < total:
|
|
the_html.write(TABLE_END)
|
|
the_html.write(ROW_END)
|
|
the_html.write(ROW_START)
|
|
the_html.write(DIVIDER % {"color": self.fground})
|
|
the_html.write(ROW_END)
|
|
the_html.write(ROW_START)
|
|
the_html.write(TABLE_START)
|
|
else:
|
|
self.setup_print_block(self.view.sel()[0])
|
|
processed_rows += "[" + str(self.curr_row) + ","
|
|
self.convert_view_to_html(the_html)
|
|
processed_rows += str(self.curr_row) + "],"
|
|
self.tables += 1
|
|
|
|
the_html.write(TABLE_END)
|
|
the_html.write(ROW_END)
|
|
the_html.write(TABLE_END)
|
|
|
|
js_options = []
|
|
if len(self.annot_tbl):
|
|
self.add_comments_table(the_html)
|
|
js_options.append(HTML_JS_WRAP % {"jscode": getjs('annotation.js')})
|
|
|
|
# Write javascript snippets
|
|
js_options.append(HTML_JS_WRAP % {"jscode": getjs('print.js')})
|
|
js_options.append(HTML_JS_WRAP % {"jscode": getjs('plaintext.js')})
|
|
js_options.append(TOGGLE_LINE_OPTIONS % {
|
|
"jscode": getjs('lines.js'),
|
|
"wrap_size": self.wrap,
|
|
"ranges": processed_rows.rstrip(','),
|
|
"tables": self.tables,
|
|
"header": ("false" if self.no_header else "true"),
|
|
"gutter": ('true' if self.numbers else 'false')
|
|
}
|
|
)
|
|
if self.auto_wrap:
|
|
js_options.append(WRAP)
|
|
|
|
if self.browser_print:
|
|
js_options.append(AUTO_PRINT)
|
|
|
|
# Write empty line to allow copying of last line and line number without issue
|
|
the_html.write(BODY_END % {"js": ''.join(js_options), "toolbar": self.get_tools(self.toolbar, len(self.annot_tbl), self.auto_wrap)})
|
|
|
|
def add_comments_table(self, the_html):
|
|
the_html.write(ANNOTATION_TBL_START)
|
|
the_html.write(''.join([ANNOTATION_ROW % {"table": t, "row": r, "link": l, "comment": c} for t, r, l, c in self.annot_tbl]))
|
|
the_html.write(ANNOTATION_FOOTER)
|
|
the_html.write(ANNOTATION_TBL_END)
|
|
|
|
def run(self, **kwargs):
|
|
inputs = self.process_inputs(**kwargs)
|
|
self.setup(**inputs)
|
|
|
|
save_location = inputs["save_location"]
|
|
time_stamp = inputs["time_stamp"]
|
|
|
|
if save_location is not None:
|
|
fname = self.view.file_name()
|
|
if (
|
|
((fname == None or not path.exists(fname)) and save_location == ".") or
|
|
not path.exists(save_location)
|
|
or not path.isdir(save_location)
|
|
):
|
|
html_file = ".html"
|
|
save_location = None
|
|
elif save_location == ".":
|
|
html_file = "%s%s.html" % (fname, time.strftime(time_stamp, self.time))
|
|
elif fname is None or not path.exists(fname):
|
|
html_file = path.join(save_location, "Untitled%s.html" % time.strftime(time_stamp, self.time))
|
|
else:
|
|
html_file = path.join(save_location, "%s%s.html" % (path.basename(fname), time.strftime(time_stamp, self.time)))
|
|
else:
|
|
html_file = ".html"
|
|
|
|
if save_location is not None:
|
|
open_html = lambda x: open(x, "w")
|
|
else:
|
|
open_html = lambda x: tempfile.NamedTemporaryFile(delete=False, suffix=x)
|
|
|
|
with open_html(html_file) as the_html:
|
|
self.write_header(the_html)
|
|
self.write_body(the_html)
|
|
if inputs["clipboard_copy"]:
|
|
the_html.seek(0)
|
|
sublime.set_clipboard(the_html.read())
|
|
sublime.status_message("Export to HTML: copied to clipboard")
|
|
|
|
if inputs["view_open"]:
|
|
self.view.window().open_file(the_html.name)
|
|
else:
|
|
# Open in web browser; check return code, if failed try webbrowser
|
|
status = desktop.open(the_html.name, status=True)
|
|
if not status:
|
|
webbrowser.open(the_html.name, new=2)
|