Heroes bileşeni, diğer kahraman bileşenlerini kullanır ve şimdiye kadarki en karmaşık olanıdır. Uygulamanın Angular versiyonuna bakarak, DOM öğelerinin bir madde listesi oluşturabiliriz.
ListHeader alt bileşeni
div
HeroList ve HeroDetail arasında geçiş yapan bir rota
ModalYesNo bileşeni (silme işlemi için)
feat/Heroes adında bir dal oluşturun. src/heroes/ klasörü altında Heroes.cy.tsx, Heroes.tsx adlı 2 dosya oluşturun. Her zamanki gibi, bileşenin işlemesini en aza indirgeyerek başlayın; dosyalara aşağıdakileri kopyalayın ve yarn cy:open-ct ile koşucuyu açtıktan sonra testi çalıştırın.
Testin çalışması için, işlemede alt bileşeni eklemeli ve gerekli nitelikleri eklemeliyiz; title, handleAdd, handleRefresh. Şimdilik herhangi bir değer uygundur (Kırmızı 1).
Hala bir test hatası alıyoruz ve bu tanıdık bir yönlendirme hatası. Bu, alt bileşen ListHeader'ın react-router'ı kullanması nedeniyledir. ListHeader ve HeaderBarBrand bileşenlerinden hatırlayın ki, react-router kullandığımız herhangi bir zaman, bileşen testinde monte edilen bileşeni BrowserRouter içinde sarmalıyız (Yeşil 1).
Simgelere tıkladığımızda tip hataları alırız. ListHeader bileşenini izole olarak test ederken, handleAdd ve handleRefresh için cy.stub kullandığımızı hatırlayın. Şimdi bileşen, bir alt bileşen olarak kullanılıyor ve React, cy.stub kullanamaz. Çocuğun tüketicisi / ebeveyni bu işleyici işlevi uygulamalıdır.
İki tıklamayla başarısız olan testleri geliştirelim. Şimdilik HeroDetail.cy.tsx ve HeroList.cy.tsx testlerinde yaptığımız gibi, konsol günlüklerini takip etmek yeterlidir (Kırmızı 2).
Heroes işlendiğinde, önce ListHeader ve HeroList görüntülenir. Bir kahramanı düzenlersek, HeroDetail görüntülenir. Bir kahramanı silersek, ModalYesNo gösterilir. İlk önce HeroList'e, sonra modale odaklanacağız. Daha sonraki bir bölümde yönlendirmeyi ayarladıktan sonra HeroDetail ile ilgileneceğiz.
HeroList alt bileşeni
Önce, HeroList işlemenin kontrol edildiği basit bir testle başlarız (Kırmızı 3).
Alt HeroList bileşenini bileşenimize ekliyoruz. Bir kahramanlar özelliği gerektirir. Bir fikir, alt bileşenler için bileşen testlerine bakmak, nasıl kullanıldıklarını görmek ve ebeveyn bileşenin testlerini yazarken bu belgelendirmeyi temel alarak çalışmaktır. Ebeveyn düzeyinde herhangi bir testi tekrarlamamıza gerek yoktur, ancak alt bileşenin nasıl monte edileceği konusunda bir fikir vermek için yardımı kullanabiliriz. HeroList.cy.tsx'ye bir göz atın. Cypress düzeneği içe aktarıyoruz ve bir özellik olarak iletiyoruz. Benzer bir süreç tekrar edebilir ve veri ve durumla ilgili kararları daha sonra yapmak zorunda kalana kadar erteleyebiliriz (Yeşil 3).
Bir bileşen, kaynak klasörünün dışındaki bir dosyayı içe aktarıyorsa, bileşen izole olarak çalışacaktır, ancak daha büyük uygulama derlenmeyecektir. cypress/fixtures/ içindeki heroes.json dosyasının src/heroes içinde bir kopyasını oluşturun ve Heroes bileşenini bu dosyayı kullanacak şekilde güncelleyin. Ağ verileriyle çalışırken daha sonra bunu zarif bir şekilde ele alacağız.
Bir kez daha çocuk bileşenlerin testlerine bakabilir, nasıl kullanıldıklarını görebilir ve ebeveyn bileşeni için testler yazarken bu belgelendirmeden yola çıkabiliriz. ModalYesNo.cy.tsx, message adlı bir string, onYes ve onNo olayları için özelliklere sahiptir. Ayrıca, modalın açılıp kapanmasına izin veren dahili bir durumu da destekler.
Hatalı bir test yazalım. Şimdilik modal için bir açma/kapama düğmemiz yok, bu nedenle sadece yeni modal testini çalıştırmalıyız (Kırmızı 4).
Çocuk bileşenini oluşturmak için, sadece message, onNo, onYes özelliklerini eklememiz yeterlidir. Şimdilik boş stringler olmaları sorun değil (Yeşil 4).
// src/components/Heroes.tsximport ListHeader from"../components/ListHeader";import ModalYesNo from"components/ModalYesNo";import HeroList from"./HeroList";import heroes from"./heroes.json";exportdefaultfunctionHeroes() {constaddNewHero= () =>console.log("handleAdd");consthandleRefresh= () =>console.log("handleRefresh");return ( <divdata-cy="heroes"> <ListHeadertitle="Heroes"handleAdd={addNewHero}handleRefresh={handleRefresh} /> <div> <div> <HeroListheroes={heroes} /> </div> </div> <ModalYesNomessage="Would you like to delete the hero?"onNo={""}onYes={""} /> </div> );}
Bu testi çalıştırdıktan sonra, gerçekten o modalı kapatıp Heroes bileşenimizi görmek istiyoruz. Bu ihtiyaç için başarısız bir test yazalım (Kırmızı 5).
Buradan itibaren, kısaltma amacıyla, bir test .only ile çalıştırıldığında, sadece ilgili kısmın kodunu gösteriyor olacağız.
Cypress koşucusunda func.apply is not a function hatası alıyoruz. Bu hatayla tanışın, olay işleyicimizin bir şey yapmadığı anlamına gelir. Şimdilik bunu çözmek için console.log kullanan bir işlev kullanın (Yeşil 5).
// src/components/Heroes.tsximport ListHeader from"../components/ListHeader";import ModalYesNo from"components/ModalYesNo";import HeroList from"./HeroList";import heroes from"./heroes.json";exportdefaultfunctionHeroes() {constaddNewHero= () =>console.log("handleAdd");consthandleRefresh= () =>console.log("handleRefresh");return ( <divdata-cy="heroes"> <ListHeadertitle="Heroes"handleAdd={addNewHero}handleRefresh={handleRefresh} /> <div> <div> <HeroListheroes={heroes} /> </div> </div> <ModalYesNomessage="Would you like to delete the hero?"onNo={() =>console.log("handleCloseModal")}onYes={""} /> </div> );}
Kendi işlevine çevirebiliriz (Yeniden düzenleme 5).
// src/components/Heroes.tsximport ListHeader from"../components/ListHeader";import ModalYesNo from"components/ModalYesNo";import HeroList from"./HeroList";import heroes from"./heroes.json";exportdefaultfunctionHeroes() {constaddNewHero= () =>console.log("handleAdd");consthandleRefresh= () =>console.log("handleRefresh");consthandleCloseModal= () => () =>console.log("handleCloseModal");return ( <divdata-cy="heroes"> <ListHeadertitle="Heroes"handleAdd={addNewHero}handleRefresh={handleRefresh} /> <div> <div> <HeroListheroes={heroes} /> </div> </div> <ModalYesNomessage="Would you like to delete the hero?"onNo={handleCloseModal}onYes={""} /> </div> );}
HeroDetail işlevinde useState kancasını ele aldık. O bölümde iki ana nokta vardı. İlk olarak, kullanıcı arayüzü durum yönetimimizi iki kategoriye basitleştirebiliriz:
Kullanıcı arayüzü durumu: modal açık, öğe vurgulanmış vb.
İkinci ana nokta, durumu en alakalı olduğu yerde yönetmeyi tercih etmemizdir. Bu durumda, modalın açık veya kapalı olması Heroes bileşeninde en alakalıdır ve useState kancası, bunu en basit şekilde karşılamak için kullanılır.
Modal hakkında 3 gereksinimimiz var. Akış şu şekildedir:
Heroes oluşturulduğunda modalın kapalı olmasını istiyoruz.
Bir kahramanı silmek istediğimizde, modalı göstermek istiyoruz.
Bu durum geçişini çalıştırmak için useState kullanmamız gerekiyor. Sert kodlanmış false değerini sevmiyoruz ve kancanın başlangıç durumu olarak kullanılabilir. Bu noktada, testin hala başarısız olması bekleniyor.
HeroList ve ModalYesNo bileşenlerinden Heroes bileşenine durumu taşıma
showModal orada harika görünüyor, ancak Delete düğmesine tıklandığında setShowModal'ı doğru olarak ayarlamamız gerekiyor. Konsola bakın, handleDeleteHero çağrılıyor ve bu işlev HeroList bileşeninde yaşıyor. Bu, iki alt bileşenin durumu paylaştığına dair bir ipucudur.
Bileşenler durumu paylaşıyorsa, durumu en yakın ortak ata bileşenine yükseltin.
Ortak ata bileşeni çok derin ve durumu yükseltmek özellik iletimi ile sonuçlanıyorsa, React'ın context API'sini kullanın.
Bunun ötesinde, durum yönetimi kütüphaneleri kullanın.
Bizim durumumuzda Heroes bileşeni, HeroList ve ModalYesNo adlı iki alt bileşeni barındırıyor; durumu ebeveyn bileşene yükseltmek en kolay seçenektir.
ModalYesNo bileşeni zaten onYes ve onNoonClick işleyicilerini üstte iletiyor. Öte yandan, HeroList kendi handleDeleteHeroonClick işleyicisini uygular. Bunun yerine, HeroList bileşenine Heroes bileşenindeki setShowModal tarafından yönlendirilen bir handleDeleteHero özelliği geçirmemiz gerekiyor. Bu nedenle, HeroList bileşeninde bir değişiklik yapmamız gerekiyor. Kendi uyguladığımız handleDeleteHero işlevini kaldırıyoruz ve bunun yerine bir özellik olarak geçiriyoruz. Şimdilik özellik türünü genel bırakıyoruz.
Eşleşen testi, yeni handleDeleteHero özelliğini kabul etmek için güncelliyoruz. Üzerine tıklanması durumunda çağrıldığını sağlamak için cy.stub kullanmak yeterlidir.
Ebeveyn bileşen Heroes'a dönersek, şimdi bir handleDeleteHero özelliği geçirebiliriz. Değerinin, setShowModal(<boolean arg>)'yi döndüren bir işlev olması gerekiyor. Tüm tıklama işleyicilerinin neden işlev olması gerekiyor? React belgelerine göre, JSX kullanırken olay işleyici olarak bir işlevi geçirirsiniz, bir dize değil. Yaptığımız değişikliklerden sonra, test başarılı oldu (Yeşil 7).
Buradayken, modal akışının diğer kolunu da kapsayabiliriz; onayda evet'e tıklandığında, modalı kapatmalıyız ve şimdilik en azından bir şeyi console.log yapmalıyız (Kırmızı 10).
// src/components/Heroes.cy.tsximport Heroes from"./Heroes";import { BrowserRouter } from"react-router-dom";import"../styles.scss";describe("Heroes", () => {it("should handle hero add and refresh", () => {cy.window().its("console").then((console) =>cy.spy(console,"log").as("log"));cy.mount( <BrowserRouter> <Heroes /> </BrowserRouter> );cy.getByCy("list-header");cy.getByCy("add-button").click();cy.get("@log").should("have.been.calledWith","handleAdd");cy.getByCy("refresh-button").click();cy.get("@log").should("have.been.calledWith","handleRefresh"); });it("should display hero list on render", () => {cy.mount( <BrowserRouter> <Heroes /> </BrowserRouter> );cy.getByCy("hero-list"); });constinvokeHeroDelete= () => {cy.getByCy("delete-button").first().click();cy.getByCy("modal-yes-no").should("be.visible"); };it("should go through the modal flow", () => {cy.window().its("console").then((console) =>cy.spy(console,"log").as("log"));cy.mount( <BrowserRouter> <Heroes /> </BrowserRouter> );cy.getByCy("modal-yes-no").should("not.exist");cy.log("do not delete flow");invokeHeroDelete();cy.getByCy("button-no").click();cy.getByCy("modal-yes-no").should("not.exist");cy.log("delete flow");invokeHeroDelete();cy.getByCy("button-yes").click();cy.getByCy("modal-yes-no").should("not.exist");cy.get("@log").should("have.been.calledWith","handleDeleteFromModal"); });});
Bu testi geçmek için, şimdilik sadece modalı kapatıp "handleDeleteFromModal" dizesini console.log yapan bir işleve ihtiyacımız var (Yeşil 9).
ListHeader alt bileşenini ( BrowserRouter'de sarılı) oluşturan bir test ekledik ve testin başarılı olması için boş özelliklere sahip bileşen ekledik (Kırmızı 1, Yeşil 1).
Konsol.log'ları kontrol eden testler ekledik, ekle ve yenile düğmelerine tıklayarak (Kırmızı 2).
Testlerin başarılı olması için işlev adını konsol.log yapan işlevler kullandık (Yeşil 2, Düzenleme 2).
Başka bir çocuk HeroList için başarısız bir test ekledik ve çocuk bileşeni çizime dahil ettik (Kırmızı 3, Yeşil 3). Şimdilik sabit veri kullandık.
Modalın görüntülenmesi için başarısız bir test ekledik (Kırmızı 4). Testin işlemesi için çoğunlukla boş dize özellikler ekledik (Yeşil 4).
Modalı kapatmayı sağlayan başka bir test yazdık ve func.apply is not a function hatası aldık, bu da olay işleyicimizin hiçbir şey yapmadığı anlamına geliyor. Bunu çözmek için olay işleyici işlevler için konsol.log kullandık ve onları yeniden düzenledik (Yeşil 5, Düzenleme 5).
Modal akışı için sahte bir test yazdık; ilk adımla modalın kapalı olması gerektiği (Kırmızı 6).
Testin başarılı olması için koşullu oluşturma için bir şablon olan false kodunu kullandık (Yeşil 6).
Akmaya yeni bir test ekledik; sil düğmesine tıkladığında modalın açılması gerektiğini kontrol etmek için (Kırmızı 7).
HeroList ve ModalYesNo çocuk bileşenlerinin durumu paylaştığını ve durumu, ebeveynleri olan Heroes bileşenine kaldırdığını fark ettik.
HeroList'in ardından bir handleDeleteHero özelliği oldu ve değerini () => setShowModal(true) ile ayarladık, silme tıklanarak modali göstermek için (Yeşil 7).
Daha önce olduğu gibi, tıklama işleyiciyi kendi işlevine çıkardık (Düzenleme 7).
Modal akışındaki bir sonraki testi yazdık, button-no'ya tıklanıldığında modal kaybolmalıdır (Kırmızı 8).
Tüm ihtiyacımız olan, mevcut konsol.log yerine handleCloseModal işlevinde setShowModal(false) kullanmaktı (Yeşil 8).
Modalın diğer dalı için; silme akışı için bir test ekledik (Kırmızı 9).
Modalı kapatıp handleDeleteFromModal adlı dizeyi konsol.loglayan bir işlev ekledik ve bunu modalın onYes işleyicisinde kullandık (Yeşil 9).
Çıkarılacak Dersler
Önceki bölümlerde defalarca gördüğümüz gibi, bileşeni tasarlarken ağ durumuyla ilgili kararları erteleyebilir ve sabit kodlu veriler kullanabilirsiniz. Olay işleyiciler için konsol.log yapan işlevler kullanabiliriz. Bu, durumu paylaşan bileşenlerin diğer bileşenlerde kullanılması durumunda yardımcı olacaktır. React belgelerine göre JSX kullanırken olay işleyici olarak bir işlevi string yerine kullanmalısınız.
Çocuk bileşenler için bileşen testlerine bakın, nasıl kullanıldıklarını görün ve ebeveyn bileşen için testler yazarken bu belgelerden faydalanın. Ebeveyn düzeyinde herhangi bir testi tekrarlamamıza gerek yoktur, ancak çocuğun nasıl monte edileceği hakkında fikir vermek için yardımcı olabilir.
Cypress bileşen testinde, func.apply is not a function hatası genellikle tıklama işleyicilerinin hiçbir şey yapmadığı anlamına gelir.
React'te, tüm tıklama işleyicileri işlev olmalıdır. React belgelerine göre JSX kullanırken olay işleyici olarak bir işlevi string yerine kullanmalısınız.
Çocuk bileşenleri izole bir şekilde test ederken (ör: ListHeader), tıklama olaylarını cy.stub ile taklit edebiliriz. Çocuk bileşen bir ebeveyn tarafından kullanılıyorsa, React açıkçası hiçbir şey taklit edemez. Bu nedenle ebeveyn/çocuğun tüketicisi, işleyici işlevini uygulamalı ve test etmelidir. Yine de, bileşen hakkında daha fazla bilgi edinilene kadar console.log kabul edilebilir.
Kent C. Dodds'dan:
Bileşenler durumu paylaşıyorsa, durumu en yakın ortak atalarına kaldırın.
Ortak atanın derinliği çok fazlaysa ve durumu kaldırmak prop-drilling'e yol açıyorsa, React'in context api'sini kullanın.
Bunun ötesinde, durum yönetimi kütüphaneleri kullanın.