mirror of
https://github.com/davidhalter/jedi.git
synced 2026-01-31 18:55:21 +08:00
Add filters for the sub module dicts and module attributes dicts.
This commit is contained in:
@@ -11,6 +11,7 @@ from jedi.api import keywords
|
||||
from jedi.evaluate import compiled
|
||||
from jedi.evaluate.helpers import call_of_leaf
|
||||
from jedi.evaluate.finder import global_names_dict_generator, filter_definition_names
|
||||
from jedi.evaluate.filters import get_global_filters
|
||||
|
||||
|
||||
def get_call_signature_param_names(call_signatures):
|
||||
@@ -182,19 +183,14 @@ class Completion:
|
||||
scope = scope.get_parent_scope()
|
||||
scope = self._evaluator.wrap(scope)
|
||||
debug.dbg('global completion scope: %s', scope)
|
||||
names_dicts = global_names_dict_generator(
|
||||
filters = get_global_filters(
|
||||
self._evaluator,
|
||||
scope,
|
||||
self._position
|
||||
)
|
||||
completion_names = []
|
||||
for names_dict, pos in names_dicts:
|
||||
names = list(chain.from_iterable(names_dict.values()))
|
||||
if not names:
|
||||
continue
|
||||
completion_names += filter_definition_names(
|
||||
names, self._module.get_statement_for_position(self._position), pos
|
||||
)
|
||||
for filter in filters:
|
||||
completion_names += filter.values()
|
||||
return completion_names
|
||||
|
||||
def _trailer_completions(self, atom_expr):
|
||||
|
||||
@@ -353,22 +353,18 @@ class CompiledObjectFilter(AbstractFilter):
|
||||
return [self.name_class(self._evaluator, self._compiled_obj, name)]
|
||||
|
||||
def values(self):
|
||||
raise NotImplementedError
|
||||
obj = self._compiled_obj.obj
|
||||
|
||||
values = []
|
||||
names = []
|
||||
for name in dir(obj):
|
||||
try:
|
||||
values.append(self[name])
|
||||
except KeyError:
|
||||
# The dir function can be wrong.
|
||||
pass
|
||||
names += self.get(name)
|
||||
|
||||
is_instance = self._is_instance or fake.is_class_instance(obj)
|
||||
# ``dir`` doesn't include the type names.
|
||||
if not inspect.ismodule(obj) and obj != type and not is_instance:
|
||||
values += create(self._evaluator, type).names_dict.values()
|
||||
return values
|
||||
for filter in create(self._evaluator, type).get_filters():
|
||||
names += filter.values()
|
||||
return names
|
||||
|
||||
|
||||
def dotted_from_fs_path(fs_path, sys_path):
|
||||
|
||||
@@ -44,7 +44,8 @@ class AbstractUsedNamesFilter(AbstractFilter):
|
||||
return list(self._filter(names))
|
||||
|
||||
def values(self):
|
||||
return self._filter(self._used_names.values())
|
||||
names = [name for name_list in self._used_names.values() for name in name_list]
|
||||
return self._filter(names)
|
||||
|
||||
|
||||
class ParserTreeFilter(AbstractUsedNamesFilter):
|
||||
@@ -105,6 +106,23 @@ class GlobalNameFilter(AbstractUsedNamesFilter):
|
||||
yield name
|
||||
|
||||
|
||||
class DictFilter(AbstractFilter):
|
||||
def __init__(self, dct, origin_scope=None):
|
||||
super(DictFilter, self).__init__(origin_scope)
|
||||
self._dct = dct
|
||||
|
||||
def get(self, name):
|
||||
try:
|
||||
leaf_name = self._dct[str(name)]
|
||||
except KeyError:
|
||||
return []
|
||||
|
||||
return list(self._filter([leaf_name]))
|
||||
|
||||
def values(self):
|
||||
return self._filter(self._dct.values())
|
||||
|
||||
|
||||
def get_global_filters(evaluator, context, until_position):
|
||||
"""
|
||||
Returns all filters in order of priority for name resolution.
|
||||
|
||||
@@ -55,7 +55,8 @@ from jedi.evaluate import helpers
|
||||
from jedi.evaluate import param
|
||||
from jedi.evaluate import flow_analysis
|
||||
from jedi.evaluate import imports
|
||||
from jedi.evaluate.filters import ParserTreeFilter, FunctionExecutionFilter, GlobalNameFilter
|
||||
from jedi.evaluate.filters import ParserTreeFilter, FunctionExecutionFilter, \
|
||||
GlobalNameFilter, DictFilter
|
||||
|
||||
|
||||
class Executed(tree.Base):
|
||||
@@ -912,15 +913,12 @@ class ModuleWrapper(use_metaclass(CachedMetaClass, tree.Module, Wrapper)):
|
||||
def get_filters(self, search_global, until_position=None):
|
||||
yield ParserTreeFilter(self._evaluator, self._module, until_position)
|
||||
yield GlobalNameFilter(self._module)
|
||||
yield DictFilter(self._sub_modules_dict())
|
||||
yield DictFilter(self._module_attributes_dict())
|
||||
# TODO
|
||||
'''
|
||||
yield self._module_attributes_dict()
|
||||
|
||||
for star_module in self.star_imports():
|
||||
yield star_module.names_dict
|
||||
|
||||
yield dict((str(n), [GlobalName(n)]) for n in self.base.global_names)
|
||||
yield self._sub_modules_dict()
|
||||
'''
|
||||
|
||||
# I'm not sure if the star import cache is really that effective anymore
|
||||
@@ -948,7 +946,7 @@ class ModuleWrapper(use_metaclass(CachedMetaClass, tree.Module, Wrapper)):
|
||||
|
||||
names = ['__file__', '__package__', '__doc__', '__name__']
|
||||
# All the additional module attributes are strings.
|
||||
return dict((n, [helpers.LazyName(n, parent_callback, is_definition=True)])
|
||||
return dict((n, helpers.LazyName(n, parent_callback, is_definition=True))
|
||||
for n in names)
|
||||
|
||||
@property
|
||||
@@ -1049,7 +1047,7 @@ class ModuleWrapper(use_metaclass(CachedMetaClass, tree.Module, Wrapper)):
|
||||
# It's obviously a relative import to the current module.
|
||||
imp = helpers.FakeImport(fake_n, self, level=1)
|
||||
fake_n.parent = imp
|
||||
names[name] = [fake_n]
|
||||
names[name] = fake_n
|
||||
|
||||
# TODO add something like this in the future, its cleaner than the
|
||||
# import hacks.
|
||||
|
||||
Reference in New Issue
Block a user