module SCSSLint # Checks the format of declared names of functions, mixins, variables, and # placeholders. class Linter::NameFormat < Linter include LinterRegistry def visit_extend(node) check_placeholder(node) end def visit_function(node) check_name(node, 'function') yield # Continue into content block of this function definition end def visit_mixin(node) check_name(node, 'mixin') unless FUNCTION_WHITELIST.include?(node.name) yield # Continue into content block of this mixin's block end def visit_mixindef(node) check_name(node, 'mixin') yield # Continue into content block of this mixin definition end def visit_script_funcall(node) check_name(node, 'function') unless FUNCTION_WHITELIST.include?(node.name) end def visit_script_variable(node) check_name(node, 'variable') end def visit_variable(node) check_name(node, 'variable') yield # Continue into expression tree for this variable definition end private FUNCTION_WHITELIST = %w[ rotateX rotateY rotateZ scaleX scaleY scaleZ skewX skewY translateX translateY translateZ ].to_set def check_name(node, node_type, node_text = node.name) node_text = trim_underscore_prefix(node_text) return unless violation = violated_convention(node_text, node_type) add_lint(node, "Name of #{node_type} `#{node_text}` #{violation[:explanation]}") end # Removes underscore prefix from name if leading underscores are allowed. def trim_underscore_prefix(name) if config['allow_leading_underscore'] # Remove if there is a single leading underscore name = name.gsub(/^_(?!_)/, '') end name end def check_placeholder(node) extract_string_selectors(node.selector).any? do |selector_str| check_name(node, 'placeholder', selector_str.gsub('%', '')) end end CONVENTIONS = { 'camel_case' => { explanation: 'should be written in camelCase format', validator: ->(name) { name =~ /^[a-z][a-zA-Z0-9]*$/ }, }, 'snake_case' => { explanation: 'should be written in snake_case', validator: ->(name) { name !~ /[^_a-z0-9]/ }, }, 'hyphenated_lowercase' => { explanation: 'should be written in all lowercase letters with hyphens ' \ 'instead of underscores', validator: ->(name) { name !~ /[_A-Z]/ }, }, } def violated_convention(name_string, type) convention_name = convention_name(type) existing_convention = CONVENTIONS[convention_name] convention = (existing_convention || { validator: ->(name) { name =~ /#{convention_name}/ } }).merge( explanation: convention_explanation(type), # Allow explanation to be customized ) convention unless convention[:validator].call(name_string) end def convention_name(type) config["#{type}_convention"] || config['convention'] || 'hyphenated_lowercase' end def convention_explanation(type) existing_convention = CONVENTIONS[convention_name(type)] config["#{type}_convention_explanation"] || config['convention_explanation'] || (existing_convention && existing_convention[:explanation]) || "should match regex /#{convention_name(type)}/" end end end