diff --git a/lib/docs/filters/cpp20/clean_html.rb b/lib/docs/filters/cpp20/clean_html.rb
new file mode 100644
index 00000000..4328e002
--- /dev/null
+++ b/lib/docs/filters/cpp20/clean_html.rb
@@ -0,0 +1,9 @@
+module Docs
+ class Cpp20
+ class CleanHtmlFilter < Filter
+ def call
+ doc
+ end
+ end
+ end
+end
diff --git a/lib/docs/filters/cpp20/entries.rb b/lib/docs/filters/cpp20/entries.rb
new file mode 100644
index 00000000..8a5aad22
--- /dev/null
+++ b/lib/docs/filters/cpp20/entries.rb
@@ -0,0 +1,82 @@
+module Docs
+ class Cpp20
+ class EntriesFilter < Docs::EntriesFilter
+ @@duplicate_names = []
+
+ REPLACE_NAMES = {
+ 'Error directive' => '#error directive',
+ 'Filename and line information' => '#line directive',
+ 'Implementation defined behavior control' => '#pragma directive',
+ 'Replacing text macros' => '#define directive',
+ 'Source file inclusion' => '#include directive' }
+
+ def get_name
+ name = at_css('#firstHeading').content.strip
+ name = format_name(name)
+ name = name.split(',').first
+ name
+ end
+
+ def get_type
+ if at_css('#firstHeading').content.include?('C++ keyword')
+ 'Keywords'
+ elsif subpath.start_with?('experimental')
+ 'Experimental libraries'
+ elsif subpath.start_with?('language/')
+ 'Language'
+ elsif subpath.start_with?('freestanding')
+ 'Utilities'
+ elsif type = at_css('.t-navbar > div:nth-child(4) > :first-child').try(:content)
+ type.strip!
+ type.remove! ' library'
+ type.remove! ' utilities'
+ type.remove! 'C++ '
+ type.capitalize!
+ type
+ end
+ end
+
+ def additional_entries
+ return [] if root_page? || self.name.start_with?('operators')
+ names = at_css('#firstHeading').content.remove(%r{\(.+?\)}).split(', ')[1..-1]
+ names.each(&:strip!).reject! do |name|
+ name.size <= 2 || name == '...' || name =~ /\A[<>]/ || name.start_with?('operator')
+ end
+ names.map { |name| [format_name(name)] }
+ end
+
+ def format_name(name)
+ name.remove! 'C++ concepts: '
+ name.remove! 'C++ keywords: '
+ name.remove! 'C++ ' unless name == 'C++ language'
+ name.remove! %r{\s\(.+\)}
+
+ name.sub! %r{\AStandard library header <(.+)>\z}, '\1'
+ name.sub! %r{(<[^>]+>)}, ''
+
+ if name.include?('operator') && name.include?(',')
+ name.sub!(%r{operator.+([\( ])}, 'operators (') || name.sub!(%r{operator.+}, 'operators')
+ name.sub! ' ', ' '
+ name << ')' unless name.last == ')' || name.exclude?('(')
+ name.sub! '()', ''
+ name.sub! %r{\(.+\)}, '' if !name.start_with?('operator') && name.length > 50
+ end
+
+ REPLACE_NAMES[name] || name
+ end
+
+ def entries
+ entries = []
+
+ # avoid duplicate pages
+ if !(@@duplicate_names.include?(name))
+ @@duplicate_names.push(name)
+ entries << default_entry if root_page? || include_default_entry?
+ entries.concat(additional_entries)
+ build_entries(entries)
+ end
+ end
+
+ end
+ end
+end
diff --git a/lib/docs/scrapers/cpp20.rb b/lib/docs/scrapers/cpp20.rb
new file mode 100644
index 00000000..14034a43
--- /dev/null
+++ b/lib/docs/scrapers/cpp20.rb
@@ -0,0 +1,40 @@
+module Docs
+ class Cpp20 < UrlScraper
+ self.name = 'C++20'
+ self.slug = 'cpp20'
+ self.type = 'c'
+ self.base_url = 'https://en.cppreference.com/w/cpp/'
+ self.root_path = 'header'
+
+ html_filters.insert_before 'clean_html', 'c/fix_code'
+ html_filters.push 'cpp20/entries', 'c/clean_html', 'title'
+
+ options[:decode_and_clean_paths] = true
+ options[:container] = '#content'
+ options[:title] = false
+ options[:root_title] = 'C++ Programming Language'
+
+ options[:skip] = %w(
+ language/extending_std.html
+ language/history.html
+ regex/ecmascript.html
+ regex/regex_token_iterator/operator_cmp.html
+ )
+
+ options[:skip_patterns] = [/experimental/]
+
+ options[:attribution] = <<-HTML
+ © cppreference.com
+ Licensed under the Creative Commons Attribution-ShareAlike Unported License v3.0.
+ HTML
+
+ # Same as get_latest_version in lib/docs/scrapers/c.rb
+ def get_latest_version(opts)
+ doc = fetch_doc('https://en.cppreference.com/w/Cppreference:Archives', opts)
+ link = doc.at_css('a[title^="File:"]')
+ date = link.content.scan(/(\d+)\./)[0][0]
+ DateTime.strptime(date, '%Y%m%d').to_time.to_i
+ end
+
+ end
+end