2019-04-29 20:27:42 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2022-07-27 22:27:38 -04:00
|
|
|
RSpec.describe SvgSprite do
|
2021-12-22 12:09:43 -05:00
|
|
|
fab!(:theme) { Fabricate(:theme) }
|
2018-11-26 16:49:57 -05:00
|
|
|
|
2019-02-06 10:51:23 -05:00
|
|
|
before { SvgSprite.expire_cache }
|
2018-11-26 16:49:57 -05:00
|
|
|
|
|
|
|
it "can generate a bundle" do
|
|
|
|
bundle = SvgSprite.bundle
|
|
|
|
expect(bundle).to match(/heart/)
|
|
|
|
expect(bundle).to match(/angle-double-down/)
|
|
|
|
end
|
|
|
|
|
2019-02-06 10:51:23 -05:00
|
|
|
it "can generate paths" do
|
|
|
|
version = SvgSprite.version # Icons won't change for this test
|
|
|
|
expect(SvgSprite.path).to eq("/svg-sprite/#{Discourse.current_hostname}/svg--#{version}.js")
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.path(1)).to eq("/svg-sprite/#{Discourse.current_hostname}/svg-1-#{version}.js")
|
2019-02-06 10:51:23 -05:00
|
|
|
|
|
|
|
# Safe mode
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.path(nil)).to eq(
|
|
|
|
"/svg-sprite/#{Discourse.current_hostname}/svg--#{version}.js",
|
|
|
|
)
|
2019-02-06 10:51:23 -05:00
|
|
|
end
|
|
|
|
|
2018-11-26 16:49:57 -05:00
|
|
|
it "can search for a specific FA icon" do
|
|
|
|
expect(SvgSprite.search("fa-heart")).to match(/heart/)
|
|
|
|
expect(SvgSprite.search("poo-storm")).to match(/poo-storm/)
|
|
|
|
expect(SvgSprite.search("this-is-not-an-icon")).to eq(false)
|
|
|
|
end
|
|
|
|
|
2019-08-08 05:44:04 -04:00
|
|
|
it "can get a raw SVG for an icon" do
|
|
|
|
expect(SvgSprite.raw_svg("fa-heart")).to match(/svg.*svg/) # SVG inside SVG
|
|
|
|
expect(SvgSprite.raw_svg("this-is-not-an-icon")).to eq("")
|
|
|
|
end
|
|
|
|
|
2018-11-26 16:49:57 -05:00
|
|
|
it "can get a consistent version string" do
|
|
|
|
version1 = SvgSprite.version
|
|
|
|
version2 = SvgSprite.version
|
|
|
|
|
|
|
|
expect(version1).to eq(version2)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "version string changes" do
|
|
|
|
version1 = SvgSprite.version
|
|
|
|
Fabricate(:badge, name: "Custom Icon Badge", icon: "fa-gamepad")
|
|
|
|
version2 = SvgSprite.version
|
|
|
|
|
|
|
|
expect(version1).not_to eq(version2)
|
|
|
|
end
|
|
|
|
|
2019-11-14 08:20:16 -05:00
|
|
|
it "version should be based on bundled output, not requested icons" do
|
|
|
|
fname = "custom-theme-icon-sprite.svg"
|
|
|
|
upload = UploadCreator.new(file_from_fixtures(fname), fname, for_theme: true).create_for(-1)
|
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
version1 = SvgSprite.version(theme.id)
|
|
|
|
bundle1 = SvgSprite.bundle(theme.id)
|
2019-11-14 08:20:16 -05:00
|
|
|
|
|
|
|
SiteSetting.svg_icon_subset = "my-custom-theme-icon"
|
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
version2 = SvgSprite.version(theme.id)
|
|
|
|
bundle2 = SvgSprite.bundle(theme.id)
|
2019-11-14 08:20:16 -05:00
|
|
|
|
|
|
|
# The contents of the bundle should not change, because the icon does not actually exist
|
|
|
|
expect(bundle1).to eq(bundle2)
|
|
|
|
# Therefore the version hash should not change
|
|
|
|
expect(version1).to eq(version2)
|
|
|
|
|
|
|
|
# Now add the icon to the theme
|
|
|
|
theme.set_field(
|
|
|
|
target: :common,
|
|
|
|
name: SvgSprite.theme_sprite_variable_name,
|
|
|
|
upload_id: upload.id,
|
|
|
|
type: :theme_upload_var,
|
|
|
|
)
|
|
|
|
theme.save!
|
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
version3 = SvgSprite.version(theme.id)
|
|
|
|
bundle3 = SvgSprite.bundle(theme.id)
|
2019-11-14 08:20:16 -05:00
|
|
|
|
|
|
|
# The version/bundle should be updated
|
|
|
|
expect(bundle3).not_to match(bundle2)
|
|
|
|
expect(version3).not_to match(version2)
|
|
|
|
expect(bundle3).to match(/my-custom-theme-icon/)
|
|
|
|
end
|
|
|
|
|
2018-11-26 16:49:57 -05:00
|
|
|
it "strips whitespace when processing icons" do
|
|
|
|
Fabricate(:badge, name: "Custom Icon Badge", icon: " fab fa-facebook-messenger ")
|
|
|
|
expect(SvgSprite.all_icons).to include("fab-facebook-messenger")
|
|
|
|
expect(SvgSprite.all_icons).not_to include(" fab-facebook-messenger ")
|
|
|
|
end
|
|
|
|
|
|
|
|
it "includes Font Awesome 5 icons from badges" do
|
|
|
|
Fabricate(:badge, name: "Custom Icon Badge", icon: "far fa-building")
|
|
|
|
expect(SvgSprite.all_icons).to include("far-building")
|
|
|
|
end
|
|
|
|
|
|
|
|
it "includes icons defined in theme settings" do
|
2019-02-06 10:51:23 -05:00
|
|
|
# Works for default settings:
|
|
|
|
theme.set_field(target: :settings, name: :yaml, value: "custom_icon: dragon")
|
2018-11-26 16:49:57 -05:00
|
|
|
theme.save!
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("dragon")
|
2018-11-26 16:49:57 -05:00
|
|
|
|
2019-02-06 10:51:23 -05:00
|
|
|
# Automatically purges cache when default changes:
|
|
|
|
theme.set_field(target: :settings, name: :yaml, value: "custom_icon: gamepad")
|
|
|
|
theme.save!
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("gamepad")
|
2018-11-26 16:49:57 -05:00
|
|
|
|
2019-02-06 10:51:23 -05:00
|
|
|
# Works when applying override
|
2018-11-26 16:49:57 -05:00
|
|
|
theme.update_setting(:custom_icon, "gas-pump")
|
2019-04-12 06:36:08 -04:00
|
|
|
theme.save!
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("gas-pump")
|
2018-11-26 16:49:57 -05:00
|
|
|
|
2019-02-06 10:51:23 -05:00
|
|
|
# Works when changing override
|
2018-11-26 16:49:57 -05:00
|
|
|
theme.update_setting(:custom_icon, "gamepad")
|
2019-04-12 06:36:08 -04:00
|
|
|
theme.save!
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("gamepad")
|
|
|
|
expect(SvgSprite.all_icons(theme.id)).not_to include("gas-pump")
|
2018-11-26 16:49:57 -05:00
|
|
|
|
|
|
|
# FA5 syntax
|
|
|
|
theme.update_setting(:custom_icon, "fab fa-bandcamp")
|
2019-04-12 06:36:08 -04:00
|
|
|
theme.save!
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("fab-bandcamp")
|
2018-11-26 16:49:57 -05:00
|
|
|
|
|
|
|
# Internal Discourse syntax + multiple icons
|
|
|
|
theme.update_setting(:custom_icon, "fab-android|dragon")
|
2019-04-12 06:36:08 -04:00
|
|
|
theme.save!
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("fab-android")
|
|
|
|
expect(SvgSprite.all_icons(theme.id)).to include("dragon")
|
2019-02-06 10:51:23 -05:00
|
|
|
|
|
|
|
# Check themes don't leak into non-theme sprite sheet
|
|
|
|
expect(SvgSprite.all_icons).not_to include("dragon")
|
|
|
|
|
|
|
|
# Check components are included
|
|
|
|
theme.update(component: true)
|
2019-04-12 06:36:08 -04:00
|
|
|
theme.save!
|
2019-02-06 10:51:23 -05:00
|
|
|
parent_theme = Fabricate(:theme)
|
2019-11-28 00:19:01 -05:00
|
|
|
parent_theme.add_relative_theme!(:child, theme)
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(parent_theme.id)).to include("dragon")
|
2018-11-26 16:49:57 -05:00
|
|
|
end
|
|
|
|
|
2020-03-13 12:45:55 -04:00
|
|
|
it "includes icons defined in theme modifiers" do
|
2021-06-17 22:16:26 -04:00
|
|
|
child_theme = Fabricate(:theme, component: true)
|
|
|
|
theme.add_relative_theme!(:child, child_theme)
|
2020-03-11 09:30:45 -04:00
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
expect(SvgSprite.all_icons(theme.id)).not_to include("dragon")
|
2020-03-11 09:30:45 -04:00
|
|
|
|
|
|
|
theme.theme_modifier_set.svg_icons = ["dragon"]
|
|
|
|
theme.save!
|
2021-06-17 22:16:26 -04:00
|
|
|
|
|
|
|
child_theme.theme_modifier_set.svg_icons = ["fly"]
|
|
|
|
child_theme.save!
|
|
|
|
|
|
|
|
icons = SvgSprite.all_icons(theme.id)
|
|
|
|
|
|
|
|
expect(icons).to include("dragon")
|
|
|
|
expect(icons).to include("fly")
|
2020-03-11 09:30:45 -04:00
|
|
|
end
|
|
|
|
|
2022-07-27 12:14:14 -04:00
|
|
|
describe "s3" do
|
2019-05-28 10:57:44 -04:00
|
|
|
let(:upload_s3) { Fabricate(:upload_s3) }
|
|
|
|
|
|
|
|
before do
|
2020-09-14 07:32:25 -04:00
|
|
|
setup_s3
|
2021-06-09 08:29:00 -04:00
|
|
|
body = <<~XML
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
<svg xmlns="http://www.w3.org/2000/svg" style="display: none;">
|
|
|
|
<symbol id="my-custom-theme-icon" viewBox="0 0 496 512">
|
|
|
|
<path d="M248 8C111.03 8 0 119.03 0 256s111.03 248 248 248 248-111.03 248-248S384.97 8 248 8zm0 376c-17.67 0-32-14.33-32-32s14.33-32 32-32 32 14.33 32 32-14.33 32-32 32zm0-128c-53.02 0-96 42.98-96 96s42.98 96 96 96c-106.04 0-192-85.96-192-192S141.96 64 248 64c53.02 0 96 42.98 96 96s-42.98 96-96 96zm0-128c-17.67 0-32 14.33-32 32s14.33 32 32 32 32-14.33 32-32-14.33-32-32-32z"></path>
|
|
|
|
</symbol>
|
|
|
|
</svg>
|
|
|
|
XML
|
|
|
|
stub_request(:get, upload_s3.url).to_return(status: 200, body: body)
|
2019-05-28 10:57:44 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "includes svg sprites in themes stored in s3" do
|
|
|
|
theme.set_field(
|
|
|
|
target: :common,
|
|
|
|
name: SvgSprite.theme_sprite_variable_name,
|
|
|
|
upload_id: upload_s3.id,
|
|
|
|
type: :theme_upload_var,
|
|
|
|
)
|
|
|
|
theme.save!
|
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
sprite_files = SvgSprite.custom_svg_sprites(theme.id).join("|")
|
2021-06-22 14:07:23 -04:00
|
|
|
expect(sprite_files).to match(/my-custom-theme-icon/)
|
2021-06-09 08:29:00 -04:00
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
SvgSprite.bundle(theme.id)
|
2021-06-09 08:29:00 -04:00
|
|
|
expect(SvgSprite.cache.hash.keys).to include("custom_svg_sprites_#{theme.id}")
|
|
|
|
|
|
|
|
external_copy = Discourse.store.download(upload_s3)
|
|
|
|
File.delete external_copy.try(:path)
|
|
|
|
|
2021-06-15 02:57:17 -04:00
|
|
|
SvgSprite.bundle(theme.id)
|
2021-06-22 14:07:23 -04:00
|
|
|
# after a temp file is missing, bundling still works
|
|
|
|
expect(SvgSprite.cache.hash.keys).to include("custom_svg_sprites_#{theme.id}")
|
2019-05-28 10:57:44 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-26 16:49:57 -05:00
|
|
|
it "includes icons from SiteSettings" do
|
|
|
|
SiteSetting.svg_icon_subset = "blender|drafting-compass|fab-bandcamp"
|
|
|
|
|
|
|
|
all_icons = SvgSprite.all_icons
|
|
|
|
expect(all_icons).to include("blender")
|
|
|
|
expect(all_icons).to include("drafting-compass")
|
|
|
|
expect(all_icons).to include("fab-bandcamp")
|
|
|
|
|
|
|
|
SiteSetting.svg_icon_subset = nil
|
2019-02-06 10:51:23 -05:00
|
|
|
SvgSprite.expire_cache
|
2018-11-26 16:49:57 -05:00
|
|
|
expect(SvgSprite.all_icons).not_to include("drafting-compass")
|
2018-11-29 19:11:32 -05:00
|
|
|
|
|
|
|
# does not fail on non-string setting
|
|
|
|
SiteSetting.svg_icon_subset = false
|
2019-02-06 10:51:23 -05:00
|
|
|
SvgSprite.expire_cache
|
2018-11-29 19:11:32 -05:00
|
|
|
expect(SvgSprite.all_icons).to be_truthy
|
2018-11-26 16:49:57 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "includes icons from plugin registry" do
|
|
|
|
DiscoursePluginRegistry.register_svg_icon "blender"
|
|
|
|
DiscoursePluginRegistry.register_svg_icon "fab fa-bandcamp"
|
|
|
|
|
|
|
|
expect(SvgSprite.all_icons).to include("blender")
|
|
|
|
expect(SvgSprite.all_icons).to include("fab-bandcamp")
|
|
|
|
end
|
|
|
|
|
2020-05-26 14:53:32 -04:00
|
|
|
it "includes Font Awesome icon from groups" do
|
2022-04-06 17:58:10 -04:00
|
|
|
_group = Fabricate(:group, flair_icon: "far-building")
|
2020-05-26 14:53:32 -04:00
|
|
|
expect(SvgSprite.bundle).to match(/far-building/)
|
2018-11-26 16:49:57 -05:00
|
|
|
end
|
2021-06-22 14:07:23 -04:00
|
|
|
|
|
|
|
describe "#custom_svg_sprites" do
|
|
|
|
it "is empty by default" do
|
|
|
|
expect(SvgSprite.custom_svg_sprites(nil)).to be_empty
|
|
|
|
expect(SvgSprite.bundle).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
context "with a plugin" do
|
|
|
|
let :plugin1 do
|
|
|
|
plugin1 = Plugin::Instance.new
|
|
|
|
plugin1.path = "#{Rails.root}/spec/fixtures/plugins/my_plugin/plugin.rb"
|
|
|
|
plugin1
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
Discourse.plugins << plugin1
|
|
|
|
plugin1.activate!
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
Discourse.plugins.delete plugin1
|
2021-10-25 04:24:21 -04:00
|
|
|
DiscoursePluginRegistry.reset!
|
2021-06-22 14:07:23 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "includes custom icons from plugins" do
|
|
|
|
expect(SvgSprite.custom_svg_sprites(nil).size).to eq(1)
|
|
|
|
expect(SvgSprite.bundle).to match(/custom-icon/)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it "includes custom icons in a theme" do
|
|
|
|
fname = "custom-theme-icon-sprite.svg"
|
|
|
|
|
|
|
|
upload = UploadCreator.new(file_from_fixtures(fname), fname, for_theme: true).create_for(-1)
|
|
|
|
|
|
|
|
theme.set_field(
|
|
|
|
target: :common,
|
|
|
|
name: SvgSprite.theme_sprite_variable_name,
|
|
|
|
upload_id: upload.id,
|
|
|
|
type: :theme_upload_var,
|
|
|
|
)
|
|
|
|
theme.save!
|
|
|
|
|
2022-07-27 22:27:38 -04:00
|
|
|
expect(Upload.exists?(id: upload.id)).to eq(true)
|
2021-06-22 14:07:23 -04:00
|
|
|
expect(SvgSprite.bundle(theme.id)).to match(/my-custom-theme-icon/)
|
|
|
|
end
|
|
|
|
|
2021-07-14 15:18:29 -04:00
|
|
|
it "does not fail on bad XML in custom icon sprite" do
|
|
|
|
fname = "bad-xml-icon-sprite.svg"
|
|
|
|
|
|
|
|
upload = UploadCreator.new(file_from_fixtures(fname), fname, for_theme: true).create_for(-1)
|
|
|
|
|
|
|
|
theme.set_field(
|
|
|
|
target: :common,
|
|
|
|
name: SvgSprite.theme_sprite_variable_name,
|
|
|
|
upload_id: upload.id,
|
|
|
|
type: :theme_upload_var,
|
|
|
|
)
|
|
|
|
theme.save!
|
|
|
|
|
2022-07-27 22:27:38 -04:00
|
|
|
expect(Upload.exists?(id: upload.id)).to eq(true)
|
2021-07-14 15:18:29 -04:00
|
|
|
expect(SvgSprite.bundle(theme.id)).to match(/arrow-down/)
|
|
|
|
end
|
|
|
|
|
2021-06-22 14:07:23 -04:00
|
|
|
it "includes custom icons in a child theme" do
|
|
|
|
fname = "custom-theme-icon-sprite.svg"
|
|
|
|
child_theme = Fabricate(:theme, component: true)
|
|
|
|
theme.add_relative_theme!(:child, child_theme)
|
|
|
|
|
|
|
|
upload = UploadCreator.new(file_from_fixtures(fname), fname, for_theme: true).create_for(-1)
|
|
|
|
|
|
|
|
child_theme.set_field(
|
|
|
|
target: :common,
|
|
|
|
name: SvgSprite.theme_sprite_variable_name,
|
|
|
|
upload_id: upload.id,
|
|
|
|
type: :theme_upload_var,
|
|
|
|
)
|
|
|
|
child_theme.save!
|
|
|
|
|
2022-07-27 22:27:38 -04:00
|
|
|
expect(Upload.exists?(id: upload.id)).to eq(true)
|
2021-06-22 14:07:23 -04:00
|
|
|
expect(SvgSprite.bundle(theme.id)).to match(/my-custom-theme-icon/)
|
|
|
|
end
|
|
|
|
end
|
2018-11-26 16:49:57 -05:00
|
|
|
end
|