Skip to content
This repository has been archived by the owner on Jul 12, 2019. It is now read-only.

[WIP] Button #9

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@
"jest-circus": "^23.6.0",
"jest-junit": "^5.2.0",
"lint-staged": "^8.1.0",
"lodash": "^4.17.11",
"prettier": "^1.15.3",
"rimraf": "^2.6.3",
"rollup": "^1.1.0",
Expand Down Expand Up @@ -87,7 +88,8 @@
"testMatch": [
"<rootDir>/**/__tests__/**/*.js?(x)",
"<rootDir>/**/?(*.)(spec|test).js?(x)",
"<rootDir>/**/?(*-)(spec|test).js?(x)"
"<rootDir>/**/?(*-)(spec|test).js?(x)",
"<rootDir>/**/tests/unit.js?(x)"
],
"transform": {
"^.+\\.(js|jsx)$": "./tasks/jest/jsTransform.js",
Expand Down
100 changes: 100 additions & 0 deletions src/components/button/config/demo.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/

/**
* Configuration for the Button component
* @module button/demo
*/
import startCase from 'lodash/startCase';
import { generate } from './generate';

/**
* Creates a button template configuration for a variant
* @returns {globalTypedefs.fractalDemo} fractal demo object
*/
const variantButtons = function() {
const variants = {};

Object.keys(this.selectors.variants).forEach(variant => {
variants[variant] = {
name: variant,
label: startCase(variant),
context: generate.apply(this, [
{
variant,
},
]),
};
});

return variants;
};

/**
* Modifier: Anchor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Personally I don't find "anchor" to be a "modifier" as there is a lot that goes in to making a "button" a link. It is not on the same level as making a button "small".

That was something we worked out extensively to define the difference between the terms variant, modifier, state, and scenario.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Assuming being an "anchor" means we set different element and an additional aria role... playing along with Carbon React for a second... how would you classify this?

Can you share those definitions?

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • variant - larger styling/functionality differences. You can't mix variants on the same instance. Examples include:
    • primary, secondary, etc for buttons
    • required action, transactional, passive modals
  • modifier - small change that can be applied to any/multiple variants. However modifiers of the same "category" can not be applied to the same instance. Examples include:
    • Size: such as trying to add compact/comfy (small/large) classes to the same instance
    • row differentiators for tables (alternating colors vs borders)
    • direction of a tooltip in relation to it's trigger
  • state - change caused by the user interacting with an element/component
  • scenario - changes the HTML structure/content model (e.g. button with an icon, data table with a toolbar, tooltip with a heading, etc)

This terminology we use in the selectors object, config definitions, etc. For demo description specifics, we also have a test category which is used to describe a demo that is created specifically to test different setups/bugs and does not define a feature of a component.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This info should be documented somewhere.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I know... I thought I had it on our site somewhere but couldn't find it. 📝

* @returns {globalTypedefs.fractalDemo} fractal demo object
*/
const anchorButton = function() {
return {
name: 'anchor',
label: 'Anchor',
context: generate.apply(this, [
{
element: 'a',

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I feel like using an a tag isn't really valid for a button. I understand that often links look like buttons but semantically it's still a link and they have different requirements that affect their HTML structure. I'd recommend having links outside the button component but you're allowed to style a link as a button (see the button demo in link.config.js of PAL

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the PAL example. I'd like to keep this proof of concept focused as narrowly on the spec structure and functionality as possible, and less on the differences between Carbon implementations. That effort begins after the proof of concept. That's why I included this in the proof of concept scope:

Carbon v10 React component visuals and behavior will be used in the proof of concept as standardizing component implementations is not in scope

Carbon React button component can render as either button or a, so this proof of concept does as well, so we can focus on the bigger issues at hand.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We obviously don't want to change what React does at this point...but if our goal is to focus on button, then that should keep the focus on what will eventually be the <button HTML element - of which a link (<a) would be out-of-scope as it's a totally different element with different a11y requirements. The React Button could still be tested with this code, we just wouldn't test against the anchor section - as there is not a compliant scenario where a <button would be an anchor element.

Testing only the button aspects of the React component would be in scope as we should expect any of the frameworks to have additional code and functionality in any given component - and our goal with the spec is to say "do you create these specific UI pieces" rather than "do you only create these specific UI pieces"

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Changing carbon-components-x -y and -z is not in scope for this proof of concept. -x, -y, and -z all implement button and link components differently and that's okay (for now). If we focus our efforts on proving the component specification concept instead of "what is a perfect button", then we'll achieve the desired outcomes in #5.

By taking a stable starting point (to prevent discussion like "what is a perfect button"), and if we can hit enough edge cases by going through these four components, we can achieve what we're after and then after that, spend the time locking down each component spec.

With that in mind, I'd argue this is a good example of when a component would need to support multiple element types.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So, we're not trying to create a perfect button by pushing back on anchor, its more about keeping button within it's basic scope of being a button as opposed to the React implementation's expanded use cases for a React component called Button. I do see that #5 includes Carbon v10 React component visuals and behavior will be used in the proof of concept as standardizing component implementations is not in scope so I can see where you're coming from, but I believe that misses out on a very important aspect of working to sync up across frameworks: User Expectations.

If we were working from a user's standpoint, the typical user goes to carbondesignsystem.com for button, they'll see nothing about an anchor element.

With that in mind, I think we may be in disagreement as to the point of the POC. What we thought we were figuring out:

  1. how can we write tests to confirm all frameworks are in sync
  2. how can we come up with a common way to write requirements so our users can understand why choices were made about a component and our framework developers can understand what outcomes are expected
  3. how can components share a common api?

If we were working from requirements, rather than using implemented code as a blueprint, this debate could easily be "what is expected according to the requirements?" Where we're coming from is that without considering a component's requirements we're going to be doomed to loops where we're essentially dealing with point-of-view and opinion rather than outcomes via reasoned deliberation.

All that being said, I totally understand the need to work on various aspects of what code we're trying to put in place, and it makes sense to have a focused PR that doesn't include requirements functionality. But in this case, there is a clear issue with this specific demo. A demo that would be very easy to leave it out of this POC. We can tackle the need to support variable elements in some of the more complicated components, but the point of starting with button is indeed it's lack of complication.

},
]),
};
};

/**
* Modifier: Small
* @returns {globalTypedefs.fractalDemo} fractal demo object
*/
const smallButton = function() {
return {
name: 'small',
label: 'Small',
context: generate.apply(this, [

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • putting the context which is used by the generate function within the generate.apply call means that that object is not available for use by the frameworks, all they would receive is the post-generate object. We put this in a separate preGenerate variable, which allows both use of the smaller config object directly...as well as giving us access to the preGenerated object in documentation
  • why is this within an array?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. Have you found use for that pre-generate object outside of documentation? I'll update.
  2. Second parameter of function.apply() is of type array.

{
size: 'small',
},
]),
};
};

/**
* State: Disabled
* @returns {globalTypedefs.fractalDemo} fractal demo object
*/
const disabledButton = function() {

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why make these functions when they don't take any parameters so it's a consistent return? Seems odd to me to define these as functions versus a constant variable and use unnecessary memory

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I did this to bind this.selectors used in the generate functions. this.selectors gets set when you instantiate the config with a prefix.

I'm open to other ways to handle prefixing to avoid binding and applying functions. Suggestions?

return {
name: 'disabled',
label: 'Disabled',
context: generate.apply(this, [
{
disabled: true,
},
]),
};
};

export default function demo() {
return {
variants: variantButtons.apply(this),
modifiers: {
types: {
anchor: anchorButton.apply(this),
},
sizes: {
small: smallButton.apply(this),
},
},
states: {
disabled: disabledButton.apply(this),
},
};
}
83 changes: 83 additions & 0 deletions src/components/button/config/generate.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/

/**
* Template configuration generate functions for Button component
mattrosno marked this conversation as resolved.
Show resolved Hide resolved
* @module button/generate
*/
import cloneDeep from 'lodash/cloneDeep';
import merge from 'lodash/merge';

// TODO JSDoc
export function generate(options) {

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I'm confused by the generate function being in the spec. My thoughts were that an individual repo would take care of this level of integration. i.e. the simpler api {variant: 'primary'} would be what is shared by c-c-X, and each repo would be in charge of converting that to the proper template configuration object for their implementation.

open to new thinking though

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it depends on the two options stated here?

This function here isn't meant to be a full template configuration object - although maybe it's close enough to act as that for Handlebars? The goal of the configuration object would be to give each component API just enough for it to render an accessible component.

let opts = merge(

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • it looks like opts is the default context a button component would receive - it would be handy to have this be an object outside of generate so it could be exported and used in other places.
  • we switched to using mergeWith due to merging issues we ran into with arrays. merge will munge arrays together with unintended consequences - so merge([1, null, 2, 3], [1, 5, 6]) becomes [1, 5, 6, 3] or similar weirdness

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where else would the component default context be useful? I guess in the React world, defining defaultProps, so you could get those without generating a default component. But then again, what's wrong with calling an empty generate function to get the default component config? E.g.

const defaultButton = config.generate();

Button.defaultProps = {
  iconDescription: 'Provide icon description if icon is used',
  tabIndex: defaultButton.tabIndex,
  type: defaultButton.type,
  disabled: defaultButton.disabled,
  small: defaultButton.size === 'small',
  kind: defaultButton.variant,
};

Maybe that was with an older version of merge? Looks like no difference if not using a customizer. https://codepen.io/mattrosno/pen/MLbPwM?editors=0011

{
element: 'button',

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what is the use-case for this variable please?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Following Carbon React's lead for this POC.

content: 'Button',
mattrosno marked this conversation as resolved.
Show resolved Hide resolved
attributes: {},
mattrosno marked this conversation as resolved.
Show resolved Hide resolved
disabled: false,
mattrosno marked this conversation as resolved.
Show resolved Hide resolved
type: 'button',
mattrosno marked this conversation as resolved.
Show resolved Hide resolved
tabIndex: 0,

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why the redundant tabIndex?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Carbon React sets tabIndex, so this proof of concept does as well.

variant: 'primary',
},
options
);

const selectors = cloneDeep(this.selectors);

let config = {
root: {},
};

Object.keys(selectors.default).forEach(element => {
config = merge(config, {
[element]: {

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This would force Icon to always be present but it's actually an optional element.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I figured that'd be up to carbon-components-x to decide. If they know to put an icon in the template, then can use this from the component config. If not, ignore and not use it.

"icon": {
  "attributes": {
    "aria-hidden": true
  },
  "classNames": "bx--btn__icon"
 }

I think it depends if the button config is responsible for its child icon config, or if carbon-components-x can call button config (knowing there's an icon) and icon config do its thing.

Doing the icon component next will help here?

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the icon component will help this debate. This seems like code that belongs in Icon

attributes: {},

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what if the user wants to define custom attributes on an element?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good catch! I'll fix.

classNames: selectors.default[element],
},
});
});

// Element

config.root.element = opts.element;

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

is there a non-anchor use-case for this variable?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just for the anchor.


// Content

config.root.content = opts.content;

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the classes object and the outputted object don't align

{
   root: ...
   content: ...
   icon: ...
}
{
   root: {
    classNames: ...
    content: ...
    attributes: ...
  },
  content: {
    classNames: ....
  },
  icon: {
     classNames: ...
     attributes: ...
  }
}

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

content is overloaded in this example, because of the bx--btn__content element? I know I broke my rule here of following Carbon React's lead since they don't have that inner span. I called that inner span bx--btn__content to show how content can be a node's name in selectors object, and still be used alongside element, attributes, className for text.

Does that help? Or is something still misaligned?


// Classes

config.root.classNames += ` ${selectors.variants[opts.variant].root}`;

if (opts.size) {
config.root.classNames += ` ${selectors.modifiers.sizes[opts.size].root}`;
}

// Attributes

if (opts.variant === 'danger') {
config.root.attributes['aria-label'] = 'danger';

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

when I was testing this with a screen reader it did not read the content of the button...

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks like I did this because of Carbon Vanilla.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll remove this aria-label to follow Carbon React's lead.

}

if (opts.element === 'button') {
config.root.attributes.type = opts.type;

if (opts.disabled) {
config.root.attributes.disabled = '';
}
} else if (opts.element === 'a') {
config.root.attributes.href = opts.href || '#';
config.root.attributes.role = 'button';

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

a button and a link have two different functions why are they being mixed here?
What about disabled anchors? (I feel like links should NOT be part of this component)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

gotta agree with Liz - the anchor should not be in this component. The functionality to create an anchor within button is specific to the React component. It looks like anchor was originally added to this component for styling purposes, which, if that's the case, the styling of an anchor to look like a button should happen within the anchor tag's component, applying button styles with classes.

I think our goal in pointing this out is that it really doesn't belong in the spec for a button, and by keeping it in here from the start, we're adding technical debt from the start. This would not be a use of button which we would be able to support due to compliance and a11y issues, so when running the spec against the handlebars button, we'd start off as failing the spec.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A follow-up to this comment, because all Carbon implementations aren't in lock-step, why this spec is coming to be, I expect every implementation to start off with failing tests to some degree.

Do you think we need to increase the scope of this proof of concept so that every Carbon implementation passes the spec for the four components? I don't think that's necessary as long as everyone can see a clear path on how to achieve that.

Should be minimal technical debt if we're not modifying every carbon-components-x to pass full spec tests for our 4 components.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should focus only on button for now, and yeah, we should increase scope to work with all four frameworks to determine that the code we create works in all four. We don't want to go deep into building stuff out for four components only to find out it doesn't work with one or more frameworks.
The goal is to make sure all four frameworks can at least get the tests running, but not necessarily that they all need to pass the tests.

Getting this all in place via button only will allow us to iterate on a known-to-be-working-cross-framework base when building out the other 3 components - which will limit how much tech debt we accrue.

}

if (config.icon) {
config.icon.attributes['aria-hidden'] = true;
}

return config;
}
67 changes: 67 additions & 0 deletions src/components/button/config/selectors.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/

/**
* Available CSS selectors for Button component
* @module button/selectors
*/

/**
* Button component classes
* @typedef {object} buttonClasses
* @property {string} root - button element
* @property {string} content - wrapper element around button content
* @property {string} icon - button SVG
*/

/**
* Button component selectors
* @typedef {object} buttonSelectors
* @property {buttonClasses} default - default button classes
* @property {object} variants - button variants
* @property {buttonClasses} variants.primary - `primary` variant
* @property {buttonClasses} variants.secondary - `secondary` variant
* @property {buttonClasses} variants.tertiary - `tertiary` variant
* @property {buttonClasses} variants.ghost - `ghost` variant
* @property {buttonClasses} variants.danger - `danger` variant
* @property {object} modifiers - style modifiers
* @property {object} modifiers.sizes - size modifiers
* @property {buttonClasses} modifiers.sizes.small - `small` modifier
*/
const selectors = {
default: {
root: 'btn',
content: 'btn__content',
icon: 'btn__icon',
},
variants: {
primary: {
root: 'btn--primary',
},
secondary: {
root: 'btn--secondary',
},
tertiary: {
root: 'btn--tertiary',
},
ghost: {
root: 'btn--ghost',
},
danger: {
root: 'btn--danger',
},
},
modifiers: {
sizes: {
small: {
root: 'btn--sm',
},
},
},
};

export default selectors;
Empty file.
Empty file.
Empty file.
35 changes: 35 additions & 0 deletions src/components/button/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/

/**
* Button component
* @module button
*/
import { prefixSelectors } from '../../tools/component/selectors';
import demo from './config/demo';
import { generate } from './config/generate';
import selectors from './config/selectors';

/**
* Button spec
* @param {string} prefix - selector prefix
* @type {globalTypedefs.componentConfig}
*/
const buttonConfig = prefix => {
const config = {
selectors: prefixSelectors(selectors, prefix),
};

return {
label: 'Button',
demo: demo.bind(config),
generate: generate.bind(config),
selectors: config.selectors,
};
};

export default buttonConfig;
6 changes: 6 additions & 0 deletions src/components/button/index.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
//
// Copyright IBM Corp. 2018, 2018
//
// This source code is licensed under the Apache-2.0 license found in the
// LICENSE file in the root directory of this source tree.
//
6 changes: 6 additions & 0 deletions src/components/button/requirements.feature
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
#
# Copyright IBM Corp. 2018, 2018
#
# This source code is licensed under the Apache-2.0 license found in the
# LICENSE file in the root directory of this source tree.
#
6 changes: 6 additions & 0 deletions src/components/button/styles/_mixins.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
//
// Copyright IBM Corp. 2018, 2018
//
// This source code is licensed under the Apache-2.0 license found in the
// LICENSE file in the root directory of this source tree.
//
6 changes: 6 additions & 0 deletions src/components/button/styles/_variables.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
//
// Copyright IBM Corp. 2018, 2018
//
// This source code is licensed under the Apache-2.0 license found in the
// LICENSE file in the root directory of this source tree.
//
6 changes: 6 additions & 0 deletions src/components/button/tests/context.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/
6 changes: 6 additions & 0 deletions src/components/button/tests/requirements.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/
6 changes: 6 additions & 0 deletions src/components/button/tests/template.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/
29 changes: 29 additions & 0 deletions src/components/button/tests/unit.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
/**
* Copyright IBM Corp. 2018, 2019
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/

import buttonConfig from '../';

describe('Button', () => {
it('Generates a button', () => {
const button = buttonConfig('bx');
const myButton = button.generate({
disabled: true,
size: 'small',
variant: 'danger',
});

console.log(JSON.stringify(button.label, null, 2));

console.log(JSON.stringify(button.demo(), null, 2));

console.log(JSON.stringify(myButton, null, 2));

console.log(JSON.stringify(button.selectors, null, 2));

expect(true).toEqual(true);
});
});
Loading