add shared items, relative dates, tags; CSS fixes

This commit is contained in:
ashastral 2024-09-18 17:42:45 -07:00
parent 35a477bf07
commit a31ab6b8ea
2 changed files with 732 additions and 315 deletions

View file

@ -1,42 +1,40 @@
/* atkinson-hyperlegible-regular - latin_latin-ext */ /* Atkinson Hyperlegible */
@font-face { @font-face {
font-display: swap; /* Check https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display for other options. */ font-display: swap;
font-family: "Atkinson Hyperlegible"; font-family: "Atkinson Hyperlegible";
font-style: normal; font-style: normal;
font-weight: 400; font-weight: 400;
src: url("../fonts/atkinson-hyperlegible-v11-latin_latin-ext-regular.woff2") src: url("../fonts/atkinson-hyperlegible-v11-latin_latin-ext-regular.woff2")
format("woff2"); /* Chrome 36+, Opera 23+, Firefox 39+, Safari 12+, iOS 10+ */ format("woff2");
} }
/* atkinson-hyperlegible-italic - latin_latin-ext */
@font-face { @font-face {
font-display: swap; /* Check https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display for other options. */ font-display: swap;
font-family: "Atkinson Hyperlegible"; font-family: "Atkinson Hyperlegible";
font-style: italic; font-style: italic;
font-weight: 400; font-weight: 400;
src: url("../fonts/atkinson-hyperlegible-v11-latin_latin-ext-italic.woff2") src: url("../fonts/atkinson-hyperlegible-v11-latin_latin-ext-italic.woff2")
format("woff2"); /* Chrome 36+, Opera 23+, Firefox 39+, Safari 12+, iOS 10+ */ format("woff2");
} }
/* atkinson-hyperlegible-700 - latin_latin-ext */
@font-face { @font-face {
font-display: swap; /* Check https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display for other options. */ font-display: swap;
font-family: "Atkinson Hyperlegible"; font-family: "Atkinson Hyperlegible";
font-style: normal; font-style: normal;
font-weight: 700; font-weight: 700;
src: url("../fonts/atkinson-hyperlegible-v11-latin_latin-ext-700.woff2") src: url("../fonts/atkinson-hyperlegible-v11-latin_latin-ext-700.woff2")
format("woff2"); /* Chrome 36+, Opera 23+, Firefox 39+, Safari 12+, iOS 10+ */ format("woff2");
} }
/* atkinson-hyperlegible-700italic - latin_latin-ext */
@font-face { @font-face {
font-display: swap; /* Check https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display for other options. */ font-display: swap;
font-family: "Atkinson Hyperlegible"; font-family: "Atkinson Hyperlegible";
font-style: italic; font-style: italic;
font-weight: 700; font-weight: 700;
src: url("./fonts/atkinson-hyperlegible-v11-latin_latin-ext-700italic.woff2") src: url("./fonts/atkinson-hyperlegible-v11-latin_latin-ext-700italic.woff2")
format("woff2"); /* Chrome 36+, Opera 23+, Firefox 39+, Safari 12+, iOS 10+ */ format("woff2");
} }
/* Keyframes */
@keyframes bounce { @keyframes bounce {
0%, 0%,
100% { 100% {
@ -111,35 +109,37 @@
} }
} }
cohost-wc * { /* Cohost post content styling */
cohost-post * {
box-sizing: border-box; box-sizing: border-box;
border-width: 0; border-width: 0;
border-style: solid; border-style: solid;
border-color: #ded9d3; border-color: #ded9d3;
} }
cohost-wc a { cohost-post a {
color: inherit; color: inherit;
text-decoration: inherit; text-decoration: inherit;
} }
cohost-wc hr { cohost-post hr {
height: 0; height: 0;
color: inherit; color: inherit;
border-top-width: 1px; border-top-width: 1px;
} }
cohost-wc blockquote, cohost-post blockquote,
cohost-wc dl, cohost-post dl,
cohost-wc dd, cohost-post dd,
cohost-wc h1, cohost-post h1,
cohost-wc h2, cohost-post h2,
cohost-wc h3, cohost-post h3,
cohost-wc h4, cohost-post h4,
cohost-wc h5, cohost-post h5,
cohost-wc h6, cohost-post h6,
cohost-wc hr, cohost-post hr,
cohost-wc figure, cohost-post figure,
cohost-wc p, cohost-post p,
cohost-wc pre { cohost-post pre {
margin: 0; margin: 0;
} }
.prose { .prose {
@ -163,7 +163,9 @@ cohost-wc pre {
font-size: 1em; font-size: 1em;
line-height: 1.75; line-height: 1.75;
} }
.prose :where(h3):not(:where([class~="not-prose"], [class~="not-prose"] *)) { cohost-post
.prose
:where(h3):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
color: var(--tw-prose-headings); color: var(--tw-prose-headings);
font-weight: 600; font-weight: 600;
font-size: 1.25em; font-size: 1.25em;
@ -171,15 +173,142 @@ cohost-wc pre {
margin-bottom: 0.6em; margin-bottom: 0.6em;
line-height: 1.6; line-height: 1.6;
} }
.prose cohost-post
:where(.prose > :last-child):not( .prose
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-bottom: 0;
}
.prose
:where(.prose > :first-child):not( :where(.prose > :first-child):not(
:where([class~="not-prose"], [class~="not-prose"] *) :where([class~="not-prose"], [class~="not-prose"] *)
) { ) {
margin-top: 0; margin-top: 0;
} }
cohost-post
.prose
:where(p):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
margin-top: 1.25em;
margin-bottom: 1.25em;
}
cohost-post
.prose
:where(hr):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
border-color: var(--tw-prose-hr);
border-top-width: 1px;
margin-top: 1em;
margin-bottom: 1em;
}
cohost-post
.prose
:where(a):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
text-decoration: underline;
font-weight: 500;
}
cohost-post
.prose
:where(ul):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
list-style-type: disc;
}
cohost-post
.prose
:where(ul > li):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
padding-inline-start: 0.375em;
}
cohost-post
.prose
:where(li):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
margin-top: 0.5em;
margin-bottom: 0.5em;
}
cohost-post
.prose
:where(.prose > :last-child):not(
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-bottom: 0;
}
cohost-post
.prose
:where(.prose > :first-child):not(
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-top: 0;
}
cohost-post .co-embed {
background-color: light-dark(rgb(255 241 223), rgb(25 25 25));
}
cohost-post .co-ui-text {
color: light-dark(rgb(25 25 25), rgb(255 249 242));
}
/* Hide interactive UI */
cohost-post button.co-link-button {
display: none;
}
/* Tailwind properties */
cohost-post .text-right {
text-align: right;
}
cohost-post .p-3 {
padding: 0.75rem;
}
cohost-post .mt-0 {
margin-top: 0px;
}
cohost-post .my-4 {
margin-top: 1rem;
margin-bottom: 1rem;
}
cohost-post .overflow-hidden {
overflow: hidden;
}
cohost-post .break-words {
overflow-wrap: break-word;
}
cohost-post .px-3 {
padding-left: 0.75rem;
padding-right: 0.75rem;
}
/* TODO: light */
@media (prefers-color-scheme: dark) {
cohost-post {
--color-notWhite: 255 249 242;
--color-notBlack: 25 25 25;
--color-cherry: 131 37 79;
--color-strawberry: 229 107 111;
--color-mango: 255 171 92;
--color-longan: 255 216 168;
--color-tertiary: var(--color-longan);
--color-tertiary-200: 255 229 196;
--color-tertiary-300: 255 216 168;
--color-tertiary-400: 255 202 122;
--color-tertiary-500: 183 133 61;
--color-tertiary-600: 183 133 61;
--color-tertiary-700: 132 94 38;
--color-text: var(--color-notWhite);
--color-bg-text: var(--color-notWhite);
--color-foreground-100: 253 206 224;
--color-foreground-200: 238 173 199;
--color-foreground-300: 211 116 155;
--color-foreground-400: 174 68 115;
--color-foreground-500: 131 37 79;
--color-foreground-600: 103 26 61;
--color-foreground-700: 81 17 46;
--color-foreground-800: 59 9 32;
--color-foreground: var(--color-cherry);
--color-secondary-200: 255 208 172;
--color-secondary-300: 255 191 131;
--color-secondary-400: 255 171 92;
--color-secondary-600: 188 109 40;
--color-secondary-700: 147 74 21;
--color-secondary: var(--color-mango);
--color-accent: var(--color-mango);
--color-background: var(--color-notBlack);
--color-sidebar-bg: var(--color-notBlack);
--color-sidebar-text: var(--color-notWhite);
--color-sidebar-accent: var(--color-mango);
--color-compose-button: var(--color-foreground);
--color-compose-button-400: var(--color-foreground-400);
--color-compose-button-600: var(--color-foreground-600);
--emoji-scale: 1.25em;
}
}

View file

@ -1,46 +1,251 @@
import { LitElement, html, css } from "lit"; import { LitElement, html, css } from "lit";
import { customElement, property, state } from "lit/decorators.js"; import { customElement, property } from "lit/decorators.js";
import { when } from "lit/directives/when.js"; import { when } from "lit/directives/when.js";
import { ifDefined } from "lit/directives/if-defined.js";
@customElement("cohost-wc") function formatRelativeTime(date: Date) {
export class CohostWebComponent extends LitElement { const millisecondsPerSecond = 1000;
const secondsPerMinute = 60;
const minutesPerHour = 60;
const hoursPerDay = 24;
const daysPerMonth = 30; // JS date handling moment
const intervals = {
month:
millisecondsPerSecond *
secondsPerMinute *
minutesPerHour *
hoursPerDay *
daysPerMonth,
day:
millisecondsPerSecond * secondsPerMinute * minutesPerHour * hoursPerDay,
hour: millisecondsPerSecond * secondsPerMinute * minutesPerHour,
minute: millisecondsPerSecond * secondsPerMinute,
second: millisecondsPerSecond,
} as const;
const relativeDateFormat = new Intl.RelativeTimeFormat("en", {
style: "short",
});
const diff = date.getTime() - new Date().getTime();
for (const interval in intervals) {
const _interval = interval as keyof typeof intervals;
if (intervals[_interval] <= Math.abs(diff)) {
return relativeDateFormat.format(
Math.trunc(diff / intervals[_interval]),
_interval,
);
}
}
return relativeDateFormat.format(diff / 1000, "second");
}
const COMMON_CSS = css`
:host {
color-scheme: light dark;
font-family: "Atkinson Hyperlegible", ui-sans-serif, system-ui, sans-serif,
"Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol",
"Noto Color Emoji";
}
* {
box-sizing: border-box;
border-width: 0;
border-style: solid;
border-color: #ded9d3;
}
a {
color: inherit;
text-decoration: inherit;
}
hr {
height: 0;
color: inherit;
border-top-width: 1px;
}
blockquote,
dl,
dd,
h1,
h2,
h3,
h4,
h5,
h6,
hr,
figure,
p,
pre {
margin: 0;
}
.post-anchor {
position: relative;
top: -5rem;
}
.co-avatar-container {
width: 2rem;
height: 2rem;
aspect-ratio: 1 / 1;
display: inline-block;
position: relative;
}
.mask {
-webkit-mask-size: contain;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-position: center;
mask-position: center;
}
.mask-squircle {
-webkit-mask-image: url();
mask-image: url();
}
.mask-roundrect {
border-radius: 12.5%;
}
.mask-circle {
border-radius: 9999px;
}
/* .mask-egg {
-webkit-mask-image: url(/static/de7a6730ae8672a12406.svg);
mask-image: url(/static/de7a6730ae8672a12406.svg);
}
.mask-capsule-big {
-webkit-mask-image: url(/static/3e2b26f2f1e719024296.svg);
mask-image: url(/static/3e2b26f2f1e719024296.svg);
}
.mask-capsule-small {
-webkit-mask-image: url(/static/8ff201350af3c70fb5b8.svg);
mask-image: url(/static/8ff201350af3c70fb5b8.svg);
} */
.co-avatar {
width: 100%;
height: 100%;
-o-object-fit: cover;
object-fit: cover;
}
.co-project-display-name {
max-width: 100%;
flex-shrink: 1;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
font-weight: 700;
color: light-dark(rgb(25 25 25), rgb(255 249 242));
}
.co-project-display-name:hover {
text-decoration: underline;
}
.co-project-handle {
color: light-dark(rgb(74 72 71), rgb(222 217 211));
}
.co-project-handle:hover {
text-decoration: underline;
}
.co-thread-header time,
.co-post-header time {
display: block;
flex: none;
font-size: 0.75rem;
line-height: 1rem;
font-variant-numeric: tabular-nums;
color: rgb(130 127 124);
}
.co-thread-header time a:hover,
.co-post-header time a:hover {
text-decoration: underline;
}
.co-hairline {
border-color: light-dark(rgb(191 186 181), rgb(74 72 71));
}
.post-headline-container {
display: flex;
flex-direction: row;
width: 100%;
padding: 0.75rem;
}
.post-headline-a {
overflow-wrap: break-word;
align-self: center;
flex-grow: 1;
}
.post-headline-a:hover {
text-decoration: underline;
}
.prose {
color: light-dark(#191919, #fff9f2);
--tw-prose-body: light-dark(#374151, #fff9f2);
--tw-prose-headings: light-dark(#111827, #fff9f2);
--tw-prose-lead: light-dark(#4b5563, #fff9f2);
--tw-prose-links: light-dark(#111827, #fff9f2);
--tw-prose-bold: light-dark(#111827, #fff9f2);
--tw-prose-counters: light-dark(#6b7280, #fff9f2);
--tw-prose-bullets: light-dark(#d1d5db, #fff9f2);
--tw-prose-hr: light-dark(#e5e7eb, #bfbab5);
--tw-prose-quotes: light-dark(#111827, #fff9f2);
--tw-prose-quote-borders: light-dark(#e5e7eb, #bfbab5);
--tw-prose-captions: light-dark(#6b7280, #fff9f2);
--tw-prose-code: light-dark(#111827, #fff9f2);
--tw-prose-pre-code: light-dark(#e5e7eb, #fff9f2);
--tw-prose-pre-bg: light-dark(#1f2937, #191919);
--tw-prose-th-borders: light-dark(#d1d5db, #bfbab5);
--tw-prose-td-borders: light-dark(#e5e7eb, #bfbab5);
font-size: 1em;
line-height: 1.75;
}
.prose :where(h3):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
color: var(--tw-prose-headings);
font-weight: 600;
font-size: 1.25em;
margin-top: 1.6em;
margin-bottom: 0.6em;
line-height: 1.6;
}
.prose
:where(.prose > :last-child):not(
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-bottom: 0;
}
.prose
:where(.prose > :first-child):not(
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-top: 0;
}
.post-body {
overflow: clip;
contain: paint;
overflow: hidden;
isolation: isolate;
position: relative;
}
.co-tags-container {
width: 100%;
max-width: 100%;
padding: 0.75rem;
}
.co-tags {
position: relative;
width: 100%;
overflow-y: hidden;
overflow-wrap: break-word;
line-height: 1;
color: light-dark(rgb(104 102 100), rgb(191 186 181));
}
.co-tags a {
display: inline-block;
margin-right: 0.5rem;
font-size: 0.875rem;
line-height: 1.25rem;
}
.co-tags a:hover {
text-decoration: underline;
}
`;
@customElement("cohost-post")
export class CohostPost extends LitElement {
static override styles = css` static override styles = css`
:host { ${COMMON_CSS}
color-scheme: light dark;
font-family: "Atkinson Hyperlegible", ui-sans-serif, system-ui, sans-serif,
"Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol",
"Noto Color Emoji";
}
* {
box-sizing: border-box;
border-width: 0;
border-style: solid;
border-color: #ded9d3;
}
a {
color: inherit;
text-decoration: inherit;
}
hr {
height: 0;
color: inherit;
border-top-width: 1px;
}
blockquote,
dl,
dd,
h1,
h2,
h3,
h4,
h5,
h6,
hr,
figure,
p,
pre {
margin: 0;
}
.co-post-box { .co-post-box {
width: 100%; width: 100%;
overflow-x: auto; overflow-x: auto;
@ -84,151 +289,9 @@ export class CohostWebComponent extends LitElement {
min-width: 0px; min-width: 0px;
display: flex; display: flex;
} }
.co-avatar-container { .header-share-icon {
width: 2rem; width: 1.25rem;
height: 2rem; height: 1.25rem;
aspect-ratio: 1 / 1;
display: inline-block;
position: relative;
}
.mask {
-webkit-mask-size: contain;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-position: center;
mask-position: center;
}
.mask-squircle {
-webkit-mask-image: url();
mask-image: url();
}
.mask-roundrect {
border-radius: 12.5%;
}
.mask-circle {
border-radius: 9999px;
}
/* .mask-egg {
-webkit-mask-image: url(/static/de7a6730ae8672a12406.svg);
mask-image: url(/static/de7a6730ae8672a12406.svg);
}
.mask-capsule-big {
-webkit-mask-image: url(/static/3e2b26f2f1e719024296.svg);
mask-image: url(/static/3e2b26f2f1e719024296.svg);
}
.mask-capsule-small {
-webkit-mask-image: url(/static/8ff201350af3c70fb5b8.svg);
mask-image: url(/static/8ff201350af3c70fb5b8.svg);
} */
.co-avatar {
width: 100%;
height: 100%;
-o-object-fit: cover;
object-fit: cover;
}
.co-project-display-name {
max-width: 100%;
flex-shrink: 1;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
font-weight: 700;
color: light-dark(rgb(25 25 25), rgb(255 249 242));
}
.co-project-display-name:hover {
text-decoration: underline;
}
.co-project-handle {
color: light-dark(rgb(74 72 71), rgb(222 217 211));
}
.co-project-handle:hover {
text-decoration: underline;
}
.co-thread-header time {
display: block;
flex: none;
font-size: 0.75rem;
line-height: 1rem;
font-variant-numeric: tabular-nums;
color: rgb(130 127 124);
}
.co-thread-header time a:hover {
text-decoration: underline;
}
.co-hairline {
border-color: light-dark(rgb(191 186 181), rgb(74 72 71));
}
.post-title-container {
display: flex;
flex-direction: row;
width: 100%;
padding: 0.75rem;
}
.post-title-a {
overflow-wrap: break-word;
align-self: center;
flex-grow: 1;
}
.post-title-a:hover {
text-decoration: underline;
}
.prose {
color: light-dark(#191919, #fff9f2);
--tw-prose-body: light-dark(#374151, #fff9f2);
--tw-prose-headings: light-dark(#111827, #fff9f2);
--tw-prose-lead: light-dark(#4b5563, #fff9f2);
--tw-prose-links: light-dark(#111827, #fff9f2);
--tw-prose-bold: light-dark(#111827, #fff9f2);
--tw-prose-counters: light-dark(#6b7280, #fff9f2);
--tw-prose-bullets: light-dark(#d1d5db, #fff9f2);
--tw-prose-hr: light-dark(#e5e7eb, #bfbab5);
--tw-prose-quotes: light-dark(#111827, #fff9f2);
--tw-prose-quote-borders: light-dark(#e5e7eb, #bfbab5);
--tw-prose-captions: light-dark(#6b7280, #fff9f2);
--tw-prose-code: light-dark(#111827, #fff9f2);
--tw-prose-pre-code: light-dark(#e5e7eb, #fff9f2);
--tw-prose-pre-bg: light-dark(#1f2937, #191919);
--tw-prose-th-borders: light-dark(#d1d5db, #bfbab5);
--tw-prose-td-borders: light-dark(#e5e7eb, #bfbab5);
font-size: 1em;
line-height: 1.75;
}
.prose
:where(h3):not(:where([class~="not-prose"], [class~="not-prose"] *)) {
color: var(--tw-prose-headings);
font-weight: 600;
font-size: 1.25em;
margin-top: 1.6em;
margin-bottom: 0.6em;
line-height: 1.6;
}
.prose
:where(.prose > :last-child):not(
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-bottom: 0;
}
.prose
:where(.prose > :first-child):not(
:where([class~="not-prose"], [class~="not-prose"] *)
) {
margin-top: 0;
}
.post-body {
overflow: clip;
contain: paint;
overflow: hidden;
isolation: isolate;
position: relative;
}
.post-body-section {
padding-left: 0.75rem;
padding-right: 0.75rem;
overflow-wrap: break-word;
overflow: hidden;
margin-top: 1rem;
margin-bottom: 1rem;
} }
.co-thread-footer { .co-thread-footer {
width: 100%; width: 100%;
@ -281,41 +344,50 @@ export class CohostWebComponent extends LitElement {
displayName = "Display Name"; displayName = "Display Name";
@property() @property()
username = "username"; handle = "handle";
@property() @property()
iso8601Timestamp = "2022-06-28T18:03:00Z"; publishedAt: string | undefined;
@property() @property()
permalink = "https://cohost.org/"; permalink: string | undefined;
@property({ type: Boolean })
sharedItems: boolean = false;
@property() @property()
postId: string | undefined = undefined; postId: string | undefined;
@property() @property()
postTitle: string | undefined = undefined; singlePostHeadline: string | undefined;
@property() @property()
sharedAvatarSrc: string | undefined = undefined; sharedAvatarSrc: string | undefined;
@property() @property()
sharedDisplayName: string | undefined = undefined; sharedAvatarShape: string | undefined;
@property() @property()
sharedUsername: string | undefined = undefined; sharedDisplayName: string | undefined;
@property() @property()
tags: string | undefined = undefined; sharedHandle: string | undefined;
@property()
tags: string | undefined;
@property({ type: Number }) @property({ type: Number })
comments = 0; comments = 0;
@state()
protected _displayTimestamp = new Date(
this.iso8601Timestamp,
).toLocaleString();
override render() { override render() {
let displayTimestamp = undefined;
let relativeTimestamp = undefined;
if (this.publishedAt) {
let date = new Date(this.publishedAt);
displayTimestamp = date.toLocaleString();
relativeTimestamp = formatRelativeTime(date);
}
return html` return html`
<article <article
data-theme="both" data-theme="both"
@ -324,98 +396,186 @@ export class CohostWebComponent extends LitElement {
> >
<header class="co-thread-header"> <header class="co-thread-header">
<div class="co-thread-header-1"> <div class="co-thread-header-1">
<div class="co-avatar-container mask"> ${when(
<img this.avatarSrc,
src="${this.avatarSrc}" () => html`
class="co-avatar mask mask-${this.avatarShape}" <div class="co-avatar-container mask">
alt="${this.displayName}" <img
/> src="${this.avatarSrc}"
</div> class="co-avatar mask mask-${this.avatarShape}"
<a alt="${this.displayName}"
rel="author" />
href="https://cohost.org/${this.username}" </div>
class="co-project-display-name" `,
title="${this.displayName}" )}
data-url-popovers="" ${when(
style="user-select: none;" this.displayName,
data-longpress-event="true" () => html`
>${this.displayName}</a <a
><a rel="author"
href="https://cohost.org/${this.username}" href="https://cohost.org/${this.handle}"
class="co-project-handle" class="co-project-display-name"
data-url-popovers="" title="${this.displayName}"
style="user-select: none;" data-url-popovers=""
data-longpress-event="true" style="user-select: none;"
>@${this.username}</a data-longpress-event="true"
><time >${this.displayName}</a
datetime="${this.iso8601Timestamp}" >
class="block flex-none text-xs tabular-nums text-gray-500" `,
title="${this._displayTimestamp}" )}
><a href="${this.permalink}" class="hover:underline" ${when(
>${this._displayTimestamp}</a this.handle,
></time () =>
> html` <a
href="https://cohost.org/${this.handle}"
class="co-project-handle"
data-url-popovers=""
style="user-select: none;"
data-longpress-event="true"
>@${this.handle}</a
>`,
)}
${when(
this.publishedAt,
() => html`
<time
datetime="${ifDefined(this.publishedAt)}"
title="${ifDefined(displayTimestamp)}"
><a
href="${ifDefined(this.permalink)}"
class="hover:underline"
>${relativeTimestamp}</a
></time
>
`,
)}
${when(
this.sharedAvatarSrc ||
this.sharedAvatarShape ||
this.sharedDisplayName ||
this.sharedHandle,
() => html`
<svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"
fill="currentColor"
aria-hidden="true"
class="co-project-display-name header-share-icon"
>
<path
fill-rule="evenodd"
d="M4.755 10.059a7.5 7.5 0 0112.548-3.364l1.903 1.903h-3.183a.75.75 0 100 1.5h4.992a.75.75 0 00.75-.75V4.356a.75.75 0 00-1.5 0v3.18l-1.9-1.9A9 9 0 003.306 9.67a.75.75 0 101.45.388zm15.408 3.352a.75.75 0 00-.919.53 7.5 7.5 0 01-12.548 3.364l-1.902-1.903h3.183a.75.75 0 000-1.5H2.984a.75.75 0 00-.75.75v4.992a.75.75 0 001.5 0v-3.18l1.9 1.9a9 9 0 0015.059-4.035.75.75 0 00-.53-.918z"
clip-rule="evenodd"
></path>
</svg>
`,
)}
${when(
this.sharedAvatarSrc,
() => html`
<div class="co-avatar-container mask">
<img
src="${ifDefined(this.sharedAvatarSrc)}"
class="co-avatar mask mask-${this.sharedAvatarShape}"
alt="${ifDefined(this.sharedDisplayName)}"
/>
</div>
`,
)}
${when(
this.sharedDisplayName,
() => html`
<a
href="https://cohost.org/${this.sharedHandle}"
class="co-project-display-name"
title="${ifDefined(this.sharedDisplayName)}"
data-url-popovers=""
style="user-select: none;"
data-longpress-event="true"
>${this.sharedDisplayName}</a
>
`,
)}
${when(
this.sharedHandle,
() =>
html` <a
href="https://cohost.org/${this.sharedHandle}"
class="co-project-handle"
data-url-popovers=""
style="user-select: none;"
data-longpress-event="true"
>@${this.sharedHandle}</a
>`,
)}
</div> </div>
<!--<button
id="headlessui-menu-button"
type="button"
aria-haspopup="menu"
aria-expanded="false"
data-headlessui-state=""
>
<svg
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
stroke-width="1.5"
stroke="currentColor"
aria-hidden="true"
class="co-action-button h-6 w-6 transition-transform ui-open:rotate-90"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
d="M6.75 12a.75.75 0 11-1.5 0 .75.75 0 011.5 0zM12.75 12a.75.75 0 11-1.5 0 .75.75 0 011.5 0zM18.75 12a.75.75 0 11-1.5 0 .75.75 0 011.5 0z"
></path>
</svg>
</button>-->
</header> </header>
<hr class="co-hairline" /> <hr class="co-hairline" />
<div> <div>
${when( ${when(
this.postId, this.postId,
() => html` () => html`
<div id="post-${this.postId}" class="relative -top-20"></div> <div id="post-${this.postId}" class="post-anchor"></div>
`, `,
)} )}
<div class="post-body-container"> ${when(
${when( this.sharedItems,
this.postTitle, () => html`<slot></slot>`,
() => html` () => html`
<div class="post-title-container flex w-full flex-row p-3"> <div class="post-body-container">
<a href="${this.permalink}" class="post-title-a prose" ${when(
><h3>${this.postTitle}</h3></a this.singlePostHeadline,
> () => html`
<div
class="post-headline-container flex w-full flex-row p-3"
>
<a
href="${ifDefined(this.permalink)}"
class="post-headline-a prose"
><h3>${this.singlePostHeadline}</h3></a
>
</div>
`,
)}
<div
class="post-body"
data-post-body="true"
data-testid="post-body"
>
<slot></slot>
</div> </div>
`, ${when(
)} this.tags && this.tags.length,
<div () => html`
class="post-body relative overflow-hidden supports-[overflow:clip]:overflow-clip isolate co-contain-paint" <div class="co-tags-container">
data-post-body="true" <div
data-testid="post-body" class="co-tags relative w-full overflow-y-hidden break-words leading-none "
> >
<div class="post-body-section prose"> <div>
<slot></slot> ${(this.tags || "")
.split(",")
.map(
(tag) => html`
<a href="https://cohost.org/tagged/${tag}"
>#${tag}</a
>
`,
)}
</div>
</div>
</div>
`,
)}
</div> </div>
</div> `,
</div> )}
</div> </div>
<hr class="co-hairline" /> <hr class="co-hairline" />
<footer class="co-thread-footer"> <footer class="co-thread-footer">
<div class="co-thread-footer-1"> <div class="co-thread-footer-1">
<div class="comments"> <div class="comments">
<a <a
href="${this.permalink}#comments" href="${ifDefined(this.permalink)}#comments"
class="text-sm hover:underline" class="text-sm hover:underline"
>${this.comments} comment${this.comments == 1 ? "" : "s"}</a >${this.comments} comment${this.comments == 1 ? "" : "s"}</a
> >
@ -458,8 +618,136 @@ export class CohostWebComponent extends LitElement {
} }
} }
declare global { @customElement("cohost-shared-item")
interface HTMLElementTagNameMap { export class CohostSharedItem extends LitElement {
"cohost-wc": CohostWebComponent; static override styles = css`
${COMMON_CSS}
.co-post-header {
display: flex;
flex-direction: flex-row;
flex-wrap: wrap;
align-items: center;
gap: 0.5rem;
padding-left: 0.75rem;
padding-right: 0.75rem;
padding-top: 0.5rem;
padding-bottom: 0.5rem;
background-color: light-dark(rgb(255 249 242), rgb(25 25 25));
}
`;
@property()
avatarSrc =
"";
@property()
avatarShape = "squircle";
@property()
displayName = "Display Name";
@property()
handle = "handle";
@property()
publishedAt: string | undefined;
@property()
permalink: string | undefined;
@property({ type: Boolean })
sharedItems: boolean = false;
@property()
postId: string | undefined;
@property()
headline: string | undefined;
@property()
tags: string | undefined;
@property({ type: Number })
comments = 0;
override render() {
let displayTimestamp = undefined;
let relativeTimestamp = undefined;
if (this.publishedAt) {
let date = new Date(this.publishedAt);
displayTimestamp = date.toLocaleString();
relativeTimestamp = formatRelativeTime(date);
}
return html`
${when(
this.postId,
() => html` <div id="post-${this.postId}" class="post-anchor"></div> `,
)}
<div class="co-post-header">
<div class="co-avatar-container mask">
<img
src="${this.avatarSrc}"
class="co-avatar mask mask-${this.avatarShape}"
alt="${this.displayName}"
/>
</div>
<a
rel="author"
href="https://cohost.org/${this.handle}"
class="co-project-display-name"
title="${this.displayName}"
data-url-popovers=""
style="user-select: none;"
data-longpress-event="true"
>${this.displayName}</a
><a
rel="author"
href="https://cohost.org/${this.handle}"
class="co-project-handle"
data-url-popovers=""
style="user-select: none;"
data-longpress-event="true"
>@${this.handle}</a
>${when(
this.publishedAt,
() =>
html`<time
datetime="${ifDefined(this.publishedAt)}"
title="${ifDefined(displayTimestamp)}"
><a href="${ifDefined(this.permalink)}"
>${relativeTimestamp}</a
></time
>`,
)}
</div>
<div class="post-body-container">
${when(
this.headline,
() => html`
<div class="post-headline-container flex w-full flex-row p-3">
<a
href="${ifDefined(this.permalink)}"
class="post-headline-a prose"
><h3>${this.headline}</h3></a
>
</div>
`,
)}
<div class="post-body" data-post-body="true" data-testid="post-body">
<div class="post-body-section">
<slot></slot>
</div>
</div>
</div>
<hr class="co-hairline" />
`;
}
}
declare global {
interface HTMLElementTagNameMap {
"cohost-post": CohostPost;
} }
} }