diff --git a/README.md b/README.md
index ca3ca01..de644d6 100644
--- a/README.md
+++ b/README.md
@@ -14,5 +14,11 @@ component](https://facebook.github.io/react/docs/components-and-props.html)
elements, the element type is compared by identity. After deserialization the
element types are compared by function name.
-Component elements are formatted with a ⍟ character after the element
-name. Properties and children are formatted by [Concordance](https://github.com/concordancejs/concordance).
+[Memoized elements](https://reactjs.org/docs/react-api.html#reactmemo) are
+supported, however different memoizations of the same function are considered
+equal if used with the same properties.
+
+Memoized elements are formatted with a ⍝ character after the element
+name. Component elements are formatted with a ⍟ character. Properties and
+children are formatted by
+[Concordance](https://github.com/concordancejs/concordance).
diff --git a/index.js b/index.js
index 30053ed..295ab7a 100644
--- a/index.js
+++ b/index.js
@@ -21,6 +21,7 @@ exports.serializerVersion = 2
exports.theme = {
react: {
functionType: '\u235F',
+ memoizedType: `\u235D`,
openTag: {
start: '<',
end: '>',
diff --git a/lib/elementFactory.js b/lib/elementFactory.js
index 2697f87..bfed954 100644
--- a/lib/elementFactory.js
+++ b/lib/elementFactory.js
@@ -5,6 +5,7 @@ const diffShallow = require('./diffShallow')
const escapeText = require('./escapeText')
const FRAGMENT_NAME = Symbol.for('react.fragment')
+const MEMO_TYPE = Symbol.for('react.memo')
function factory (api, reactTags) {
const tag = Symbol('@concordance/react.ElementValue')
@@ -62,7 +63,11 @@ function factory (api, reactTags) {
function describe (props) {
const element = props.value
- const type = element.type
+ let type = element.type
+ const hasMemoizedType = type.$$typeof === MEMO_TYPE
+ // Dereference underlying type if memoized.
+ if (hasMemoizedType) type = type.type
+
const hasTypeFn = typeof type === 'function'
const typeFn = hasTypeFn ? type : null
const name = hasTypeFn ? type.displayName || type.name : type
@@ -78,6 +83,7 @@ function factory (api, reactTags) {
return new DescribedElementValue(Object.assign({
children,
+ hasMemoizedType,
hasProperties,
hasTypeFn,
name,
@@ -96,6 +102,7 @@ function factory (api, reactTags) {
super(props)
this.isFragment = props.name === FRAGMENT_NAME
this.name = props.name
+ this.hasMemoizedType = props.hasMemoizedType
this.hasProperties = props.hasProperties
this.hasTypeFn = props.hasTypeFn
@@ -110,13 +117,15 @@ function factory (api, reactTags) {
formatName (theme) {
const formatted = api.wrapFromTheme(theme.react.tagName, this.isFragment ? 'React.Fragment' : this.name)
- return this.hasTypeFn
- ? formatted + theme.react.functionType
- : formatted
+ if (this.hasMemoizedType) return formatted + theme.react.memoizedType
+ if (this.hasTypeFn) return formatted + theme.react.functionType
+ return formatted
}
compareNames (expected) {
- return this.name === expected.name && this.hasTypeFn === expected.hasTypeFn
+ return this.name === expected.name &&
+ this.hasMemoizedType === expected.hasMemoizedType &&
+ this.hasTypeFn === expected.hasTypeFn
}
formatShallow (theme, indent) {
@@ -216,7 +225,14 @@ function factory (api, reactTags) {
}
serialize () {
- return [this.isFragment, this.isFragment ? null : this.name, this.hasProperties, this.hasTypeFn, super.serialize()]
+ return [
+ this.isFragment,
+ this.isFragment ? null : this.name,
+ this.hasMemoizedType,
+ this.hasProperties,
+ this.hasTypeFn,
+ super.serialize()
+ ]
}
}
Object.defineProperty(ElementValue.prototype, 'tag', {value: tag})
@@ -325,11 +341,14 @@ function factory (api, reactTags) {
function DeserializedMixin (base) {
return class extends api.DeserializedMixin(base) {
constructor (state, recursor) {
- super(state[4], recursor)
+ const legacy = state.length === 5
+ super(state[legacy ? 4 : 5], recursor)
+
this.isFragment = state[0]
this.name = this.isFragment ? FRAGMENT_NAME : state[1]
- this.hasProperties = state[2]
- this.hasTypeFn = state[3]
+ this.hasMemoizedType = legacy ? false : state[2]
+ this.hasProperties = state[legacy ? 2 : 3]
+ this.hasTypeFn = state[legacy ? 3 : 4]
}
createRecursor () {
diff --git a/test/backcompat.js b/test/backcompat.js
new file mode 100644
index 0000000..242b6e5
--- /dev/null
+++ b/test/backcompat.js
@@ -0,0 +1,24 @@
+import test from 'ava'
+import {compareDescriptors, describe, deserialize} from 'concordance'
+
+import React from 'react'
+
+import plugin from '..'
+import HelloMessage from './fixtures/react/HelloMessage'
+
+const plugins = [plugin]
+
+const equalsSerialization = (t, buffer, getValue) => {
+ const expected = describe(getValue(), {plugins})
+
+ const deserialized = deserialize(buffer, {plugins})
+ t.true(
+ compareDescriptors(deserialized, expected),
+ 'the deserialized descriptor equals the expected value')
+}
+
+test('element serialization before React.memo support was added',
+ equalsSerialization,
+ Buffer.from('AwAfAAAAAQERARJAY29uY29yZGFuY2UvcmVhY3QBAgEBAQEBAlwAAABiAAAAEwEFEBEBDEhlbGxvTWVzc2FnZQ8PEwEGEQEGT2JqZWN0AQERAQZPYmplY3QQEBAAAQ0AAQEAAQ8AEwECFAEDAAEFEQEEbmFtZRQBAwABBREBBEpvaG4=', 'base64'), // eslint-disable-line max-len
+ () =>