Skip to content

Commit

Permalink
Convert Excerpt component to Tailwind
Browse files Browse the repository at this point in the history
  • Loading branch information
lyzadanger committed Mar 24, 2022
1 parent 2e26f16 commit a95447b
Show file tree
Hide file tree
Showing 5 changed files with 70 additions and 94 deletions.
75 changes: 58 additions & 17 deletions src/sidebar/components/Excerpt.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,20 +20,32 @@ import { applyTheme } from '../helpers/theme';
* @param {InlineControlsProps} props
*/
function InlineControls({ isCollapsed, setCollapsed, linkStyle = {} }) {
const toggleLabel = isCollapsed ? 'More' : 'Less';

return (
<div className="Excerpt__inline-controls">
<div className="Excerpt__toggle-container">
<div
className={classnames(
// Position these controls at the bottom right of the excerpt
'absolute block right-0 bottom-0',
// Give extra width for larger tap target and gradient fade
// Fade transparent-to-white left-to-right to make the toggle
// control text (More/Less) more readable above other text.
// This gradient is implemented to-left to take advantage of Tailwind's
// automatic to-transparent calculation: this avoids Safari's problem
// with transparents in gradients:
// https://bugs.webkit.org/show_bug.cgi?id=150940
// https://tailwindcss.com/docs/gradient-color-stops#fading-to-transparent
'w-20 bg-gradient-to-l from-white'
)}
>
<div className="flex justify-end">
<LinkButton
classes="InlineLinkButton InlineLinkButton--underlined"
classes="inline underline"
onClick={() => setCollapsed(!isCollapsed)}
expanded={!isCollapsed}
title="Toggle visibility of full excerpt text"
style={linkStyle}
variant="dark"
>
{toggleLabel}
{isCollapsed ? 'More' : 'Less'}
</LinkButton>
</div>
</div>
Expand Down Expand Up @@ -86,14 +98,13 @@ function Excerpt({
const [collapsedByInlineControls, setCollapsedByInlineControls] =
useState(true);

// Container for the excerpt's content.
const contentElement = /** @type {{ current: HTMLDivElement }} */ (useRef());

// Measured height of `contentElement` in pixels.
// Measured height of `contentElement` in pixels
const [contentHeight, setContentHeight] = useState(0);

// Update the measured height of the content after the initial render and
// when the size of the content element changes.
// Update the measured height of the content container after initial render,
// and when the size of the content element changes.
const updateContentHeight = useCallback(() => {
const newContentHeight = contentElement.current.clientHeight;
setContentHeight(newContentHeight);
Expand Down Expand Up @@ -133,18 +144,48 @@ function Excerpt({
: onToggleCollapsed(collapsed);

return (
<div className="Excerpt" style={contentStyle}>
<div className="Excerpt__content" ref={contentElement}>
<div
data-testid="excerpt-container"
className={classnames(
'relative overflow-hidden',
'transition-[max-height] ease-in duration-150'
)}
style={contentStyle}
>
<div
className={classnames(
// Establish new block-formatting context to prevent margin-collapsing
// in descendent elements from potentially "leaking out" and pushing
// this element down from the top of the container.
// See https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Block_formatting_context
// See https://github.com/hypothesis/client/issues/1518
'inline-block w-full'
)}
data-testid="excerpt-content"
ref={contentElement}
>
{children}
</div>
<div
data-testid="excerpt-expand"
role="presentation"
onClick={() => setCollapsed(false)}
className={classnames({
Excerpt__shadow: true,
'Excerpt__shadow--transparent': inlineControls,
'is-hidden': !isExpandable,
})}
className={classnames(
// This element provides a clickable area at the bottom of an
// expandable excerpt to expand it.
'transition-[opacity] duration-150 ease-linear',
'absolute w-full bottom-0 h-touch-minimum',
{
// For expandable excerpts not using inlineControls, style this
// element with a custom shadow-like gradient
'bg-gradient-to-b from-excerptStop1 via-excerptStop2 to-excerptStop3':
!inlineControls && isExpandable,
'bg-none': inlineControls,
// Don't make this shadow visible OR clickable if there's nothing
// to do here (the excerpt isn't expandable)
'opacity-0 pointer-events-none': !isExpandable,
}
)}
title="Show the full excerpt"
/>
{isOverflowing && inlineControls && (
Expand Down
8 changes: 5 additions & 3 deletions src/sidebar/components/test/Excerpt-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -50,12 +50,14 @@ describe('Excerpt', () => {
});

function getExcerptHeight(wrapper) {
return wrapper.find('.Excerpt').prop('style')['max-height'];
return wrapper.find('[data-testid="excerpt-container"]').prop('style')[
'max-height'
];
}

it('renders content in container', () => {
const wrapper = createExcerpt();
const contentEl = wrapper.find('.Excerpt__content');
const contentEl = wrapper.find('[data-testid="excerpt-content"]');
assert.include(contentEl.html(), 'default content');
});

Expand Down Expand Up @@ -109,7 +111,7 @@ describe('Excerpt', () => {
it('calls `onToggleCollapsed` when user clicks in bottom area to expand excerpt', () => {
const onToggleCollapsed = sinon.stub();
const wrapper = createExcerpt({ onToggleCollapsed }, TALL_DIV);
const control = wrapper.find('.Excerpt__shadow');
const control = wrapper.find('[data-testid="excerpt-expand"]');
assert.equal(getExcerptHeight(wrapper), 40);
control.simulate('click');
assert.called(onToggleCollapsed);
Expand Down
73 changes: 0 additions & 73 deletions src/styles/sidebar/components/Excerpt.scss

This file was deleted.

1 change: 0 additions & 1 deletion src/styles/sidebar/components/_index.scss
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@
// ----------
@use './AnnotationShareControl';
@use './AutocompleteList';
@use './Excerpt';
@use './FilterSelect';
@use './FilterStatus';
@use './GroupList';
Expand Down
7 changes: 7 additions & 0 deletions tailwind.config.mjs
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,13 @@ export default {
'px-tiny': ['10px'],
'px-base': ['16px'],
},
gradientColorStops: {
// These gradient stops define a custom gradient shown at the bottom of
// long annotation body excerpts.
excerptStop1: 'rgba(255, 255, 255, 0) 50%',
excerptStop2: 'rgba(0, 0, 0, 0.08) 95%',
excerptStop3: 'rgba(0, 0, 0, 0.13) 100%',
},
keyframes: {
adderPopDown: {
'0%': {
Expand Down

0 comments on commit a95447b

Please sign in to comment.