Over time we have curated a growing checklist of things we feel improve the experience of using a custom UIKit component. Many of these checks are performed by humans but we're now increasing the number of checks that can be performed by scripts.
Before a component is built, the API proposed must be agreed upon by the main contributors.
- Create a pull request with only the .h files of the proposed component linked back to the original issue for the component's creation.
- Enter the pull request # or NO
Every component has a README.md file describing what it is, what it does, when to use it, etc, in the root of the component's folder. To create a new README.md file see the template at writing_readmes.
- Verify the component has a filled out README.md
- Verify Swift code samples appear before Objective-C samples.
- Enter YES or NO
Sometimes, the inline comments and README.md will not be sufficient to describe usage of the component. In these cases, create additional README.md files in child folders of the component. See: Collections for a good example.
- Verify the component has filled out README.md files in its child folders.
- Enter YES, NO or N/A
Each component must have a short video captured from either iPhone or iPhone simulator of it in action.
- Verify the component's
.../docs/assets
folder contains a video namedcomponent_name.mp4
. - Enter YES or NO
Each component must also have a still image to use when video cannot play.
- Verify the component's
.../docs/assets
folder contains a still namedcomponent_name.png
. - Enter YES or NO
The included catalog application uses Core Graphics to draw landing page tiles for each component. These tiles are created by Google's Material Design department specifically for this purpose and then converted to Core Graphics code via PaintCode.
- Set the canvas size to 82 x 82.
- Import the file (.svg or .ai) into PaintCode.
- Massage values until it matches the original (colors, gradients, spacing, etc).
- Make sure all shapes are enclosed in a group named Component Name Group.
- Set the group's origin to (1,1) and size to 80 X 80
- Enclose the group in a frame with the same bounds and origin as the canvas, not the group.
- Set the group's springs and struts to:
- Top and Left pinned
- All others resize
- In
catalog/MDCCCatalog/MDCCatalogTiles.h
, declare a function for the new component. - In
catalog/MDCCCatalog/MDCCatalogTiles.m
, add that function (empty.) - Copy and paste the generated iOS Objc code into the function.
- Replace color variable values with values taken from the passed MDCColorScheme if possible.
- In
catalog/MDCCatalog/MDCCatalogTileView.swift
, add a new case for the new component and have it createnewImage
from the new function.
- Run the catalog application and look for the component. Make sure the tile shown is specific to the component and not a placeholder nor empty view.
- Enter YES or NO
The material.io site contains a list of all components, each with an icon. These icons are created by Google's Material Design department specifically for this purpose. Adding it to the site is done by the core team.
- Enter YES or NO or N/A
The material.io site's content is generated from the headers of MDC's files using Jazzy and Jekyll. This requires some YAML.
- The root README.md of each component should start with a comment containing specific key-value pairs. Here's an example:
<!--docs:
title: "App Bars"
layout: detail
section: components
excerpt: "The App Bar is a flexible navigation bar designed to provide a typical Material Design navigation experience."
iconId: toolbar
path: /catalog/app-bars/
api_doc_root: true
-->
The root of each component should also contain a .jazzy.yaml
file that's auto-generated during the publishing process.
- Enter YES or NO or N/A
Every source file must have the Apache 2.0 license stanza at the top of the file. The copyright should be assigned to “the Material Components for iOS authors“. You can do this manually or use a tool such as autogen to add a license.
Sample:
/*
Copyright 2016-present the Material Components for iOS authors. All Rights Reserved.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
- Add the license to every source and script file in your component.
- Enter YES or NO
Unit tests in MDC are run by the developer, the continuous integration service, and the release engineer: the developer runs them regularly during development, the CI service when a pull request is submitted, and the release engineer as part of the release process prior to final merging. If a developer submits a PR with broken unit tests, the CI service will prevent merging thru GitHub.
- Store unit test files for a component in the directory
components/ComponentName/tests/unit/
- Unit tests may be written in either Objc or Swift. Swift is preferred as merely writing them can verify the Objc-to-Swift experience is seamless.
- //TODO: helpful ideas for tests
- Include unit tests for new functionality.
- Inlcude unit tests for bug fixes and changes initiated from GitHub issues. Name them
ClassNameIssue
+ issue number +Tests
. e.g.AppBarContainerIssue246Tests
,FlexibleHeaderControllerIssue176Tests
. - Update existing unit tests for new changes.
- Ensure unit tests run with no errors.
- Enter YES or NO
Visual components should have interaction tests built with Earl Grey.
//TODO and explain how
MDC-iOS uses Google's continuous integration service Kokoro for automated tests on each PR. Kokoro builds MDC with Google's open source build system, Bazel. For more information, see the Kokoro & Bazel document.
- Add a Bazel
BUILD
file to the root directory of the component. - Add
BUILD
files for any components that are dependencies of the component. (If necessary.) - Test locally with the .kokoro script in the repo's root directory. If necessary, propose adjustments to the script to support custom features of your target (like a private dependency.)
- Enter YES or NO
Google strives to support as many different written languages as possible in components containing static text. The necessary translations must be written by Google's internal translators. To request translations, open an issue with all text and all requested languages.
- Any strings that are added must be internationalized. Please don’t include English strings in code. We have language bundles for the various components. Add your English strings there and use a key to access it via code (key will be generated after compilation).
- Make sure strings are internationalized (takes about a week) and dumped before releasing a feature (manual process).
- Edge case: Some strings are super long in some languages, make sure UI displays/handles long text correctly.
- Enter YES, NO or N/A
Any UI code that isn’t centered - e.g. has directionality - will need RTL support.
- Prefer .leading and .trailing over .left and .right when laying out view hierarchies.
- Prefer Natural or Left when setting the alignment of text fields and labels.
- Add an example to our catalog.
- Enter YES, NO or N/A
Custom controls should support VoiceOver. See Apple's Accessibility Programming Guide for iOS for further information.
- Test your control on a device in VoiceOver mode and ensure the bahavior is at least as robust as UIKit.
- Enter YES, NO or N/A
Any component that has text, interacts with text, or lays itself out in relation to text should implement Dynamic Type. See Apple's Building Apps with Dynamic Type video for further information.
- Set fonts with
[UIFont preferredTextForStyle:]
or[UIFont mdc_perferredTextForMaterialStyle:]
, or viaUIFontMetrics
. - Implement
mdc_adjustsFontForContentSizeCategory
as aBOOL
property. - Add code that listens to and reacts to the
UIContentSizeCategoryDidChangeNotification
. - Never layout code around text with 'magic numbers'. Instead, use a dynamic layout like Apple's auto layout.
- Make sure labels have an appropriate overflow behavior. Usually that will mean setting label's
numberOfLines
to 0. - Add any necessary client code into your examples.
- Test all your code with Dynamic Type settings from very small to very large.
- Enter YES, NO or N/A
Any component that has visual elements that can be colorized should include a color themer. A color themer applies a set of colors, known as a color scheme, to a component in a systematic way. The user of the color themer passes a color scheme and component to the color themer and the component is automatically colorized in the correct way.
- Make sure the color themer static method signatures adhere to existing conventions:
applyColorScheme:toComponent
. - Add a
ColorThemer
directory with the color themer to thesrc
directory of the component. - Update component entry in
MaterialComponents.podspec
to include the color themer. - Enter YES, NO or N/A
Comments are useful when used properly. In addition, they are necessary for the system of documentation generation used in MDC.
- Carefully review all comments for necessity and brevity.
- Make sure all classes have complete header comments that comply with HeaderDoc for Jazzy parsing.
- Enter YES or NO
Every component has to support all features outlined for it in the Material Design guidelines. It can support additional features or customization.
- Review what the Material Design Guidelines says about the component and make sure the component can at least satisfy those requirements.
- Enter YES or NO
Both documentation and code can contain URLs to assets or additional text. Make sure they are converted to production values.
- Verify all URLs inline are set to production values.
- Enter YES, NO or N/A
Each component must have its own standalone examples in Swift and Objective-C. If you only include a single example, use Swift. If you are including multiple examples, it's preferable to use Swift for some and Objective-C for others so our users can get a feel for how the component works in both languages.
Examples will appear automatically in MDCCatalog if they are placed on disk in conformance to the “Catalog by Convention.” These examples should follow the format set forth in // TODO: Link to doc on examples and supplemental
They should focus on educating thru the catalog’s visual result and the code itself. Include comments when helpful.
- Include an example of typical usage.
- Include examples of additional features of the component, if possible.
- Enter YES or NO
Our users create their views both in code and in Interface Builder. It’s important to support both usages. Almost all components should be able to be added to a view hierarchy thru Interface Builder.
- UIView subclasses must support initWithCoder along with initWithFrame. The recommended practice is to override both init methods and have them both call a commonInit method with required initialization logic.
- Do not include @IBIspectable as it interferes with UIAppearance support.
- Enter YES, NO or N/A
If a component supports Interface Builder usage, then we need to show our users how to do that.
- Include an example of typical usage via storyboard.
- Include examples of additional features of the component, if possible.
- Enter YES, NO or N/A
Sometimes the operating system changes in ways that cause unpredictable or surprising behavior (even if your code is unchanged.)
- Test your component on devices running every major OS version we support. For example, if our current iOS minimum target was 8.0 and the highest version in beta was 10.x, it would be acceptable to test on three versions: 8.x, 9.x and 10.x.
- Enter YES or NO
Many components could be sensibly used in an extension. But sometimes code prevents a component from working correctly in extensions (or at all.)
- Do not use
[UIApplication sharedApplication]
. - Conditional compilation and runtime checks might be necessary in some cases, but generally liberal application of
NS_EXTENSION_UNAVAILABLE_IOS
can take care of most cases. - Enter YES or NO (with reason)
We want to avoid misuse of initializers both in the calling of existing classes and the implementation of our new classes. Aside from being a best practice in Objc, it is mandatory in Swift. Don't forget that some classes have more than one designated initializer (e.g. UIView
.)
- Add the
NS_DESIGNATED_INITIALIZER
macro to new designated initializers in all new classes (even private.) Remember, designated initializers must call an initializer of the super class. All others (the convenience initializers) must call an initializer within the class (self
level, notsuper
). - If a class provides one or more designated initializers, it must also implement all of the designated initializers of its superclass; mark them
NS_DESIGNATED_INITIALIZER
if you want them to still to be designated. If those initializers should no longer be called, declare themNS_UNAVAILABLE
. - If a class has no new designated initializers and no existing designated initializers have been marked
NS_UNAVAILABLE
, nothing needs to be done. - Call convenience initializers that refer to the designated initializer or the designated initializer itself. Only call
init
if you know that it is, or refers to, the designated initializer. - Enter YES or NO
Conforming to NSCoding is necessary for Interface Builder support in views and could be used to serialize non-view classes. Tip: write a unit test for this.
- Implement
initWithCoder:
. - Implement
encodeWithCoder:
. - Enter YES or NO
Classes that set ivar values or perform other commands from the initializer, should avoid duplicate code by writing a common*MDCClass*Init
method to call from all initializers.
- The method should be named
common
+ the name of the class prefixed with MDC +Init
. - The method should be called from all initializers (initWithFrame:, initWithCoder:, etc.)
- Enter YES, NO or N/A
- Components should support autolayout (see:
intrinsicContentSize
.) - Components don’t have to adopt autolayout (adding constraints.)
- When adopting autolayout, be aware of performance implications, such as continually adding/removing subviews.
- Implement
intrinsicContentSize
on your view. This will ensure that, if your view is used in an auto-layout-based hierarchy, that the system knows the view’s preferred size. This is analogous to implementingsizeThatFits:
. - If any size-impacting features of your view (i.e. new text, different image, etc) change, call
invalidateIntrinsicContentSize
to tell the layout engine that something changed.
- Make sure to use .leading and .trailing instead of .left and .right to support RTL languages correctly.
- If your view sets its constraints in
updateConstraints
, overriderequiresConstraintBasedLayout
to return YES. - Make sure your view does not add constraints which force its width/height to be a specific value. Prefer
intrinsicContentSize
along with content-hugging / compression settings (see:UIView
.) - Enter YES, NO or N/A
Advanced Auto Layout Toolbox Auto Layout Performance on iOS
All of our components should work as expected on iOS 11, and support new devices like the iPhone X.
- Make sure your component takes into account the Safe Area and responds to its changes.
- Test it both on 11.0 and 11.1, and pay special attention to the iPhone X on landscape.
- Enter YES, NO or N/A
We use the UIAppearance
proxy with our visible components to allow setting default values of properties and state.
- Attempt to name properties in line with Apple’s style:
UISlider.thumbTintColor
,UISlider.maximumTrackTintColor
,UISwitch.onTintColor
,UINavigationBar.barTintColor
, etc. tintColor
is not compatible with theUIAppearance
proxy. Avoid usingtintColor
unless we are implementing it asUIView
does.NSObject
properties should be nullable.- Avoid “primary”, “secondary” or “accent” in your property names to avoid confusion with our color schemes.
- Enter YES, NO or N/A
We use class properties to pass defaults to instance properties to mimic the functionality of UIAppearance
.
- Properties with defaults, that are objects, should be
null_resettable
. - Properties with defaults should also always return the default if their value is nil.
- Class property defaults should be named as
nameOfPropertyItAffects
+Default
. For example, the classMDCTextInputControllerDefault
inherits fromNSObject
. ItsborderFillColor
has aborderFillColorDefault
class property. - Class property default declarations should be placed directly beneath the property they affect's declaration in the
@interface
. - Class property default implementations should be placed directly beneath the property they affect's implementation in the
@implementation
. - The property they affect should have in its header documentation
Default is [class property name]
, the class property should sayDefault value for [regular property name]
andDefault is [whatever value is default]
. - Class properties that are primitives should have default values assigned when their static variables are declared and defined and not in any instance code like a common init. For example
static BOOL _floatingEnabledDefault = YES;
.
Consider adding support for IBDesignable. If you have created a public subclass of UIView this may be as simple as adding IB_DESIGNABLE above your @interface declaration. Note: Do not include @IBInspectable as it interferes with UIAppearance support.
IB_DESIGNABLE
@interface MDCBrandNewView : UIView
- Verify inclusion of
IB_DESIGNABLE
. - Mark YES, NO or N/A
Nullability annotations improve Swift usage of a component's APIs. Learn more in Apple's documentation.
Material Components explicitly annotate all public APIs rather than use NS_ASSUME_NONNULL_BEGIN
. This is an intentional deviation from Apple’s practice of using the ASSUME
macros. Further reading
- Add nullability annotations to every header of your component.
- Enter YES or NO
Swift coding conventions differ from Objective-C. It's important to consider the experience of users implementing Material Components in Swift targets and add naming annotations, refinements, and other cross-language features accordingly. See Apple's documentation.
- Use lightweight Objective-C generics whenever possible.
- Add Swift annotation macros where appropriate:
NS_SWIFT_NAME
NS_SWIFT_NOTHROW
NS_SWIFT_UNAVAILABLE
NS_REFINED_FOR_SWIFT
- Use
typedef NS_ENUM
for enums to allow truncation of enumeration value name prefixes. - Typedef string constants with proper macros for better swift translation:
NS_STRING_ENUM
for importing as aString
backed enum.NS_EXTENSIBLE_STRING_ENUM
for importing as an extensiblestruct
in Swift.- Mark non-escaping blocks as
NS_NOESCAPE
and non-escaping closures as@noescape
. - Enter YES, NO or N/A
Material Components for iOS is built primarily for adoption with CocoaPods. There is MaterialComponents.podspec
file in the root folder of the project. It contains information on component naming, dependencies, resources, etc. To learn more about .podspec
files, go to CocoaPods.org
- Verify
MaterialComponents.podspec
contains a properly filled out entry for the component. - Enter YES or NO
Material Components should always use umbrella headers to access API of other components.
We have automated some of the above checks in a set of scripts. To run all the checks against every component, run:
scripts/check_components
To run the checks against particular components, list their directories on the command line:
scripts/check_components components/ActivityIndicator components/Buttons
Each check is a small script in the scripts/check
directory. To run only particular checks, use the -c
flag:
scripts/check_components -c scripts/check/readme -c scripts/check/video
Errors are printed out and summarized at the end of the checks:
Error: '/Users/ajsecord/Source/Git/mdc-fork/components/ActivityIndicator/examples' has no Swift examples.
The following components failed: components/ActivityIndicator.
To create a new check, see scripts/check/README.md
.