diff --git a/docs/pages/qa/cards.vue b/docs/pages/qa/cards.vue index 53cd1706d..f1cb46683 100644 --- a/docs/pages/qa/cards.vue +++ b/docs/pages/qa/cards.vue @@ -168,14 +168,14 @@ :headingLevel="3" orientation="horizontal" :thumbnailSrc="require('../../assets/hummingbird-large-cc-by-sa-4.jpg')" - title="First card when link" + title="First card" /> @@ -187,13 +187,13 @@ :headingLevel="3" orientation="horizontal" :thumbnailSrc="require('../../assets/hummingbird-large-cc-by-sa-4.jpg')" - title="First card when not link" + title="First card" /> diff --git a/lib/cards/KCard.vue b/lib/cards/KCard.vue index fee4137ec..a3f9581d4 100644 --- a/lib/cards/KCard.vue +++ b/lib/cards/KCard.vue @@ -17,7 +17,9 @@ :class="['card-area', ...cardAreaClasses ]" :style="{ backgroundColor: $themeTokens.surface }" > - - - {{ title }} - - - - - - - - - - -
- - - + + + + +
+ + +
+ +
+ + +
+
+ +
- - - -
-
- - -
-
- - + + + + + + +
+
in KImg with z-index 1 should cover the placeholder after loaded */ - display: flex; - align-items: center; - justify-content: center; - height: 100%; - } - .title { display: inline-block; // allows title placeholder in the skeleton card width: 100%; // allows title placeholder in the skeleton card @@ -634,57 +603,94 @@ outline: none; // the focus ring is moved to the whole
  • } - /************* Manage spaces *************/ + // Because the title and the areas above and below it + // are grouped together in all layoyts, abstract them + // into one whole here. Simplifies common spacing + // styles as well as layout-specific styles. + .around-title { + display: flex; + flex-direction: column; + + .heading { + order: 2; + } + + .above-title { + order: 1; + } + + .below-title { + order: 3; + } + } + + /************* Spacing *************/ + // first reset + .around-title, .heading, .above-title, .below-title, .footer { padding: 0; - margin-top: 0; - margin-right: $spacer; - margin-bottom: calc(#{$spacer} / 2); - margin-left: $spacer; + margin: 0; } - // when the 'aboveTitle' area is present, - // apply top margin to it and also set smaller - // margin between the area and the heading... - .with-above-title { - .above-title { - margin-top: $spacer; - margin-bottom: calc(#{$spacer} / 2); - } + /* stylelint-disable no-duplicate-selectors */ + .around-title { + padding: $spacer; } + /* stylelint-enable no-duplicate-selectors */ - // ...and when the 'aboveTitle' area is not present, - // instead apply the top margin to the heading - .without-above-title { - .heading { - margin-top: $spacer; - } + .footer { + padding: 0 $spacer $spacer $spacer; } - // if there's the 'belowTitle' area present, - // override the heading's margin to smaller one - .with-below-title { - .heading { - margin-bottom: calc(#{$spacer} / 2); - } + .above-title { + padding-bottom: calc(#{$spacer} / 2); } + .below-title { + padding-top: calc(#{$spacer} / 2); + } + + /************* Thumbnail **************/ + /* stylelint-disable no-duplicate-selectors */ - .footer { - margin-top: auto; // push footer to the bottom + .thumbnail { + position: relative; /* for absolute positioning of .thumbnail-placeholder */ } /* stylelint-enable no-duplicate-selectors */ + .thumbnail-placeholder { + position: absolute; + top: 0; + right: 0; + bottom: 0; + left: 0; + z-index: 0; /* in KImg with z-index 1 should cover the placeholder after loaded */ + display: flex; + align-items: center; + justify-content: center; + height: 100%; + } + /************* Layout-specific styles *************/ .vertical-with-large-thumbnail { + .upper-card-area { + display: flex; + flex-direction: column; + } + .thumbnail { + order: 1; height: 180px; } + + .around-title { + order: 2; + } } .vertical-with-small-thumbnail { @@ -698,104 +704,92 @@ } } - .horizontal-with-large-thumbnail { - $thumbnail-width: 40%; + .horizontal-with-small-thumbnail { + .upper-card-area { + display: flex; + flex-direction: row; + align-items: flex-start; + } .thumbnail { - position: absolute; - width: $thumbnail-width; - height: 100%; + width: 30%; + margin-top: $spacer; + margin-bottom: $spacer; } - .heading, - .above-title, - .below-title, - .footer { - width: calc(100% - #{$thumbnail-width} - 2 * #{$spacer}); + .around-title { + width: 70%; } &.thumbnail-align-left { - align-items: flex-end; - .thumbnail { - left: 0; + order: 1; + margin-left: $spacer; + } + + .around-title { + order: 2; } } &.thumbnail-align-right { - align-items: flex-start; - .thumbnail { - right: 0; + order: 2; + margin-right: $spacer; + } + + .around-title { + order: 1; } } } - .horizontal-with-small-thumbnail { - $thumbnail-width: null; - - /* - Coordinates space taken by the thumbnail area and the content area - next to it more intelligently in browsers that support `clamp()` by: - - - Instead of defining 'width', 'min-width', and 'max-width' separately, - `clamp()` is used with the goal to have the actual thumbnail width - saved in the single `$thumbnail-width` value. - - - The `$thumbnail-width` value is then referenced when calculating - the remaining space for the content area, ensuring the precise - distribution of space. - - Resolves some issues related to unprecise calculations, most importantly - this removes the area of empty space between the thumbnail and content areas - in some card's sizes, wasting space that can be used for card's textual content. - */ - @mixin clamp-with-fallback($min, $preferred, $max) { - // fallback for browsers that don't support 'clamp()' - $thumbnail-width: $preferred; - - width: $thumbnail-width; - min-width: $min; - max-width: $max; + .horizontal-with-large-thumbnail { + // override few styles from .horizontal-with-small-thumbnail + // to stretch the thumbnail to the full height of the card - // clamp(1px, 1%, 1px) only used for testing support - @supports (width: clamp(1px, 1%, 1px)) { - $thumbnail-width: clamp(#{$min}, #{$preferred}, #{$max}); + /* stylelint-disable scss/at-extend-no-missing-placeholder */ + @extend .horizontal-with-small-thumbnail; + /* stylelint-enable scss/at-extend-no-missing-placeholder */ - width: $thumbnail-width; - } + &.card-area { + position: relative; /* for absolute positioning of .thumbnail */ } .thumbnail { - @include clamp-with-fallback(100px, 35%, 120px); - position: absolute; - top: $spacer; - } - - .heading, - .above-title, - .below-title { - width: calc(100% - #{$thumbnail-width} - 3 * #{$spacer}); + width: 40%; + height: 100%; + margin-top: 0; + margin-bottom: 0; } + .around-title, .footer { - width: calc(100% - 2 * #{$spacer}); + width: 60%; } &.thumbnail-align-left { - align-items: flex-end; - .thumbnail { - left: $spacer; + left: 0; + margin-left: 0; + } + + .around-title, + .footer { + margin-left: auto; } } &.thumbnail-align-right { - align-items: flex-start; - .thumbnail { - right: $spacer; + right: 0; + margin-right: 0; + } + + .around-title, + .footer { + margin-right: auto; } } } diff --git a/lib/cards/__tests__/KCard.spec.js b/lib/cards/__tests__/KCard.spec.js index 978fd9a11..a7041dddf 100644 --- a/lib/cards/__tests__/KCard.spec.js +++ b/lib/cards/__tests__/KCard.spec.js @@ -30,41 +30,31 @@ describe('KCard', () => { // Check for: // - // li - // |-- div - // |--span (visually hidden title text) - // |-- h[2-6] - // |--a (with aria-labelledby pointing to the span above) - // |-- span (with aria-hidden) + //
  • + // |-- with visually hidden title text + // |-- + // |-- with aria-labelledby pointing to the span with the visually hidden title text + // |-- with aria-hidden // const el1 = wrapper.find(':first-child'); expect(el1.exists()).toBe(true); expect(el1.element.tagName.toLowerCase()).toBe('li'); - const el2 = wrapper.find('li > :first-child'); + const el2 = wrapper.find('li .visuallyhidden'); expect(el2.exists()).toBe(true); - expect(el2.element.tagName.toLowerCase()).toBe('div'); + expect(el2.text()).toBe('Test Title'); + const spanId = el2.attributes('id'); - const el3 = wrapper.find('li > div > :nth-child(1)'); + const el3 = wrapper.find('li h4'); expect(el3.exists()).toBe(true); - expect(el3.element.tagName.toLowerCase()).toBe('span'); - expect(el3.classes()).toContain('visuallyhidden'); - expect(el3.text()).toBe('Test Title'); - const spanId = el3.attributes('id'); - const el4 = wrapper.find('li > div > :nth-child(2)'); + const el4 = wrapper.find(`li h4 [aria-labelledby="${spanId}"]`); expect(el4.exists()).toBe(true); - expect(el4.element.tagName.toLowerCase()).toBe('h4'); + expect(el4.element.tagName.toLowerCase()).toBe('a'); - const el5 = wrapper.find(`li > div > h4 > :first-child`); + const el5 = wrapper.find(`li h4 [aria-labelledby="${spanId}"] > :first-child`); expect(el5.exists()).toBe(true); - expect(el5.element.tagName.toLowerCase()).toBe('a'); - expect(el5.attributes('aria-labelledby')).toBe(spanId); - - const el6 = wrapper.find(`li > div > h4 > a > :first-child`); - expect(el6.exists()).toBe(true); - expect(el6.element.tagName.toLowerCase()).toBe('span'); - expect(el6.attributes('aria-hidden')).toBe('true'); + expect(el5.attributes('aria-hidden')).toBe('true'); }); it(`renders the correct accessibility structure when card is not link`, () => { @@ -74,42 +64,31 @@ describe('KCard', () => { // Check for: // - // li - // |-- div - // |--span (visually hidden title text) - // |-- h[2-6] - // |--span (focusable and with aria-labelledby pointing to the span above) - // |-- span (with aria-hidden) + //
  • + // |-- with visually hidden title text + // |-- + // |-- with aria-labelledby pointing to the span with the visually hidden title text + // |-- with aria-hidden // const el1 = wrapper.find(':first-child'); expect(el1.exists()).toBe(true); expect(el1.element.tagName.toLowerCase()).toBe('li'); - const el2 = wrapper.find('li > :first-child'); + const el2 = wrapper.find('li .visuallyhidden'); expect(el2.exists()).toBe(true); - expect(el2.element.tagName.toLowerCase()).toBe('div'); + expect(el2.text()).toBe('Test Title'); + const spanId = el2.attributes('id'); - const el3 = wrapper.find('li > div > :nth-child(1)'); + const el3 = wrapper.find('li h4'); expect(el3.exists()).toBe(true); - expect(el3.element.tagName.toLowerCase()).toBe('span'); - expect(el3.classes()).toContain('visuallyhidden'); - expect(el3.text()).toBe('Test Title'); - const spanId = el3.attributes('id'); - const el4 = wrapper.find('li > div > :nth-child(2)'); + const el4 = wrapper.find(`li h4 [aria-labelledby="${spanId}"]`); expect(el4.exists()).toBe(true); - expect(el4.element.tagName.toLowerCase()).toBe('h4'); + expect(el4.element.tagName.toLowerCase()).toBe('span'); - const el5 = wrapper.find(`li > div > h4 > :first-child`); + const el5 = wrapper.find(`li h4 [aria-labelledby="${spanId}"] > :first-child`); expect(el5.exists()).toBe(true); - expect(el5.element.tagName.toLowerCase()).toBe('span'); - expect(el5.attributes('aria-labelledby')).toBe(spanId); - expect(el5.attributes('tabindex')).toBe('0'); - - const el6 = wrapper.find(`li > div > h4 > span > :first-child`); - expect(el6.exists()).toBe(true); - expect(el6.element.tagName.toLowerCase()).toBe('span'); - expect(el6.attributes('aria-hidden')).toBe('true'); + expect(el5.attributes('aria-hidden')).toBe('true'); }); describe('on long click', () => {