DEV: Convert `home-logo` widget to a glimmer component (#25989)
Part of the larger effort to upgrade the header from widgets to glimmer components: https://github.com/discourse/discourse/pull/25214
This commit is contained in:
parent
04b0f40db8
commit
405bfba3c0
|
@ -3,8 +3,8 @@ import { hash } from "@ember/helper";
|
|||
import { inject as service } from "@ember/service";
|
||||
import and from "truth-helpers/helpers/and";
|
||||
import BootstrapModeNotice from "../bootstrap-mode-notice";
|
||||
import MountWidget from "../mount-widget";
|
||||
import PluginOutlet from "../plugin-outlet";
|
||||
import HomeLogo from "./home-logo";
|
||||
import SidebarToggle from "./sidebar-toggle";
|
||||
import TopicInfo from "./topic/info";
|
||||
|
||||
|
@ -28,10 +28,7 @@ export default class Contents extends Component {
|
|||
|
||||
<div class="home-logo-wrapper-outlet">
|
||||
<PluginOutlet @name="home-logo-wrapper">
|
||||
<MountWidget
|
||||
@widget="home-logo"
|
||||
@args={{hash minimized=this.topicPresent}}
|
||||
/>
|
||||
<HomeLogo @minimized={{this.topicPresent}} />
|
||||
</PluginOutlet>
|
||||
</div>
|
||||
|
||||
|
|
|
@ -0,0 +1,118 @@
|
|||
import Component from "@glimmer/component";
|
||||
import { on } from "@ember/modifier";
|
||||
import { action } from "@ember/object";
|
||||
import { inject as service } from "@ember/service";
|
||||
import concatClass from "discourse/helpers/concat-class";
|
||||
import { wantsNewWindow } from "discourse/lib/intercept-click";
|
||||
import DiscourseURL from "discourse/lib/url";
|
||||
import icon from "discourse-common/helpers/d-icon";
|
||||
import getURL from "discourse-common/lib/get-url";
|
||||
import Logo from "./logo";
|
||||
|
||||
export default class HomeLogo extends Component {
|
||||
@service session;
|
||||
@service site;
|
||||
@service siteSettings;
|
||||
|
||||
href = getURL("/");
|
||||
darkModeAvailable = this.session.darkModeAvailable;
|
||||
|
||||
get showMobileLogo() {
|
||||
return this.site.mobileView && this.logoResolver("mobile_logo").length > 0;
|
||||
}
|
||||
|
||||
get logoUrl() {
|
||||
return this.logoResolver("logo");
|
||||
}
|
||||
|
||||
get logoUrlDark() {
|
||||
return this.logoResolver("logo", { dark: this.darkModeAvailable });
|
||||
}
|
||||
|
||||
get logoSmallUrl() {
|
||||
return this.logoResolver("logo_small");
|
||||
}
|
||||
|
||||
get logoSmallUrlDark() {
|
||||
return this.logoResolver("logo_small", { dark: this.darkModeAvailable });
|
||||
}
|
||||
|
||||
get mobileLogoUrl() {
|
||||
return this.logoResolver("mobile_logo");
|
||||
}
|
||||
|
||||
get mobileLogoUrlDark() {
|
||||
return this.logoResolver("mobile_logo", { dark: this.darkModeAvailable });
|
||||
}
|
||||
|
||||
logoResolver(name, opts = {}) {
|
||||
// get alternative logos for browser dark dark mode switching
|
||||
if (opts.dark) {
|
||||
return this.siteSettings[`site_${name}_dark_url`];
|
||||
}
|
||||
|
||||
// try dark logos first when color scheme is dark
|
||||
// this is independent of browser dark mode
|
||||
// hence the fallback to normal logos
|
||||
if (this.session.defaultColorSchemeIsDark) {
|
||||
return (
|
||||
this.siteSettings[`site_${name}_dark_url`] ||
|
||||
this.siteSettings[`site_${name}_url`] ||
|
||||
""
|
||||
);
|
||||
}
|
||||
|
||||
return this.siteSettings[`site_${name}_url`] || "";
|
||||
}
|
||||
|
||||
@action
|
||||
click(e) {
|
||||
if (wantsNewWindow(e)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
e.preventDefault();
|
||||
DiscourseURL.routeToTag(e.target.closest("a"));
|
||||
}
|
||||
|
||||
<template>
|
||||
{{! template-lint-disable no-invalid-interactive }}
|
||||
<div
|
||||
class={{concatClass (if @minimized "title--minimized") "title"}}
|
||||
{{on "click" this.click}}
|
||||
>
|
||||
<a href={{this.href}} data-auto-route="true">
|
||||
{{#if @minimized}}
|
||||
{{#if this.logoSmallUrl}}
|
||||
<Logo
|
||||
@key="logo-small"
|
||||
@url={{this.logoSmallUrl}}
|
||||
@title={{this.siteSettings.title}}
|
||||
@darkUrl={{this.logoSmallUrlDark}}
|
||||
/>
|
||||
{{else}}
|
||||
{{icon "home"}}
|
||||
{{/if}}
|
||||
{{else if this.showMobileLogo}}
|
||||
<Logo
|
||||
@key="logo-mobile"
|
||||
@url={{this.mobileLogoUrl}}
|
||||
@title={{this.siteSettings.title}}
|
||||
@darkUrl={{this.mobileLogoUrlDark}}
|
||||
/>
|
||||
{{else if this.logoUrl}}
|
||||
<Logo
|
||||
@key="logo-big"
|
||||
@url={{this.logoUrl}}
|
||||
@title={{this.siteSettings.title}}
|
||||
@darkUrl={{this.logoUrlDark}}
|
||||
/>
|
||||
{{else}}
|
||||
<h1 id="site-text-logo" class="text-logo">
|
||||
{{this.siteSettings.title}}
|
||||
</h1>
|
||||
{{/if}}
|
||||
</a>
|
||||
</div>
|
||||
</template>
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
import getURL from "discourse-common/lib/get-url";
|
||||
import and from "truth-helpers/helpers/and";
|
||||
import eq from "truth-helpers/helpers/eq";
|
||||
import notEq from "truth-helpers/helpers/not-eq";
|
||||
|
||||
const Logo = <template>
|
||||
{{#if (and @darkUrl (notEq @url @darkUrl))}}
|
||||
<picture>
|
||||
<source srcset={{getURL @darkUrl}} media="(prefers-color-scheme: dark)" />
|
||||
<img
|
||||
id="site-logo"
|
||||
class={{@key}}
|
||||
src={{getURL @url}}
|
||||
width={{if (eq @key "logo-small") "36"}}
|
||||
alt={{@title}}
|
||||
/>
|
||||
</picture>
|
||||
{{else}}
|
||||
<img
|
||||
id="site-logo"
|
||||
class={{@key}}
|
||||
src={{getURL @url}}
|
||||
width={{if (eq @key "logo-small") "36"}}
|
||||
alt={{@title}}
|
||||
/>
|
||||
{{/if}}
|
||||
</template>;
|
||||
|
||||
export default Logo;
|
|
@ -156,6 +156,7 @@ const DEPRECATED_HEADER_WIDGETS = [
|
|||
"header-icons",
|
||||
"header-topic-info",
|
||||
"header-notifications",
|
||||
"home-logo",
|
||||
];
|
||||
|
||||
// This helper prevents us from applying the same `modifyClass` over and over in test mode.
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// deprecated in favor of components/glimmer-header/home-logo.gjs
|
||||
import { h } from "virtual-dom";
|
||||
import { wantsNewWindow } from "discourse/lib/intercept-click";
|
||||
import DiscourseURL from "discourse/lib/url";
|
||||
|
|
|
@ -0,0 +1,175 @@
|
|||
import { getOwner } from "@ember/application";
|
||||
import { render } from "@ember/test-helpers";
|
||||
import { module, test } from "qunit";
|
||||
import HomeLogo from "discourse/components/glimmer-header/home-logo";
|
||||
import { setupRenderingTest } from "discourse/tests/helpers/component-test";
|
||||
|
||||
const bigLogo = "/images/d-logo-sketch.png?test";
|
||||
const smallLogo = "/images/d-logo-sketch-small.png?test";
|
||||
const mobileLogo = "/images/d-logo-sketch.png?mobile";
|
||||
const darkLogo = "/images/d-logo-sketch.png?dark";
|
||||
const title = "Cool Forum";
|
||||
const prefersDark = "(prefers-color-scheme: dark)";
|
||||
|
||||
module("Integration | Component | home-logo", function (hooks) {
|
||||
setupRenderingTest(hooks);
|
||||
|
||||
hooks.afterEach(function () {
|
||||
this.session = getOwner(this).lookup("service:session");
|
||||
this.session.set("darkModeAvailable", null);
|
||||
this.session.set("defaultColorSchemeIsDark", null);
|
||||
});
|
||||
|
||||
test("basics", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_small_url = smallLogo;
|
||||
this.siteSettings.title = title;
|
||||
|
||||
await render(<template><HomeLogo @minimized={{false}} /></template>);
|
||||
assert.dom(".title").exists({ count: 1 });
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", bigLogo);
|
||||
assert.dom("#site-logo").hasAttribute("alt", title);
|
||||
});
|
||||
|
||||
test("basics - minimized", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_small_url = smallLogo;
|
||||
this.siteSettings.title = title;
|
||||
|
||||
await render(<template><HomeLogo @minimized={{true}} /></template>);
|
||||
assert.dom("img.logo-small").exists({ count: 1 });
|
||||
assert.dom("img.logo-small").hasAttribute("src", smallLogo);
|
||||
assert.dom("img.logo-small").hasAttribute("alt", title);
|
||||
assert.dom("img.logo-small").hasAttribute("width", "36");
|
||||
});
|
||||
|
||||
test("no logo", async function (assert) {
|
||||
this.siteSettings.site_logo_url = "";
|
||||
this.siteSettings.site_logo_small_url = "";
|
||||
this.siteSettings.title = title;
|
||||
|
||||
await render(<template><HomeLogo @minimized={{false}} /></template>);
|
||||
assert.dom("h1#site-text-logo.text-logo").exists({ count: 1 });
|
||||
assert.dom("#site-text-logo").hasText(title, "has title as text logo");
|
||||
});
|
||||
|
||||
test("no logo - minimized", async function (assert) {
|
||||
this.siteSettings.site_logo_url = "";
|
||||
this.siteSettings.site_logo_small_url = "";
|
||||
this.siteSettings.title = title;
|
||||
|
||||
await render(<template><HomeLogo @minimized={{true}} /></template>);
|
||||
assert.dom(".d-icon-home").exists({ count: 1 });
|
||||
});
|
||||
|
||||
test("mobile logo", async function (assert) {
|
||||
this.siteSettings.site_mobile_logo_url = mobileLogo;
|
||||
this.siteSettings.site_logo_small_url = smallLogo;
|
||||
this.site.mobileView = true;
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-mobile").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", mobileLogo);
|
||||
});
|
||||
|
||||
test("mobile without logo", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.site.mobileView = true;
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", bigLogo);
|
||||
});
|
||||
|
||||
test("logo with dark mode alternative", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_dark_url = darkLogo;
|
||||
this.session.set("darkModeAvailable", true);
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", bigLogo);
|
||||
assert
|
||||
.dom("picture source")
|
||||
.hasAttribute("media", prefersDark, "includes dark mode media attribute");
|
||||
assert
|
||||
.dom("picture source")
|
||||
.hasAttribute(
|
||||
"srcset",
|
||||
darkLogo,
|
||||
"includes dark mode alternative logo source"
|
||||
);
|
||||
});
|
||||
|
||||
test("mobile logo with dark mode alternative", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_mobile_logo_url = mobileLogo;
|
||||
this.siteSettings.site_mobile_logo_dark_url = darkLogo;
|
||||
this.session.set("darkModeAvailable", true);
|
||||
|
||||
this.site.mobileView = true;
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
|
||||
assert.dom("#site-logo").hasAttribute("src", mobileLogo);
|
||||
assert
|
||||
.dom("picture source")
|
||||
.hasAttribute("media", prefersDark, "includes dark mode media attribute");
|
||||
assert
|
||||
.dom("picture source")
|
||||
.hasAttribute(
|
||||
"srcset",
|
||||
darkLogo,
|
||||
"includes dark mode alternative logo source"
|
||||
);
|
||||
});
|
||||
|
||||
test("dark mode enabled but no dark logo set", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_dark_url = "";
|
||||
this.session.set("darkModeAvailable", true);
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", bigLogo);
|
||||
assert.dom("picture").doesNotExist("does not include alternative logo");
|
||||
});
|
||||
|
||||
test("dark logo set but no dark mode", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_dark_url = darkLogo;
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", bigLogo);
|
||||
assert.dom("picture").doesNotExist("does not include alternative logo");
|
||||
});
|
||||
|
||||
test("dark color scheme and dark logo set", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_dark_url = darkLogo;
|
||||
this.session.set("defaultColorSchemeIsDark", true);
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert.dom("#site-logo").hasAttribute("src", darkLogo, "uses dark logo");
|
||||
assert.dom("picture").doesNotExist("does not add dark mode alternative");
|
||||
});
|
||||
|
||||
test("dark color scheme and dark logo not set", async function (assert) {
|
||||
this.siteSettings.site_logo_url = bigLogo;
|
||||
this.siteSettings.site_logo_dark_url = "";
|
||||
this.session.set("defaultColorSchemeIsDark", true);
|
||||
|
||||
await render(<template><HomeLogo /></template>);
|
||||
assert.dom("img#site-logo.logo-big").exists({ count: 1 });
|
||||
assert
|
||||
.dom("#site-logo")
|
||||
.hasAttribute(
|
||||
"src",
|
||||
bigLogo,
|
||||
"uses regular logo on dark scheme if no dark logo"
|
||||
);
|
||||
});
|
||||
});
|
|
@ -1,3 +1,4 @@
|
|||
// deprecated in favor of discourse/tests/integration/components/home-logo-test.gjs
|
||||
import { getOwner } from "@ember/application";
|
||||
import { render } from "@ember/test-helpers";
|
||||
import { module, test } from "qunit";
|
||||
|
|
Loading…
Reference in New Issue