react.dev
Open in
urlscan Pro
76.76.21.164
Public Scan
URL:
https://react.dev/learn/typescript
Submission: On September 26 via manual from HU — Scanned from IT
Submission: On September 26 via manual from HU — Scanned from IT
Form analysis
5 forms found in the DOMPOST https://codesandbox.io/api/v1/sandboxes/define
<form action="https://codesandbox.io/api/v1/sandboxes/define" method="POST" target="_blank" style="visibility: hidden;"><input type="hidden" value="create-react-app" name="environment"><input type="hidden"
value="N4IgZglgNgpgziAXKOAnAxgeggOwCYwAeAdAFYLIjoD2OALjPUiBALYAO1qdABAEowAhujoAaHsB4BlOqggiAstQI8AvjzCpqrHgB0QqISP0BuXTjaduEnukOCGfatV7rN2vQaN0AtHm2Y6FAQjHSm5pZcvPrEmHB0AJ6wcMTocHDhOBEcUTwAguzsGlo6MZgF7JnmNDjxPFouPAC8tvaOznQAFP7oAK6socQA5jB0AKKwA_QAQgkAknid-g1hIACUa2Y4K8SG-DCoSzg8PAA8MnKKyjAAfOYnJ6cVPJh3x2eYF_J0SgRvmyBRCw4NNcIJUAkkGBBFA4DBVED2MIANaCEZkOC0JCgGoMJiIEDAe6eHCCAb6RCeewiYgEABu-lExP0dIOcAgtApngADMRedzGcyQKxBLgufo4hhsPsSORBe99HA7BB2HQMkgJMSToq6ODVpTlt4fEq5Kq4Dx4nr5Q9PAAjXrQPDirzCXwmlVqnj2x3Wh76BjxZ3Ut3Ks08AO8Hw-Rh0prkfysX3akAwUgwYwaw2u42hz2p9OrYkIoUEdiMAg4dAhdWUonvZPB50APQAjAAOPl8pNUo0J5vtzsCwFanvZ91m5sAVkH-iLTIVIHpABEYGX9pXq1zgKpzKpAcDQaSIVCYXCESB2L1bcEsLgCCQABZ0VhQbFUWh4ujMU4AQiXAHkAGEABUAE0AAUxh4J8XzeU4YKgHgoEEHAhiafRGH0OCHyEPA3keAZdVsB9wThOh0JAABVYCADEfDbLDiVOQjBB4UkBgoukQgAdysVZbA_UIKO4iA8DoB8mnpeQYB8ESxIfcRcAgOgIBhHMYRgJoWy7EB8LOFS6FgG4l2oPopjoU5MAMozzEsnDBDw2zbWUBI9NOPAIDpHhRIolYsMsjy6TgzBnLwVzbMwBCbn3CAQTBY9EGhWF4SBNAsAqYg1UIN9cVCZgwF6SsVNoHgFASaZejoOhaE6SRrJgNRa3DZTYEpeI5FQtQ1k1d5DDoXpUGOI4bVOe0qtoG5gHq1RLLG6qcD0zZd3McwiD4ngCGhXooF4Aqio5Y4yoqTpurrE4-oGoaR3czy9JGh8WxuAB1GAoBoAZw2oHhWASHhBEKOzHpHR4yoq8bjnqii5gAch0Vi5s5EAXjuj5AsWrYdxwGK4qPSFEtPFKQDS8pCgxHLBPxFgcmsCpgKkAANYoPDKDKssyNbck2wRtt4Z4Wlphmtmxw9wTxpKz1SqV4iSeBUnScn6DygkACoepOZzCGNCAAC9cCGSlnNQAhUB8DWMZWnBQt-s6NA_HxoVYaAEjalC4GNA4IDALYThFVAhlwSkACZuXYQhvZ4JE8A81DKW5c2shwB61e-8F_ZwHxqvYWPw7AO32W1mAg8D0P4_MB9A-T3204z6gs54OPiVz-gtYLoOQ7D5aE4fABmSvU9wGu64b94m7dHXC54dsS87suABY-79gfM-zxu8_HykWwANmnnBMbLycF-r5f65ztfW8n2ed73xPN8Ppfa5Xkez4nlti473eLZoFQbdHluX-IQOMBWClxwNtZOkdo5DB8LgYIOAZKWm4G3K-FthbxTFgTVQ54GAcGQgwZgdghAMB8MGHw_12AgFUEAA"
name="parameters"><input type="hidden" value="file=%2Fsrc%2FApp.tsx&utm_medium=sandpack" name="query"></form>
POST https://codesandbox.io/api/v1/sandboxes/define
<form action="https://codesandbox.io/api/v1/sandboxes/define" method="POST" target="_blank" style="visibility: hidden;"><input type="hidden" value="create-react-app" name="environment"><input type="hidden"
value="N4IgZglgNgpgziAXKOAnAxgeggOwCYwAeAdAFYLIjoD2OALjPUiBALYAO1qdABAEowAhujoAaHsB4BlOqggiAstQI8AvjzCpqrHgB0QqISP0BuXTjaduEnukOCGfatV7rN2vQaN0AtHm2Y6FAQjHSm5pZcvPrEmHB0AJ6wcMTocHDhOBEcUTwAguzsGlo6MZgF7JnmNDjxPFouPAC8tvaOznQAFP7oAK6socQA5jB0AKKwA_QAQgkAknid-g1hIACUa2Y4K8SG-DCoSzg8PAA8MnKKyjAAfOYnJ6cVPJh3x2eYF_J0SgRvmyBRCw4NNcIJUAkkGBBFA4DBVED2MIANaCEZkOC0JCgGoMJiIEDAe6eHCCAb6RCeewiYgEABu-lExP0dIOcAgtApngADMRedzGcyQKxBLgufo4hhsPsSORBe99HA7BB2HQMkgJMSToq6ODVpTlt4fEq5Kq4Dx4nr5Q9PAAjXrQPDirzCXwmlVqnj2x3Wh76BjxZ3Ut3Ks08AO8Hw-Rh0prkfysX3akAwUgwYwaw2u42hz2p9OrYkIoUEdiMAg4dAhdWUonvZPB50APQAjAAOPl8pNUo0J5vtzsCwFanvZ91m5sAVkH-iLTIVIHpABEYGX9pXq1zgKpzKpAcDQaSIVCYXCESB2L1bcEsLgCCQABZ0VhQbFUWh4ujMU4AQiXAHkAGEABUAE0AAUxh4J8XzeU4YKgHgoEEHAhiafRGH0OCHyEPA3keAZdVsB9wThOh0JAABVYCADEfDbLDiVOQjBB4UkBgoukQgAdysVZbA_UIKO4iA8DoB8mnpeQYB8ESxIfcRcAgOgIBhHMYRgJoWy7EB8LOFS6FgG4l2oPopjoU5MAMozzEsnDBDw2zbWUBI9NOPAIDpHhRIolYsMsjy6TgzBnLwVzbMwBCbn3CAQTBY9EGhWF4SBNAsAqYg1UIN9cVCZhcAYVBoXQGAeAUBJpl6Og6FocCtHYc06xOTAACoWp4YCcPDIheBqngPLgdhkISbzalE0rxNK-1qtoHgWswYlrJgSl4jkVCtmatqeAAdRwybUHDLrppq450BQr1SoKg5XRgPAeBE8S5oW94BsEa9bspZzqFgFCth3LIcDAXpKxU2bysqmacE6SQlvEV73ru1RKXBqqTrq6gGrWTV3kMOhelQY4jhtU5jtm-HYDwJpgHJ27VBuYAltUSzSZwPTNl3cxzCIPj-pgaFeigXggZBjljnKipOixpr6lGfHCZHdzPL04mHxbG5tpgKAaAGcNqB4VgRsEQo7LVkdHhRyHw2U2AKLmAByHRWJpu6Wf0frYreimqdkXp4VeBXMECtm_vMGK4qPSFEtPFKQDS8pCgxHLBPxFgcmsCpgKkAANYoPDKDKssybncgIfnBfyQpmgr9hM6zrYw8PcFI6Ss9UqleIkngVJ0iT-g8oJdrpecwhjQgAAvXAhk-rgCFQHxh5DgHQpG6WwA_HxoVYaAEhWlC4GNA4IDADb9fBIZcEpAAmbl2EIE-kTwDzUMpblF_MVXsZOEVUHPnAfBq9gL8T5r3oKPMey0eCX0vrfN-OAHyX0_qfH-uB_4YyAcSEBbpx4QOvjAjmAMHwAGZEHf1_qgwBPBX4YPXuycBlJ2x4JwP9d-AAWEhZ8UEAPQe8TBYCIEtgAGyMOYXAyc7DkF_y4ZQ4BNDsH0JYcIzmcCBHiLIVIqhPDZF0J4C2aBd98HVGuIg3htD-HEEvjAVgsCBaIIfk_IYPhcDBBwDJS03Ar4330UwpRDd4rN2jqoc8DAODIQYMwOwQgGA-GDD4I27AQCqCAA"
name="parameters"><input type="hidden" value="file=%2Fsrc%2FApp.tsx&utm_medium=sandpack" name="query"></form>
POST https://codesandbox.io/api/v1/sandboxes/define
<form action="https://codesandbox.io/api/v1/sandboxes/define" method="POST" target="_blank" style="visibility: hidden;"><input type="hidden" value="create-react-app" name="environment"><input type="hidden"
value="N4IgZglgNgpgziAXKOAnAxgeggOwCYwAeAdAFYLIjoD2OALjPUiBALYAO1qdABAEowAhujoAaHsB4BlOqggiAstQI8AvjzCpqrHgB0QqISP0BuXTjaduEnukOCGfatV7rN2vQaN0AtHm2Y6FAQjHSm5pZcvPrEmHB0AJ6wcMTocHDhOBEcUTwAguzsGlo6MZgF7JnmNDjxPFouPAC8tvaOznQAFP7oAK6socQA5jB0AKKwA_QAQgkAknid-g1hIACUa2Y4K8SG-DCoSzg8PAA8MnKKyjAAfOYnJ6cVPJh3x2eYF_J0SgRvmyBRCw4NNcIJUAkkGBBFA4DBVED2MIANaCEZkOC0JCgGoMJiIEDAe6eHCCAb6RCeewiYgEABu-lExP0dIOcAgtApngADMRedzGcyQKxBLgufo4hhsPsSORBe99HA7BB2HQMkgJMSToq6ODVpTlt4fEq5Kq4Dx4nr5Q9PAAjXrQPDirzCXwmlVqnj2x3Wh76BjxZ3Ut3Ks08AO8Hw-Rh0prkfysX3akAwUgwYwaw2u42hz2p9OrYkIoUEdiMAg4dAhdWUonvZPB50APQAjAAOPl8pNUo0J5vtzsCwFanvZ91m5sAVkH-iLTIVIHpABEYGX9pXq1zgKpzKpAcDQaSIVCYXCESB2L1bcEsLgCCQABZ0VhQbFUWh4ujMU4AQiXAHkAGEABUAE0AAUxh4J8XzeU4YKgHgoEEHAhiafRGH0OCHyEPA3keAZdVsB9wThOh0JAABVYCADEfDbLDiVOQjBB4UkBgoukQgAdysVZbA_UIKO4iA8DoB8mnpeQYB8ESxIfcRcAgOgIBhHMYRgJoWy7EB8LOFS6FgG4l2oPopjoU5MAMozzEsnDBDw2zbWUBI9NOPAIDpHhRIolYsMsjy6TgzBnLwVzbMwBCbn3CAQTBY9EGhWF4SBNAsAqYg1UIN9cVCZhImsYBejhAQ8F6dADjcEoeAAcmDGqtgiegDmhCrpF1BhNXrGhenoSkcH6W0Dj0HBVEa-gEjLHhAOoXqGFQPIRA5Y50PeAAfGxEjLA0vDI_Q1GJDbJC2mAdrIma5tMHg6RhXpTvahwYAAbX0Hr6H0ABdA6skrWg6iUlSYRkR7KWBzqWkkN66Epbk1HGsBeqW2gLQ6mBSvKg5OktBhQdR8RXWWykLuahakZwNZccerqTjgES6HQB8eE6AnaEyyaYDWamHnQQQ4R7PakBHE5DDoXpUGOAHVKgMGYC2G0eb5xVRmJ_UhfqUYxeOSRiB17GYHEKHKRZnBiBuqA7rhkcCGhXooGhtXxK0bi2JgZ2xlQLRDn0SicGRHBqG445jf0TYi13cxzCIPieGtwRbd4BHKxU5GKk6Tm6xOGo6ievXxA8uAkXph8vpaYq0ZgMqKsOPX0arxSLEB6XUdDn7M7-3gHLwGjPJgZomc5pobhj2LC4ZzpjvZs7ldm97AWu277r11IZ94ABqHhJzUFu29qXhDDIvu0-aIf89Hh9x_DSf-dGfbVBb4kRc1pmR3czy9JteCWxuAB1GAoBoAY4ZqA8FYAkASc0Dh2S_hHesDxTjsBuCrWsS8oaqEsggtWpx7R0DoMjWggEbzIiaMATu3dWSqBuHkPAeAN6WWwbgnA784H0LwTgAh8giHAH3qMChAgyJ0N6Dg2gTCApv2JC3HcP0YpxSPJCRKp4UogDSuUQoGIcqCXxCwHI1gKjASkAADWKB4MoGUsqZCjrkWO8d8iFD7rogxWxpGHnBHIpKZ5UpSniEkeAqR0jqOapogAVFzZyhBjQQAAF64CGJSZyqACCoB8KErYkjzChTARnDQH4fDQlYNABIlI4AoTgMaA4EAwByxAeCIYuBKQACZuTsEIJUpE1DokwxSTAh8LYuYilQDUnAPhcHsA6cSMA2T2QRPunUupTTOk_QfHU3p1TcBDOoCMng3JKnjPoOEqZ9TGnNPDgsgAzMs_pqzhmjPeDst0kT7rtjmcc8wD4AAs5yBlrI2VssZEz7mUhbAANieaNLpm9Ml9M-VczZ2y_n7J4C2V5ILUk4AfICj5lz1nXJOLcvZDzZlHNBT9GgKhMm4smQ84gdSYCsHmeYW2XNWkeVQj4XAwQcAyUtNwA5yKYFOPiq4hRqhzwMA4MhBgzA7BCAYD4YMPhBCFBAKoIAA"
name="parameters"><input type="hidden" value="file=%2Fsrc%2FApp.tsx&utm_medium=sandpack" name="query"></form>
POST https://codesandbox.io/api/v1/sandboxes/define
<form action="https://codesandbox.io/api/v1/sandboxes/define" method="POST" target="_blank" style="visibility: hidden;"><input type="hidden" value="create-react-app" name="environment"><input type="hidden"
value="N4IgZglgNgpgziAXKOAnAxgeggOwCYwAeAdAFYLIjoD2OALjPUiBALYAO1qdABAEowAhujoAaHsB4BlOqggiAstQI8AvjzCpqrHgB0QqISP0BuXTjaduEnukOCGfatV7rN2vQaN0AtHm2Y6FAQjHSm5pZcvPrEmHB0AJ6wcMTocHDhOBEcUTwAguzsGlo6MZgF7JnmNDjxPFouPAC8tvaOznQAFP7oAK6socQA5jB0AKKwA_QAQgkAknid-g1hIACUa2Y4K8SG-DCoSzg8PAA8MnKKyjAAfOYnJ6cVPJh3x2eYF_J0SgRvmyBRCw4NNcIJUAkkGBBFA4DBVED2MIANaCEZkOC0JCgGoMJiIEDAe6eHCCAb6RCeewiYgEABu-lExP0dIOcAgtApngADMRedzGcyQKxBLgufo4hhsPsSORBe99HA7BB2HQMkgJMSToq6ODVpTlt4fEq5Kq4Dx4nr5Q9PAAjXrQPDirzCXwmlVqnj2x3Wh76BjxZ3Ut3Ks08AO8Hw-Rh0prkfysX3akAwUgwYwaw2u42hz2p9OrYkIoUEdiMAg4dAhdWUonvZPB50APQAjAAOPl8pNUo0J5vtzsCwFanvZ91m5sAVkH-iLTIVIHpABEYGX9pXq1zgKpzKpAcDQaSIVCYXCESB2L1bcEsLgCCQABZ0VhQbFUWh4ujMU4AQiXAHkAGEABUAE0AAUxh4J8XzeU4YKgHgoEEHAhiafRGH0OCHyEPA3keAZdVsB9wThOh0JAABVYCADEfDbLDiVOQjBB4UkBgoukQgAdysVZbA_UIKO4iA8DoB8mnpeQYB8ESxIfcRcAgOgIBhHMYRgJoWy7EB8LOFS6FgG4l2oPopjoU5MAMozzEsnDBDw2zbWUBI9NOPAIDpHhRIolYsMsjy6TgzBnLwVzbMwBCbn3CAQTBY9EGhWF4SBNAsAqYg1UIN9cVCZhImsSQ7CEBhAMEwgxB4Xo4TK-giEq6qYBkBwYDUYoPAAcmDDqtnMRIyx4YCcIGZpPGCIYn30HgAB9PDwcFkSm2bFQSeIYETEAthqOohvWmBaoYCrRuKlqDvq05doGG4lhAOBVoYDbNnMapaDqRqAHFRku1qWk6NZmhuKqavKrpvrOiqnqyHAiD4ngCGhXooF4MBekrFTaB4BQEgqP7NXebbeAAbXEvbxDI76AF1Rsa5qGAu4bbk6DrxqfDrIeJQw6F6VBjiOG16b28G6GIcCtC4ghUB4OkYV6TTgBJgZVD0_msbKnJofoF5lY-MGQZFsXRIOPS1l3Z6cBRtGOWOVXtE4DWun-usTgJ8MGepuFProb6_t695Oe53mR3czztcedgbkA7m9l4BWYFrWPVEs8Og8wQLjdNnAYrio9IUS08UtuqUMvIHLBPxFgcmsCpgKkAANdrShAWIMqyzIYdyeHBER3hnhaau662LPD3BXOkrPVKpXiJJ4FSdJS7q8uACo8ZOZzCGNCAAC9cCGSlnNQCWfDXrYdyh0KEhXjQPx8aFWGgBJKTgFC4GNA4IDALYThFVAhlwSkACZuTsEIJ_HgSI8AeVQpSbkJ8zYPhbJfb-v8cA-DoNQdg0DQFgGvuyTecceD_3_sA2BUMHz_0QeCZBqD0GYOJNg-gG88EAKASAjO5gHwAGYKE_1wNQjBPAYF0JwVvfB7ZiFsJwA-AALNwqhaD-GCPePQt0IjKQtgAGziJwKfdhk5ZG8PkbQpRwimE8BbFIrROjJHqP0SgwxAisEmNEUQ1h2izY0BUE7K-DDcGiOIP_daJDzCI0vuAyBQwfC4GCNDY0upuDMMsWbIe8VR751UOeB67BkIMGYCdBgPhgw-EEIUEAqggA"
name="parameters"><input type="hidden" value="file=%2Fsrc%2FApp.tsx&utm_medium=sandpack" name="query"></form>
POST https://codesandbox.io/api/v1/sandboxes/define
<form action="https://codesandbox.io/api/v1/sandboxes/define" method="POST" target="_blank" style="visibility: hidden;"><input type="hidden" value="create-react-app" name="environment"><input type="hidden"
value="N4IgZglgNgpgziAXKOAnAxgeggOwCYwAeAdAFYLIjoD2OALjPUiBALYAO1qdABAEowAhujoAaHsB4BlOqggiAstQI8AvjzCpqrHgB0QqISP0BuXTjaduEnukOCGfatV7rN2vQaN0AtHm2Y6FAQjHSm5pZcvPrEmHB0AJ6wcMTocHDhOBEcUTwAguzsGlo6MZgF7JnmNDjxPFouPAC8tvaOznQAFP7oAK6socQA5jB0AKKwA_QAQgkAknid-g1hIACUa2Y4K8SG-DCoSzg8PAA8MnKKyjAAfOYnJ6cVPJh3x2eYF_J0SgRvmyBRCw4NNcIJUAkkGBBFA4DBVED2MIANaCEZkOC0JCgGoMJiIEDAe6eHCCAb6RCeewiYgEABu-lExP0dIOcAgtApngADMRedzGcyQKxBLgufo4hhsPsSORBe99HA7BB2HQMkgJMSToq6ODVpTlt4fEq5Kq4Dx4nr5Q9PAAjXrQPDirzCXwmlVqnj2x3Wh76BjxZ3Ut3Ks08AO8Hw-Rh0prkfysX3akAwUgwYwaw2u42hz2p9OrYkIoUEdiMAg4dAhdWUonvZPB50APQAjAAOPl8pNUo0J5vtzsCwFanvZ91m5sAVkH-iLTIVIHpABEYGX9pXq1zgKpzKpAcDQaSIVCYXCESB2L1bcEsLgCCQABZ0VhQbFUWh4ujMU4AQiXAHkAGEABUAE0AAUxh4J8XzeU4YKgHgoEEHAhiafRGH0OCHyEPA3keAZdVsB9wThOh0JAABVYCADEfDbLDiVOQjBB4UkBgoukQgAdysVZbA_UIKO4iA8DoB8mnpeQYB8ESxIfcRcAgOgIBhHMYRgJoWy7EB8LOFS6FgG4l2oPopjoU5MAMozzEsnDBDw2zbWUBI9NOPAIDpHhRIolYsMsjy6TgzBnLwVzbMwBCbn3CAQTBY9EGhWF4SBNAsAqYg1UIN9cVCZhImsSRejhGQHBgNRig8AByYMqq2cwiD4ngCGhXooF4MBekrFTaB4GiuFYTo1k1d4ajqABtOkYV6GBxDIgA1aaYAAXWaHhipgUqGCWEBAJI1DyvJdZ6veTruo5Y59rwWA9pQkZOhgVl6EpARXWIW6DrGJ6LIACWAhQABk5hwS9xkmUIbmGusbQWpaHu-1JelQPY6GA8ERjoYgpqgGbNiLcxiUMOgkeOI4bVOPTydwUGeGxmammAOn4R4WgPpGBmrpu_aRnUV4R0edgbkWnGYFrJnVEswWR0svS8ZwHccBiuKj0hRLTxSkA0vKQoMRywT8RYHJrAqYCpAADUq0oQFiDKssyRrchawQ2t4Z4WhN82tiVw9wVVpKz1SqV4iSeBUnSPX6DygkACoRpOZzCGNCAAC9cCGSlnNQAhUB8BOtgV8xQoSOONA_HxoVYaAEkpOAULgY0DggMAthOEVUCGXBKQAJm5dhCBbngkTwDzUMpbl84JnAHxbEu247nAfDoah2DHgewDL9lk9Fngu67vuJ6yKeu9n9HcEX5fV-Jdf6CTrfu97_vd0nh8AGYT_bs-l5Xnhx6vjeU-3u2feT9D4PgACzv3nufb-v9Tr_zvjwFsAA2YB8tn6TkgZ_C-P817wMAWA1BBcp5IMwQvL-l84E303oAvej80GHxoCoaGpcqEAMpNpLuMBWAH3MG1EuQ8R5DB8LgYIOAZKWm4PfQhk9vbxT9urVQ54GAcGQgwZgdghAMB8MGHwghCggFUEAA"
name="parameters"><input type="hidden" value="file=%2Fsrc%2FApp.tsx&utm_medium=sandpack" name="query"></form>
Text Content
React v18.3.1 Search⌘CtrlK Learn Reference Community Blog GET STARTED * Quick Start * Tutorial: Tic-Tac-Toe * Thinking in React * Installation * Start a New React Project * Add React to an Existing Project * Editor Setup * Using TypeScript * React Developer Tools * React Compiler - This feature is available in the latest Canary * LEARN REACT * Describing the UI * Your First Component * Importing and Exporting Components * Writing Markup with JSX * JavaScript in JSX with Curly Braces * Passing Props to a Component * Conditional Rendering * Rendering Lists * Keeping Components Pure * Your UI as a Tree * Adding Interactivity * Responding to Events * State: A Component's Memory * Render and Commit * State as a Snapshot * Queueing a Series of State Updates * Updating Objects in State * Updating Arrays in State * Managing State * Reacting to Input with State * Choosing the State Structure * Sharing State Between Components * Preserving and Resetting State * Extracting State Logic into a Reducer * Passing Data Deeply with Context * Scaling Up with Reducer and Context * Escape Hatches * Referencing Values with Refs * Manipulating the DOM with Refs * Synchronizing with Effects * You Might Not Need an Effect * Lifecycle of Reactive Effects * Separating Events from Effects * Removing Effect Dependencies * Reusing Logic with Custom Hooks Is this page useful? Learn React Installation USING TYPESCRIPT TypeScript is a popular way to add type definitions to JavaScript codebases. Out of the box, TypeScript supports JSX and you can get full React Web support by adding @types/react and @types/react-dom to your project. YOU WILL LEARN * TypeScript with React Components * Examples of typing with Hooks * Common types from @types/react * Further learning locations INSTALLATION All production-grade React frameworks offer support for using TypeScript. Follow the framework specific guide for installation: * Next.js * Remix * Gatsby * Expo ADDING TYPESCRIPT TO AN EXISTING REACT PROJECT To install the latest version of React’s type definitions: Terminal Copy npm install @types/react @types/react-dom The following compiler options need to be set in your tsconfig.json: 1. dom must be included in lib (Note: If no lib option is specified, dom is included by default). 2. jsx must be set to one of the valid options. preserve should suffice for most applications. If you’re publishing a library, consult the jsx documentation on what value to choose. TYPESCRIPT WITH REACT COMPONENTS NOTE Every file containing JSX must use the .tsx file extension. This is a TypeScript-specific extension that tells TypeScript that this file contains JSX. Writing TypeScript with React is very similar to writing JavaScript with React. The key difference when working with a component is that you can provide types for your component’s props. These types can be used for correctness checking and providing inline documentation in editors. Taking the MyButton component from the Quick Start guide, we can add a type describing the title for the button: App.tsx App.tsx Reset ForkTypeScript Playground function MyButton({ title }: { title: string }) { return ( <button>{title}</button> ); } export default function MyApp() { return ( <div> <h1>Welcome to my app</h1> <MyButton title="I'm a button" /> </div> ); } NOTE These sandboxes can handle TypeScript code, but they do not run the type-checker. This means you can amend the TypeScript sandboxes to learn, but you won’t get any type errors or warnings. To get type-checking, you can use the TypeScript Playground or use a more fully-featured online sandbox. This inline syntax is the simplest way to provide types for a component, though once you start to have a few fields to describe it can become unwieldy. Instead, you can use an interface or type to describe the component’s props: App.tsx App.tsx Reset ForkTypeScript Playground interface MyButtonProps { /** The text to display inside the button */ title: string; /** Whether the button can be interacted with */ disabled: boolean; } function MyButton({ title, disabled }: MyButtonProps) { return ( <button disabled={disabled}>{title}</button> ); } export default function MyApp() { return ( <div> <h1>Welcome to my app</h1> <MyButton title="I'm a disabled button" disabled={true}/> </div> ); } Show more The type describing your component’s props can be as simple or as complex as you need, though they should be an object type described with either a type or interface. You can learn about how TypeScript describes objects in Object Types but you may also be interested in using Union Types to describe a prop that can be one of a few different types and the Creating Types from Types guide for more advanced use cases. EXAMPLE HOOKS The type definitions from @types/react include types for the built-in Hooks, so you can use them in your components without any additional setup. They are built to take into account the code you write in your component, so you will get inferred types a lot of the time and ideally do not need to handle the minutiae of providing the types. However, we can look at a few examples of how to provide types for Hooks. USESTATE The useState Hook will re-use the value passed in as the initial state to determine what the type of the value should be. For example: // Infer the type as "boolean" const [enabled, setEnabled] = useState(false); This will assign the type of boolean to enabled, and setEnabled will be a function accepting either a boolean argument, or a function that returns a boolean. If you want to explicitly provide a type for the state, you can do so by providing a type argument to the useState call: // Explicitly set the type to "boolean" const [enabled, setEnabled] = useState<boolean>(false); This isn’t very useful in this case, but a common case where you may want to provide a type is when you have a union type. For example, status here can be one of a few different strings: type Status = "idle" | "loading" | "success" | "error"; const [status, setStatus] = useState<Status>("idle"); Or, as recommended in Principles for structuring state, you can group related state as an object and describe the different possibilities via object types: type RequestState = | { status: 'idle' } | { status: 'loading' } | { status: 'success', data: any } | { status: 'error', error: Error }; const [requestState, setRequestState] = useState<RequestState>({ status: 'idle' }); USEREDUCER The useReducer Hook is a more complex Hook that takes a reducer function and an initial state. The types for the reducer function are inferred from the initial state. You can optionally provide a type argument to the useReducer call to provide a type for the state, but it is often better to set the type on the initial state instead: App.tsx App.tsx Reset ForkTypeScript Playground import {useReducer} from 'react'; interface State { count: number }; type CounterAction = | { type: "reset" } | { type: "setCount"; value: State["count"] } const initialState: State = { count: 0 }; function stateReducer(state: State, action: CounterAction): State { switch (action.type) { case "reset": return initialState; case "setCount": return { ...state, count: action.value }; default: throw new Error("Unknown action"); } } export default function App() { const [state, dispatch] = useReducer(stateReducer, initialState); const addFive = () => dispatch({ type: "setCount", value: state.count + 5 }); const reset = () => dispatch({ type: "reset" }); return ( <div> <h1>Welcome to my counter</h1> <p>Count: {state.count}</p> <button onClick={addFive}>Add 5</button> <button onClick={reset}>Reset</button> </div> ); } Show more We are using TypeScript in a few key places: * interface State describes the shape of the reducer’s state. * type CounterAction describes the different actions which can be dispatched to the reducer. * const initialState: State provides a type for the initial state, and also the type which is used by useReducer by default. * stateReducer(state: State, action: CounterAction): State sets the types for the reducer function’s arguments and return value. A more explicit alternative to setting the type on initialState is to provide a type argument to useReducer: import { stateReducer, State } from './your-reducer-implementation'; const initialState = { count: 0 }; export default function App() { const [state, dispatch] = useReducer<State>(stateReducer, initialState); } USECONTEXT The useContext Hook is a technique for passing data down the component tree without having to pass props through components. It is used by creating a provider component and often by creating a Hook to consume the value in a child component. The type of the value provided by the context is inferred from the value passed to the createContext call: App.tsx App.tsx Reset ForkTypeScript Playground import { createContext, useContext, useState } from 'react'; type Theme = "light" | "dark" | "system"; const ThemeContext = createContext<Theme>("system"); const useGetTheme = () => useContext(ThemeContext); export default function MyApp() { const [theme, setTheme] = useState<Theme>('light'); return ( <ThemeContext.Provider value={theme}> <MyComponent /> </ThemeContext.Provider> ) } function MyComponent() { const theme = useGetTheme(); return ( <div> <p>Current theme: {theme}</p> </div> ) } Show more This technique works when you have a default value which makes sense - but there are occasionally cases when you do not, and in those cases null can feel reasonable as a default value. However, to allow the type-system to understand your code, you need to explicitly set ContextShape | null on the createContext. This causes the issue that you need to eliminate the | null in the type for context consumers. Our recommendation is to have the Hook do a runtime check for it’s existence and throw an error when not present: import { createContext, useContext, useState, useMemo } from 'react'; // This is a simpler example, but you can imagine a more complex object here type ComplexObject = { kind: string }; // The context is created with `| null` in the type, to accurately reflect the default value. const Context = createContext<ComplexObject | null>(null); // The `| null` will be removed via the check in the Hook. const useGetComplexObject = () => { const object = useContext(Context); if (!object) { throw new Error("useGetComplexObject must be used within a Provider") } return object; } export default function MyApp() { const object = useMemo(() => ({ kind: "complex" }), []); return ( <Context.Provider value={object}> <MyComponent /> </Context.Provider> ) } function MyComponent() { const object = useGetComplexObject(); return ( <div> <p>Current object: {object.kind}</p> </div> ) } USEMEMO The useMemo Hooks will create/re-access a memorized value from a function call, re-running the function only when dependencies passed as the 2nd parameter are changed. The result of calling the Hook is inferred from the return value from the function in the first parameter. You can be more explicit by providing a type argument to the Hook. // The type of visibleTodos is inferred from the return value of filterTodos const visibleTodos = useMemo(() => filterTodos(todos, tab), [todos, tab]); USECALLBACK The useCallback provide a stable reference to a function as long as the dependencies passed into the second parameter are the same. Like useMemo, the function’s type is inferred from the return value of the function in the first parameter, and you can be more explicit by providing a type argument to the Hook. const handleClick = useCallback(() => { // ... }, [todos]); When working in TypeScript strict mode useCallback requires adding types for the parameters in your callback. This is because the type of the callback is inferred from the return value of the function, and without parameters the type cannot be fully understood. Depending on your code-style preferences, you could use the *EventHandler functions from the React types to provide the type for the event handler at the same time as defining the callback: import { useState, useCallback } from 'react'; export default function Form() { const [value, setValue] = useState("Change me"); const handleChange = useCallback<React.ChangeEventHandler<HTMLInputElement>>((event) => { setValue(event.currentTarget.value); }, [setValue]) return ( <> <input value={value} onChange={handleChange} /> <p>Value: {value}</p> </> ); } USEFUL TYPES There is quite an expansive set of types which come from the @types/react package, it is worth a read when you feel comfortable with how React and TypeScript interact. You can find them in React’s folder in DefinitelyTyped. We will cover a few of the more common types here. DOM EVENTS When working with DOM events in React, the type of the event can often be inferred from the event handler. However, when you want to extract a function to be passed to an event handler, you will need to explicitly set the type of the event. App.tsx App.tsx Reset ForkTypeScript Playground import { useState } from 'react'; export default function Form() { const [value, setValue] = useState("Change me"); function handleChange(event: React.ChangeEvent<HTMLInputElement>) { setValue(event.currentTarget.value); } return ( <> <input value={value} onChange={handleChange} /> <p>Value: {value}</p> </> ); } Show more There are many types of events provided in the React types - the full list can be found here which is based on the most popular events from the DOM. When determining the type you are looking for you can first look at the hover information for the event handler you are using, which will show the type of the event. If you need to use an event that is not included in this list, you can use the React.SyntheticEvent type, which is the base type for all events. CHILDREN There are two common paths to describing the children of a component. The first is to use the React.ReactNode type, which is a union of all the possible types that can be passed as children in JSX: interface ModalRendererProps { title: string; children: React.ReactNode; } This is a very broad definition of children. The second is to use the React.ReactElement type, which is only JSX elements and not JavaScript primitives like strings or numbers: interface ModalRendererProps { title: string; children: React.ReactElement; } Note, that you cannot use TypeScript to describe that the children are a certain type of JSX elements, so you cannot use the type-system to describe a component which only accepts <li> children. You can see an example of both React.ReactNode and React.ReactElement with the type-checker in this TypeScript playground. STYLE PROPS When using inline styles in React, you can use React.CSSProperties to describe the object passed to the style prop. This type is a union of all the possible CSS properties, and is a good way to ensure you are passing valid CSS properties to the style prop, and to get auto-complete in your editor. interface MyComponentProps { style: React.CSSProperties; } FURTHER LEARNING This guide has covered the basics of using TypeScript with React, but there is a lot more to learn. Individual API pages on the docs may contain more in-depth documentation on how to use them with TypeScript. We recommend the following resources: * The TypeScript handbook is the official documentation for TypeScript, and covers most key language features. * The TypeScript release notes cover new features in depth. * React TypeScript Cheatsheet is a community-maintained cheatsheet for using TypeScript with React, covering a lot of useful edge cases and providing more breadth than this document. * TypeScript Community Discord is a great place to ask questions and get help with TypeScript and React issues. PreviousEditor Setup NextReact Developer Tools -------------------------------------------------------------------------------- Copyright © Meta Platforms, Inc no uwu plz uwu? Logo by@sawaratsuki1004 Learn React Quick Start Installation Describing the UI Adding Interactivity Managing State Escape Hatches API Reference React APIs React DOM APIs Community Code of Conduct Meet the Team Docs Contributors Acknowledgements More Blog React Native Privacy Terms ON THIS PAGE * Overview * Installation * Adding TypeScript to an existing React project * TypeScript with React Components * Example Hooks * useState * useReducer * useContext * useMemo * useCallback * Useful Types * DOM Events * Children * Style Props * Further learning