class Capybara::Selector::FilterSet

def add(name, &block)

def add(name, &block)
  all[name.to_sym] = FilterSet.new(name.to_sym, &block)
end

def add_filter(name, filter_class, *types, matcher: nil, **options, &block)

def add_filter(name, filter_class, *types, matcher: nil, **options, &block)
  types.each { |type| options[type] = true }
  raise 'ArgumentError', ':default option is not supported for filters with a :matcher option' if matcher && options[:default]
  if filter_class <= Filters::ExpressionFilter
    @expression_filters[name] = filter_class.new(name, matcher, block, options)
  else
    @node_filters[name] = filter_class.new(name, matcher, block, options)
  end
end

def all

def all
  @filter_sets ||= {} # rubocop:disable Naming/MemoizedInstanceVariableName
end

def describe(what = nil, &block)

def describe(what = nil, &block)
  case what
  when nil
    undeclared_descriptions.push block
  when :node_filters
    node_filter_descriptions.push block
  when :expression_filters
    expression_filter_descriptions.push block
  else
    raise ArgumentError, 'Unknown description type'
  end
end

def description(node_filters: true, expression_filters: true, **options)

def description(node_filters: true, expression_filters: true, **options)
  opts = options_with_defaults(options)
  description = +''
  description += undeclared_descriptions.map { |desc| desc.call(opts).to_s }.join
  description += expression_filter_descriptions.map { |desc| desc.call(opts).to_s }.join if expression_filters
  description += node_filter_descriptions.map { |desc| desc.call(opts).to_s }.join if node_filters
  description
end

def descriptions

def descriptions
  warn 'DEPRECATED: FilterSet#descriptions is deprecated without replacement'
  [undeclared_descriptions, node_filter_descriptions, expression_filter_descriptions].flatten
end

def expression_filter(name, *types_and_options, &block)

def expression_filter(name, *types_and_options, &block)
  add_filter(name, Filters::ExpressionFilter, *types_and_options, &block)
end

def expression_filter_descriptions

def expression_filter_descriptions
  @descriptions[:expression_filters]
end

def import(name, filters = nil)

def import(name, filters = nil)
  f_set = self.class.all[name]
  filter_selector = filters.nil? ? ->(*) { true } : ->(filter_name, _) { filters.include? filter_name }
  expression_filters.merge!(f_set.expression_filters.select(&filter_selector))
  node_filters.merge!(f_set.node_filters.select(&filter_selector))
  f_set.undeclared_descriptions.each { |desc| describe(&desc) }
  f_set.expression_filter_descriptions.each { |desc| describe(:expression_filters, &desc) }
  f_set.node_filter_descriptions.each { |desc| describe(:node_filters, &desc) }
end

def initialize(name, &block)

def initialize(name, &block)
  @name = name
  @node_filters = {}
  @expression_filters = {}
  @descriptions = Hash.new { |hsh, key| hsh[key] = [] }
  instance_eval(&block)
end

def node_filter(name, *types_and_options, &block)

def node_filter(name, *types_and_options, &block)
  add_filter(name, Filters::NodeFilter, *types_and_options, &block)
end

def node_filter_descriptions

def node_filter_descriptions
  @descriptions[:node_filters]
end

def options_with_defaults(options)

def options_with_defaults(options)
  options = options.dup
  [expression_filters, node_filters].each do |filters|
    filters.select { |_n, filter| filter.default? }.each do |name, filter|
      options[name] = filter.default unless options.key?(name)
    end
  end
  options
end

def remove(name)

def remove(name)
  all.delete(name.to_sym)
end

def undeclared_descriptions

def undeclared_descriptions
  @descriptions[:undeclared]
end