Created
January 3, 2018 10:39
-
-
Save sunil-bagde/78ffe1d0c3e7035d70aa13d59a6668f5 to your computer and use it in GitHub Desktop.
Tooltip
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# | |
# sublimelinter.py | |
# Part of SublimeLinter3, a code checking framework for Sublime Text 3 | |
# | |
# Written by Ryan Hileman and Aparajita Fishman | |
# | |
# Project: https://github.com/SublimeLinter/SublimeLinter3 | |
# License: MIT | |
# | |
"""This module provides the SublimeLinter plugin class and supporting methods.""" | |
import os | |
import html | |
import re | |
import sublime | |
import sublime_plugin | |
from .lint.linter import Linter | |
from .lint.highlight import HighlightSet | |
from .lint.queue import queue | |
from .lint import persist, util | |
from string import Template | |
def plugin_loaded(): | |
"""Entry point for SL plugins.""" | |
persist.plugin_is_loaded = True | |
persist.settings.load() | |
persist.debug('debug mode: on') | |
util.create_tempdir() | |
for linter in persist.linter_classes.values(): | |
linter.initialize() | |
plugin = SublimeLinter.shared_plugin() | |
queue.start(plugin.lint) | |
util.generate_menus() | |
util.generate_color_scheme(from_reload=False) | |
persist.settings.on_update_call(SublimeLinter.on_settings_updated) | |
# This ensures we lint the active view on a fresh install | |
window = sublime.active_window() | |
if window: | |
plugin.on_activated_async(window.active_view()) | |
class SublimeLinter(sublime_plugin.EventListener): | |
"""The main ST3 plugin class.""" | |
# We use this to match linter settings filenames. | |
LINTER_SETTINGS_RE = re.compile('^SublimeLinter(-.+?)?\.sublime-settings') | |
shared_instance = None | |
@classmethod | |
def shared_plugin(cls): | |
"""Return the plugin instance.""" | |
return cls.shared_instance | |
def __init__(self, *args, **kwargs): | |
"""Initialize a new instance.""" | |
super().__init__(*args, **kwargs) | |
# Keeps track of which views we have assigned linters to | |
self.loaded_views = set() | |
# Keeps track of which views have actually been linted | |
self.linted_views = set() | |
# A mapping between view ids and syntax names | |
self.view_syntax = {} | |
self.__class__.shared_instance = self | |
@classmethod | |
def lint_all_views(cls): | |
"""Simulate a modification of all views, which will trigger a relint.""" | |
def apply(view): | |
if view.id() in persist.view_linters: | |
cls.shared_instance.hit(view) | |
util.apply_to_all_views(apply) | |
def lint(self, view_id, hit_time=None, callback=None): | |
""" | |
Lint the view with the given id. | |
This method is called asynchronously by queue.Daemon when a lint | |
request is pulled off the queue. | |
If provided, hit_time is the time at which the lint request was added | |
to the queue. It is used to determine if the view has been modified | |
since the lint request was queued. If so, the lint is aborted, since | |
another lint request is already in the queue. | |
callback is the method to call when the lint is finished. If not | |
provided, it defaults to highlight(). | |
""" | |
# If the view has been modified since the lint was triggered, | |
# don't lint again. | |
if hit_time is not None and persist.last_hit_times.get(view_id, 0) > hit_time: | |
return | |
view = Linter.get_view(view_id) | |
if view is None: | |
return | |
filename = view.file_name() | |
code = Linter.text(view) | |
callback = callback or self.highlight | |
Linter.lint_view(view, filename, code, hit_time, callback) | |
def highlight(self, view, linters, hit_time): | |
""" | |
Highlight any errors found during a lint of the given view. | |
This method is called by Linter.lint_view after linting is finished. | |
linters is a list of the linters that ran. hit_time has the same meaning | |
as in lint(), and if the view was modified since the lint request was | |
made, this method aborts drawing marks. | |
If the view has not been modified since hit_time, all of the marks and | |
errors from the list of linters are aggregated and drawn, and the status | |
is updated. | |
""" | |
vid = view.id() | |
# If the view has been modified since the lint was triggered, | |
# don't draw marks. | |
if hit_time is not None and persist.last_hit_times.get(vid, 0) > hit_time: | |
return | |
errors = {} | |
highlights = persist.highlights[vid] = HighlightSet() | |
for linter in linters: | |
if linter.highlight: | |
highlights.add(linter.highlight) | |
if linter.errors: | |
for line, errs in linter.errors.items(): | |
errors.setdefault(line, []).extend(errs) | |
# Keep track of one view in each window that shares view's buffer | |
window_views = {} | |
buffer_id = view.buffer_id() | |
for window in sublime.windows(): | |
wid = window.id() | |
for other_view in window.views(): | |
if other_view.buffer_id() == buffer_id: | |
vid = other_view.id() | |
persist.highlights[vid] = highlights | |
highlights.clear(other_view) | |
highlights.draw(other_view) | |
persist.errors[vid] = errors | |
if window_views.get(wid) is None: | |
window_views[wid] = other_view | |
for view in window_views.values(): | |
self.display_errors(view) | |
def hit(self, view): | |
"""Record an activity that could trigger a lint and enqueue a desire to lint.""" | |
vid = view.id() | |
self.check_syntax(view) | |
self.linted_views.add(vid) | |
if view.size() == 0: | |
for linter in Linter.get_linters(vid): | |
linter.clear() | |
return | |
persist.last_hit_times[vid] = queue.hit(view) | |
def check_syntax(self, view): | |
""" | |
Check and return if view's syntax has changed. | |
If the syntax has changed, a new linter is assigned. | |
""" | |
vid = view.id() | |
syntax = persist.get_syntax(view) | |
# Syntax either has never been set or just changed | |
if vid not in self.view_syntax or self.view_syntax[vid] != syntax: | |
self.view_syntax[vid] = syntax | |
Linter.assign(view, reset=True) | |
self.clear(view) | |
return True | |
else: | |
return False | |
def clear(self, view): | |
"""Clear all marks, errors and status from the given view.""" | |
Linter.clear_view(view) | |
def is_scratch(self, view): | |
""" | |
Return whether a view is effectively scratch. | |
There is a bug (or feature) in the current ST3 where the Find panel | |
is not marked scratch but has no window. | |
There is also a bug where settings files opened from within .sublime-package | |
files are not marked scratch during the initial on_modified event, so we have | |
to check that a view with a filename actually exists on disk if the file | |
being opened is in the Sublime Text packages directory. | |
""" | |
if view.is_scratch() or view.is_read_only() or view.window() is None or view.settings().get("repl") is not None: | |
return True | |
elif ( | |
view.file_name() and | |
view.file_name().startswith(sublime.packages_path() + os.path.sep) and | |
not os.path.exists(view.file_name()) | |
): | |
return True | |
else: | |
return False | |
def view_has_file_only_linter(self, vid): | |
"""Return True if any linters for the given view are file-only.""" | |
for lint in persist.view_linters.get(vid, []): | |
if lint.tempfile_suffix == '-': | |
return True | |
return False | |
# sublime_plugin.EventListener event handlers | |
def on_modified_async(self, view): | |
"""Ran when view is motified.""" | |
if self.is_scratch(view): | |
return | |
if view.id() not in persist.view_linters: | |
syntax_changed = self.check_syntax(view) | |
if not syntax_changed: | |
return | |
else: | |
syntax_changed = False | |
if syntax_changed or persist.settings.get('lint_mode', 'background') == 'background': | |
self.hit(view) | |
else: | |
self.clear(view) | |
def on_activated_async(self, view): | |
"""Ran when a view gains input focus.""" | |
if self.is_scratch(view): | |
return | |
# Reload the plugin settings. | |
persist.settings.load() | |
self.check_syntax(view) | |
view_id = view.id() | |
if view_id not in self.linted_views: | |
if view_id not in self.loaded_views: | |
self.on_new_async(view) | |
if persist.settings.get('lint_mode', 'background') in ('background', 'load/save'): | |
self.hit(view) | |
self.display_errors(view) | |
def on_open_settings(self, view): | |
""" | |
Ran when any settings file is opened. | |
view is the view that contains the text of the settings file. | |
""" | |
if self.is_settings_file(view, user_only=True): | |
persist.settings.save(view=view) | |
def is_settings_file(self, view, user_only=False): | |
"""Return True if view is a SublimeLinter settings file.""" | |
filename = view.file_name() | |
if not filename: | |
return False | |
if not filename.startswith(sublime.packages_path()): | |
return False | |
dirname, filename = os.path.split(filename) | |
dirname = os.path.basename(dirname) | |
if self.LINTER_SETTINGS_RE.match(filename): | |
if user_only: | |
return dirname == 'User' | |
else: | |
return dirname in (persist.PLUGIN_DIRECTORY, 'User') | |
@classmethod | |
def on_settings_updated(cls, relint=False): | |
"""Ran when the settings are updated.""" | |
if relint: | |
cls.lint_all_views() | |
else: | |
Linter.redraw_all() | |
def on_new_async(self, view): | |
"""Ran when a new buffer is created.""" | |
self.on_open_settings(view) | |
if self.is_scratch(view): | |
return | |
vid = view.id() | |
self.loaded_views.add(vid) | |
self.view_syntax[vid] = persist.get_syntax(view) | |
def get_focused_view_id(self, view): | |
""" | |
Return the focused view which shares view's buffer. | |
When updating the status, we want to make sure we get | |
the selection of the focused view, since multiple views | |
into the same buffer may be open. | |
""" | |
active_view = view.window().active_view() | |
if active_view is None: | |
return | |
for view in view.window().views(): | |
if view == active_view: | |
return view | |
def on_selection_modified_async(self, view): | |
"""Ran when the selection changes (cursor moves or text selected).""" | |
self.display_errors(view, tooltip=True) | |
def display_errors(self, view, tooltip=False): | |
""" | |
Display lint errors in the view status. | |
If tooltip is set to True, also display lint errors in a tooltip | |
at the caret's position. | |
""" | |
if self.is_scratch(view): | |
return | |
view = self.get_focused_view_id(view) | |
if view is None: | |
return | |
vid = view.id() | |
# Get the line number of the first line of the first selection. | |
try: | |
lineno = view.rowcol(view.sel()[0].begin())[0] | |
except IndexError: | |
lineno = -1 | |
if vid in persist.errors: | |
errors = persist.errors[vid] | |
if errors: | |
lines = sorted(list(errors)) | |
counts = [len(errors[line]) for line in lines] | |
count = sum(counts) | |
plural = 's' if count > 1 else '' | |
if lineno in errors: | |
# Sort the errors by column | |
line_errors = sorted(errors[lineno], key=lambda error: error[0]) | |
line_errors = [error[1] for error in line_errors] | |
if plural: | |
# Sum the errors before the first error on this line | |
index = lines.index(lineno) | |
first = sum(counts[0:index]) + 1 | |
if len(line_errors) > 1: | |
last = first + len(line_errors) - 1 | |
status = '{}-{} of {} errors: '.format(first, last, count) | |
else: | |
status = '{} of {} errors: '.format(first, count) | |
else: | |
status = 'Error: ' | |
status += '; '.join(line_errors) | |
# view.show_popup('<div class="content">' + ''.join(line_errors) + '</div>', | |
# location=-1, | |
# max_width=600) | |
view.show_popup('<div class="content">' | |
+ '<ul style="margin:8px; padding-left: 18px">' + ''.join(map(lambda e: '<li>' + e + '</li>', line_errors)) + '</ul></div>', | |
sublime.COOPERATE_WITH_AUTO_COMPLETE, | |
location=-1, | |
max_width=600) | |
if persist.settings.get('tooltips') and tooltip: | |
self.open_tooltip(lineno, line_errors) | |
else: | |
status = '%i error%s' % (count, plural) | |
view.set_status('sublimelinter', status) | |
else: | |
view.erase_status('sublimelinter') | |
def get_active_view(self): | |
"""Return the active view in the currently active window.""" | |
return sublime.active_window().active_view() | |
def get_template(self): | |
""" | |
If no tooltip theme has been configured, returns False. | |
Otherwise, returns a Template object with the template text. | |
""" | |
tooltip_theme = persist.settings.get('tooltip_theme') | |
if tooltip_theme == 'none': | |
return False | |
theme_path = os.path.dirname(tooltip_theme) | |
template_path = os.path.join(theme_path, 'tooltip.html') | |
tooltip_text = sublime.load_resource(template_path) | |
return Template(tooltip_text) | |
def open_tooltip(self, line, errors): | |
""" | |
Show a tooltip containing all linting errors on a given line. | |
Does nothing if no tooltip template can be created, or if another popup | |
is already displayed. | |
""" | |
template = self.get_template() | |
if not template: | |
return | |
active_view = self.get_active_view() | |
# Leave any existing popup open without replacing it | |
if active_view.is_popup_visible(): | |
return | |
tooltip_content = template.substitute(line=line + 1, | |
message='<br />'.join(html.escape(e, quote=False) for e in errors), | |
font_size=persist.settings.get('tooltip_fontsize')) | |
active_view.show_popup(tooltip_content, | |
flags=sublime.HIDE_ON_MOUSE_MOVE_AWAY, | |
location=-1, | |
max_width=600) | |
def on_pre_save_async(self, view): | |
""" | |
Ran before view is saved. | |
If a settings file is the active view and is saved, | |
copy the current settings first so we can compare post-save. | |
""" | |
window = view.window() | |
if window and window.active_view() == view and self.is_settings_file(view): | |
persist.settings.copy() | |
def on_post_save_async(self, view): | |
"""Ran after view is saved.""" | |
if self.is_scratch(view): | |
return | |
# First check to see if the project settings changed | |
if view.window().project_file_name() == view.file_name(): | |
self.lint_all_views() | |
else: | |
# Now see if a .sublimelinterrc has changed | |
filename = os.path.basename(view.file_name()) | |
if filename == '.sublimelinterrc': | |
# If a .sublimelinterrc has changed, to be safe | |
# clear the rc cache and relint. | |
util.get_rc_settings.cache_clear() | |
self.lint_all_views() | |
# If a file other than one of our settings files changed, | |
# check if the syntax changed or if we need to show errors. | |
elif filename != 'SublimeLinter.sublime-settings': | |
self.file_was_saved(view) | |
def file_was_saved(self, view): | |
"""Check if the syntax changed or if we need to show errors.""" | |
syntax_changed = self.check_syntax(view) | |
vid = view.id() | |
mode = persist.settings.get('lint_mode', 'background') | |
show_errors = persist.settings.get('show_errors_on_save', False) | |
if syntax_changed: | |
self.clear(view) | |
if vid in persist.view_linters: | |
if mode != 'manual': | |
self.hit(view) | |
else: | |
show_errors = False | |
else: | |
show_errors = False | |
else: | |
if show_errors: | |
# if showing errors on save, linting must be synchronized. | |
self.lint(vid) | |
elif ( | |
mode in ('load/save', 'save only') or | |
mode == 'background' and self.view_has_file_only_linter(vid) | |
): | |
self.hit(view) | |
elif mode == 'manual': | |
show_errors = False | |
if show_errors and vid in persist.errors and persist.errors[vid]: | |
view.run_command('sublimelinter_show_all_errors') | |
def on_close_async(self, view): | |
"""Ran after view is closed.""" | |
if self.is_scratch(view): | |
return | |
vid = view.id() | |
if vid in self.loaded_views: | |
self.loaded_views.remove(vid) | |
if vid in self.linted_views: | |
self.linted_views.remove(vid) | |
if vid in self.view_syntax: | |
del self.view_syntax[vid] | |
persist.view_did_close(vid) | |
class SublimelinterEditCommand(sublime_plugin.TextCommand): | |
"""A plugin command used to generate an edit object for a view.""" | |
def run(self, edit): | |
"""Run the command.""" | |
persist.edit(self.view.id(), edit) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment