1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/mirrors-ebiten

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
Клонировать/Скачать
image_test.go 110 КБ
Копировать Редактировать Web IDE Исходные данные Просмотреть построчно История
Hajime Hoshi Отправлено 19.02.2025 17:27 bcb5a0a
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810
// Copyright 2016 Hajime Hoshi
//
// 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.
package ebiten_test
import (
"bytes"
"fmt"
"image"
"image/color"
"image/draw"
_ "image/png"
"math"
"math/rand/v2"
"runtime"
"testing"
"github.com/hajimehoshi/ebiten/v2"
"github.com/hajimehoshi/ebiten/v2/examples/resources/images"
"github.com/hajimehoshi/ebiten/v2/internal/graphics"
t "github.com/hajimehoshi/ebiten/v2/internal/testing"
"github.com/hajimehoshi/ebiten/v2/internal/ui"
)
// maxImageSize is a maximum image size that should work in almost every environment.
const maxImageSize = 4096 - 2
func skipTooSlowTests(t *testing.T) bool {
if testing.Short() {
t.Skip("skipping test in short mode")
return true
}
if runtime.GOOS == "js" {
t.Skip("too slow or fragile on Wasm")
return true
}
return false
}
func TestMain(m *testing.M) {
ui.SetPanicOnErrorOnReadingPixelsForTesting(true)
t.MainWithRunLoop(m)
}
func openEbitenImage() (*ebiten.Image, image.Image, error) {
img, _, err := image.Decode(bytes.NewReader(images.Ebiten_png))
if err != nil {
return nil, nil, err
}
eimg := ebiten.NewImageFromImage(img)
return eimg, img, nil
}
func abs(x int) int {
if x < 0 {
return -x
}
return x
}
// sameColors compares c1 and c2 and returns a boolean value indicating
// if the two colors are (almost) same.
//
// Pixels read from GPU might include errors (#492), and
// sameColors considers such errors as delta.
func sameColors(c1, c2 color.RGBA, delta int) bool {
return abs(int(c1.R)-int(c2.R)) <= delta &&
abs(int(c1.G)-int(c2.G)) <= delta &&
abs(int(c1.B)-int(c2.B)) <= delta &&
abs(int(c1.A)-int(c2.A)) <= delta
}
func TestImagePixels(t *testing.T) {
img0, img, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
if got := img0.Bounds().Size(); got != img.Bounds().Size() {
t.Fatalf("img size: got %d; want %d", got, img.Bounds().Size())
}
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
// Check out of range part
w2, h2 := graphics.InternalImageSize(w), graphics.InternalImageSize(h)
for j := -100; j < h2+100; j++ {
for i := -100; i < w2+100; i++ {
got := img0.At(i, j)
want := color.RGBAModel.Convert(img.At(i, j))
if got != want {
t.Errorf("img0 At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
pix := make([]byte, 4*w*h)
img0.ReadPixels(pix)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (j*w + i)
got := color.RGBA{R: pix[idx], G: pix[idx+1], B: pix[idx+2], A: pix[idx+3]}
want := color.RGBAModel.Convert(img.At(i, j))
if got != want {
t.Errorf("(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
}
func TestImageComposition(t *testing.T) {
img2Color := color.NRGBA{R: 0x24, G: 0x3f, B: 0x6a, A: 0x88}
img3Color := color.NRGBA{R: 0x85, G: 0xa3, B: 0x08, A: 0xd3}
// TODO: Rename this to img0
img1, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := img1.Bounds().Dx(), img1.Bounds().Dy()
img2 := ebiten.NewImage(w, h)
img3 := ebiten.NewImage(w, h)
img2.Fill(img2Color)
img3.Fill(img3Color)
img_12_3 := ebiten.NewImage(w, h)
img2.DrawImage(img1, nil)
img3.DrawImage(img2, nil)
img_12_3.DrawImage(img3, nil)
img2.Fill(img2Color)
img3.Fill(img3Color)
img_1_23 := ebiten.NewImage(w, h)
img3.DrawImage(img2, nil)
img3.DrawImage(img1, nil)
img_1_23.DrawImage(img3, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
c1 := img_12_3.At(i, j).(color.RGBA)
c2 := img_1_23.At(i, j).(color.RGBA)
if !sameColors(c1, c2, 1) {
t.Errorf("img_12_3.At(%d, %d) = %v; img_1_23.At(%[1]d, %[2]d) = %#[4]v", i, j, c1, c2)
}
if c1.A == 0 {
t.Fatalf("img_12_3.At(%d, %d).A = 0; nothing is rendered?", i, j)
}
if c2.A == 0 {
t.Fatalf("img_1_23.At(%d, %d).A = 0; nothing is rendered?", i, j)
}
}
}
}
func TestImageSelf(t *testing.T) {
// Note that mutex usages: without defer, unlocking is not called when panicking.
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawImage must panic but not")
}
}()
img, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
img.DrawImage(img, nil)
}
func TestImageScale(t *testing.T) {
for _, scale := range []int{2, 3, 4} {
img0, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
img1 := ebiten.NewImage(w*scale, h*scale)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(float64(scale), float64(scale))
img1.DrawImage(img0, op)
for j := 0; j < h*scale; j++ {
for i := 0; i < w*scale; i++ {
c0 := img0.At(i/scale, j/scale).(color.RGBA)
c1 := img1.At(i, j).(color.RGBA)
if c0 != c1 {
t.Fatalf("img0.At(%[1]d, %[2]d) should equal to img1.At(%[3]d, %[4]d) (with scale %[5]d) but not: %[6]v vs %[7]v", i/2, j/2, i, j, scale, c0, c1)
}
}
}
}
}
func TestImage90DegreeRotate(t *testing.T) {
img0, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
img1 := ebiten.NewImage(h, w)
op := &ebiten.DrawImageOptions{}
op.GeoM.Rotate(math.Pi / 2)
op.GeoM.Translate(float64(h), 0)
img1.DrawImage(img0, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
c0 := img0.At(i, j).(color.RGBA)
c1 := img1.At(h-j-1, i).(color.RGBA)
if c0 != c1 {
t.Errorf("img0.At(%[1]d, %[2]d) should equal to img1.At(%[3]d, %[4]d) but not: %[5]v vs %[6]v", i, j, h-j-1, i, c0, c1)
}
}
}
}
func TestImageDotByDotInversion(t *testing.T) {
img0, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
img1 := ebiten.NewImage(w, h)
op := &ebiten.DrawImageOptions{}
op.GeoM.Rotate(math.Pi)
op.GeoM.Translate(float64(w), float64(h))
img1.DrawImage(img0, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
c0 := img0.At(i, j).(color.RGBA)
c1 := img1.At(w-i-1, h-j-1).(color.RGBA)
if c0 != c1 {
t.Errorf("img0.At(%[1]d, %[2]d) should equal to img1.At(%[3]d, %[4]d) but not: %[5]v vs %[6]v", i, j, w-i-1, h-j-1, c0, c1)
}
}
}
}
func TestImageWritePixels(t *testing.T) {
// Create a dummy image so that the shared texture is used and origImg's position is shifted.
dummyImg := ebiten.NewImageFromImage(image.NewRGBA(image.Rect(0, 0, 16, 16)))
defer dummyImg.Deallocate()
_, origImg, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
// Convert to *image.RGBA just in case.
img := image.NewRGBA(origImg.Bounds())
draw.Draw(img, img.Bounds(), origImg, image.ZP, draw.Src)
size := img.Bounds().Size()
img0 := ebiten.NewImage(size.X, size.Y)
img0.WritePixels(img.Pix)
for j := 0; j < img0.Bounds().Dy(); j++ {
for i := 0; i < img0.Bounds().Dx(); i++ {
got := img0.At(i, j)
want := img.At(i, j)
if got != want {
t.Errorf("img0 At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
p := make([]uint8, 4*size.X*size.Y)
for i := range p {
p[i] = 0x80
}
img0.WritePixels(p)
// Even if p is changed after calling ReplacePixel, img0 uses the original values.
for i := range p {
p[i] = 0
}
for j := 0; j < img0.Bounds().Dy(); j++ {
for i := 0; i < img0.Bounds().Dx(); i++ {
got := img0.At(i, j)
want := color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0x80}
if got != want {
t.Errorf("img0 At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
}
func TestImageWritePixelsNil(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("WritePixels(nil) must panic")
}
}()
img := ebiten.NewImage(16, 16)
img.Fill(color.White)
img.WritePixels(nil)
}
func TestImageDispose(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Fill(color.White)
img.Dispose()
// The color is transparent (color.RGBA{}).
// Note that the value's type must be color.RGBA.
got := img.At(0, 0)
want := color.RGBA{}
if got != want {
t.Errorf("img.At(0, 0) got: %v, want: %v", got, want)
}
}
func TestImageDeallocate(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Fill(color.White)
img.Deallocate()
// The color is transparent (color.RGBA{}).
got := img.At(0, 0)
want := color.RGBA{}
if got != want {
t.Errorf("img.At(0, 0) got: %v, want: %v", got, want)
}
}
func TestImageBlendLighter(t *testing.T) {
img0, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
img1 := ebiten.NewImage(w, h)
img1.Fill(color.RGBA{R: 0x01, G: 0x02, B: 0x03, A: 0x04})
op := &ebiten.DrawImageOptions{}
op.Blend = ebiten.BlendLighter
img1.DrawImage(img0, op)
for j := 0; j < img1.Bounds().Dy(); j++ {
for i := 0; i < img1.Bounds().Dx(); i++ {
got := img1.At(i, j).(color.RGBA)
want := img0.At(i, j).(color.RGBA)
want.R = uint8(min(0xff, int(want.R)+1))
want.G = uint8(min(0xff, int(want.G)+2))
want.B = uint8(min(0xff, int(want.B)+3))
want.A = uint8(min(0xff, int(want.A)+4))
if got != want {
t.Errorf("img1 At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
}
func TestNewImageFromEbitenImage(t *testing.T) {
img, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
_ = ebiten.NewImageFromImage(img)
}
func TestNewImageFromSubImage(t *testing.T) {
_, img, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := img.Bounds().Dx(), img.Bounds().Dy()
subImg := img.(*image.NRGBA).SubImage(image.Rect(1, 1, w-1, h-1))
eimg := ebiten.NewImageFromImage(subImg)
sw, sh := subImg.Bounds().Dx(), subImg.Bounds().Dy()
w2, h2 := eimg.Bounds().Dx(), eimg.Bounds().Dy()
if w2 != sw {
t.Errorf("eimg Width: got %v; want %v", w2, sw)
}
if h2 != sh {
t.Errorf("eimg Width: got %v; want %v", h2, sh)
}
for j := 0; j < h2; j++ {
for i := 0; i < w2; i++ {
got := eimg.At(i, j)
want := color.RGBAModel.Convert(img.At(i+1, j+1))
if got != want {
t.Errorf("img0 At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
}
type mutableRGBA struct {
r, g, b, a uint8
}
func (c *mutableRGBA) RGBA() (r, g, b, a uint32) {
return uint32(c.r) * 0x101, uint32(c.g) * 0x101, uint32(c.b) * 0x101, uint32(c.a) * 0x101
}
func TestImageFill(t *testing.T) {
w, h := 10, 10
img := ebiten.NewImage(w, h)
clr := &mutableRGBA{0x80, 0x80, 0x80, 0x80}
img.Fill(clr)
clr.r = 0
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0x80}
if got != want {
t.Errorf("img At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
}
// Issue #740
func TestImageClear(t *testing.T) {
const w, h = 128, 256
img := ebiten.NewImage(w, h)
img.Fill(color.White)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if got != want {
t.Errorf("img At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
img.Clear()
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img.At(i, j)
want := color.RGBA{}
if got != want {
t.Errorf("img At(%d, %d): got %v; want %v", i, j, got, want)
}
}
}
}
// Issue #317, #558, #724
func TestImageEdge(t *testing.T) {
// TODO: This test is not so meaningful after #1218. Do we remove this?
if skipTooSlowTests(t) {
return
}
const (
img0Width = 10
img0Height = 10
img0InnerWidth = 10
img0InnerHeight = 10
img1Width = 32
img1Height = 32
)
img0 := ebiten.NewImage(img0Width, img0Height)
pixels := make([]uint8, 4*img0Width*img0Height)
for j := 0; j < img0Height; j++ {
for i := 0; i < img0Width; i++ {
idx := 4 * (i + j*img0Width)
pixels[idx] = 0xff
pixels[idx+1] = 0
pixels[idx+2] = 0
pixels[idx+3] = 0xff
}
}
img0.WritePixels(pixels)
img1 := ebiten.NewImage(img1Width, img1Height)
red := color.RGBA{R: 0xff, A: 0xff}
transparent := color.RGBA{}
angles := []float64{}
for a := 0; a < 1440; a++ {
angles = append(angles, float64(a)/1440*2*math.Pi)
}
for a := 0; a < 4096; a += 3 {
// a++ should be fine, but it takes long to test.
angles = append(angles, float64(a)/4096*2*math.Pi)
}
for _, s := range []float64{1, 0.5, 0.25} {
for _, f := range []ebiten.Filter{ebiten.FilterNearest, ebiten.FilterLinear} {
for _, a := range angles {
for _, testDrawTriangles := range []bool{false, true} {
img1.Clear()
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
b := img0.Bounds()
var geo ebiten.GeoM
geo.Translate(-float64(w)/2, -float64(h)/2)
geo.Scale(s, s)
geo.Rotate(a)
geo.Translate(img1Width/2, img1Height/2)
if !testDrawTriangles {
op := &ebiten.DrawImageOptions{}
op.GeoM = geo
op.Filter = f
img1.DrawImage(img0, op)
} else {
op := &ebiten.DrawTrianglesOptions{}
dx0, dy0 := geo.Apply(0, 0)
dx1, dy1 := geo.Apply(float64(w), 0)
dx2, dy2 := geo.Apply(0, float64(h))
dx3, dy3 := geo.Apply(float64(w), float64(h))
vs := []ebiten.Vertex{
{
DstX: float32(dx0),
DstY: float32(dy0),
SrcX: float32(b.Min.X),
SrcY: float32(b.Min.Y),
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: float32(dx1),
DstY: float32(dy1),
SrcX: float32(b.Max.X),
SrcY: float32(b.Min.Y),
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: float32(dx2),
DstY: float32(dy2),
SrcX: float32(b.Min.X),
SrcY: float32(b.Max.Y),
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: float32(dx3),
DstY: float32(dy3),
SrcX: float32(b.Max.X),
SrcY: float32(b.Max.Y),
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
op.Filter = f
img1.DrawTriangles(vs, is, img0, op)
}
allTransparent := true
for j := 0; j < img1Height; j++ {
for i := 0; i < img1Width; i++ {
c := img1.At(i, j)
if c == transparent {
continue
}
allTransparent = false
switch f {
case ebiten.FilterNearest:
if c == red {
continue
}
case ebiten.FilterLinear:
if _, g, b, _ := c.RGBA(); g == 0 && b == 0 {
continue
}
}
t.Fatalf("img1.At(%d, %d) (filter: %d, scale: %f, angle: %f, draw-triangles?: %t) want: red or transparent, got: %v", i, j, f, s, a, testDrawTriangles, c)
}
}
if allTransparent {
t.Fatalf("img1 (filter: %d, scale: %f, angle: %f, draw-triangles?: %t) is transparent but should not", f, s, a, testDrawTriangles)
}
}
}
}
}
}
// Issue #419
func TestImageTooManyFill(t *testing.T) {
const width = 1024
indexToColor := func(index int) uint8 {
return uint8((17*index + 0x40) % 256)
}
src := ebiten.NewImage(1, 1)
dst := ebiten.NewImage(width, 1)
for i := 0; i < width; i++ {
c := indexToColor(i)
src.Fill(color.RGBA{R: c, G: c, B: c, A: 0xff})
op := &ebiten.DrawImageOptions{}
op.GeoM.Translate(float64(i), 0)
dst.DrawImage(src, op)
}
for i := 0; i < width; i++ {
c := indexToColor(i)
got := dst.At(i, 0).(color.RGBA)
want := color.RGBA{R: c, G: c, B: c, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i, 0, got, want)
}
}
}
func BenchmarkDrawImage(b *testing.B) {
img0 := ebiten.NewImage(16, 16)
img1 := ebiten.NewImage(16, 16)
op := &ebiten.DrawImageOptions{}
for i := 0; i < b.N; i++ {
img0.DrawImage(img1, op)
}
}
func BenchmarkDrawTriangles(b *testing.B) {
const w, h = 16, 16
img0 := ebiten.NewImage(w, h)
img1 := ebiten.NewImage(w, h)
op := &ebiten.DrawTrianglesOptions{}
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
for i := 0; i < b.N; i++ {
img0.DrawTriangles(vs, is, img1, op)
}
}
func TestImageLinearGraduation(t *testing.T) {
img0 := ebiten.NewImage(2, 2)
img0.WritePixels([]byte{
0xff, 0x00, 0x00, 0xff,
0x00, 0xff, 0x00, 0xff,
0x00, 0x00, 0xff, 0xff,
0xff, 0xff, 0xff, 0xff,
})
const w, h = 32, 32
img1 := ebiten.NewImage(w, h)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(w, h)
op.GeoM.Translate(-w/4, -h/4)
op.Filter = ebiten.FilterLinear
img1.DrawImage(img0, op)
for j := 1; j < h-1; j++ {
for i := 1; i < w-1; i++ {
c := img1.At(i, j).(color.RGBA)
if c.R == 0 || c.R == 0xff {
t.Errorf("img1.At(%d, %d).R must be in between 0x01 and 0xfe but %v", i, j, c)
}
}
}
}
func TestImageOutside(t *testing.T) {
src := ebiten.NewImage(5, 10) // internal texture size is 8x16.
dst := ebiten.NewImage(4, 4)
src.Fill(color.RGBA{R: 0xff, A: 0xff})
cases := []struct {
X, Y, Width, Height int
}{
{-4, -4, 4, 4},
{5, 0, 4, 4},
{0, 10, 4, 4},
{5, 10, 4, 4},
{8, 0, 4, 4},
{0, 16, 4, 4},
{8, 16, 4, 4},
{8, -4, 4, 4},
{-4, 16, 4, 4},
{5, 10, 0, 0},
{5, 10, -2, -2}, // non-well-formed rectangle
}
for _, c := range cases {
dst.Clear()
op := &ebiten.DrawImageOptions{}
op.GeoM.Translate(0, 0)
dst.DrawImage(src.SubImage(image.Rectangle{
Min: image.Pt(c.X, c.Y),
Max: image.Pt(c.X+c.Width, c.Y+c.Height),
}).(*ebiten.Image), op)
for j := 0; j < 4; j++ {
for i := 0; i < 4; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{}
if got != want {
t.Errorf("src(x: %d, y: %d, w: %d, h: %d), dst At(%d, %d): got %v, want: %v", c.X, c.Y, c.Width, c.Height, i, j, got, want)
}
}
}
}
}
func TestImageOutsideUpperLeft(t *testing.T) {
src := ebiten.NewImage(4, 4)
dst1 := ebiten.NewImage(16, 16)
dst2 := ebiten.NewImage(16, 16)
src.Fill(color.RGBA{R: 0xff, A: 0xff})
op := &ebiten.DrawImageOptions{}
op.GeoM.Rotate(math.Pi / 4)
dst1.DrawImage(src.SubImage(image.Rect(-4, -4, 8, 8)).(*ebiten.Image), op)
op = &ebiten.DrawImageOptions{}
op.GeoM.Rotate(math.Pi / 4)
dst2.DrawImage(src, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst1.At(i, j).(color.RGBA)
want := dst2.At(i, j).(color.RGBA)
if got != want {
t.Errorf("got: dst1.At(%d, %d): %v, want: dst2.At(%d, %d): %v", i, j, got, i, j, want)
}
}
}
}
func TestImageSize(t *testing.T) {
const (
w = 17
h = 31
)
img := ebiten.NewImage(w, h)
gotW, gotH := img.Bounds().Dx(), img.Bounds().Dy()
if gotW != w {
t.Errorf("got: %d, want: %d", gotW, w)
}
if gotH != h {
t.Errorf("got: %d, want: %d", gotH, h)
}
}
func TestImageSize1(t *testing.T) {
src := ebiten.NewImage(1, 1)
dst := ebiten.NewImage(1, 1)
src.Fill(color.White)
dst.DrawImage(src, nil)
got := src.At(0, 0).(color.RGBA)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("got: %v, want: %v", got, want)
}
}
// TODO: Enable this test again. This test fails after #1217 is fixed.
func Skip_TestImageSize4096(t *testing.T) {
src := ebiten.NewImage(4096, 4096)
dst := ebiten.NewImage(4096, 4096)
pix := make([]byte, 4096*4096*4)
for i := 0; i < 4096; i++ {
j := 4095
idx := 4 * (i + j*4096)
pix[idx] = uint8(i + j)
pix[idx+1] = uint8((i + j) >> 8)
pix[idx+2] = uint8((i + j) >> 16)
pix[idx+3] = 0xff
}
for j := 0; j < 4096; j++ {
i := 4095
idx := 4 * (i + j*4096)
pix[idx] = uint8(i + j)
pix[idx+1] = uint8((i + j) >> 8)
pix[idx+2] = uint8((i + j) >> 16)
pix[idx+3] = 0xff
}
src.WritePixels(pix)
dst.DrawImage(src, nil)
for i := 4095; i < 4096; i++ {
j := 4095
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: uint8(i + j), G: uint8((i + j) >> 8), B: uint8((i + j) >> 16), A: 0xff}
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
for j := 4095; j < 4096; j++ {
i := 4095
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: uint8(i + j), G: uint8((i + j) >> 8), B: uint8((i + j) >> 16), A: 0xff}
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
func TestImageCopy(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("copying image and using it must panic")
}
}()
img0 := ebiten.NewImage(256, 256)
img1 := *img0
img1.Fill(color.Transparent)
}
// Issue #611, #907
func TestImageStretch(t *testing.T) {
if skipTooSlowTests(t) {
return
}
const w = 16
dst := ebiten.NewImage(w, maxImageSize)
loop:
for h := 1; h <= 32; h++ {
src := ebiten.NewImage(w+2, h+2)
pix := make([]byte, 4*(w+2)*(h+2))
for i := 0; i < (w+2)*(h+2); i++ {
pix[4*i] = 0xff
pix[4*i+3] = 0xff
}
src.WritePixels(pix)
dh := dst.Bounds().Dy()
for i := 0; i < dh; {
dst.Clear()
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(1, float64(i)/float64(h))
dst.DrawImage(src.SubImage(image.Rect(1, 1, w+1, h+1)).(*ebiten.Image), op)
for j := -1; j <= 1; j++ {
if i+j < 0 {
continue
}
got := dst.At(0, i+j).(color.RGBA)
want := color.RGBA{}
if j < 0 {
want = color.RGBA{R: 0xff, A: 0xff}
}
if got != want {
t.Errorf("At(%d, %d) (height=%d, scale=%d/%d): got: %v, want: %v", 0, i+j, h, i, h, got, want)
continue loop
}
}
switch i % 32 {
case 31, 0:
i++
case 1:
i += 32 - 2
default:
panic("not reached")
}
}
}
}
func TestImageSprites(t *testing.T) {
const (
width = 512
height = 512
)
src := ebiten.NewImage(4, 4)
src.Fill(color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff})
dst := ebiten.NewImage(width, height)
for j := 0; j < height/4; j++ {
for i := 0; i < width/4; i++ {
op := &ebiten.DrawImageOptions{}
op.GeoM.Translate(float64(i*4), float64(j*4))
dst.DrawImage(src, op)
}
}
for j := 0; j < height/4; j++ {
for i := 0; i < width/4; i++ {
got := dst.At(i*4, j*4).(color.RGBA)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i*4, j*4, got, want)
}
}
}
}
// Disabled: it does not make sense to expect deterministic mipmap results (#909).
func Disabled_TestImageMipmap(t *testing.T) {
src, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := src.Bounds().Dx(), src.Bounds().Dy()
l1 := ebiten.NewImage(w/2, h/2)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(1/2.0, 1/2.0)
op.Filter = ebiten.FilterLinear
l1.DrawImage(src, op)
l1w, l1h := l1.Bounds().Dx(), l1.Bounds().Dy()
l2 := ebiten.NewImage(l1w/2, l1h/2)
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(1/2.0, 1/2.0)
op.Filter = ebiten.FilterLinear
l2.DrawImage(l1, op)
gotDst := ebiten.NewImage(w, h)
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(1/5.0, 1/5.0)
op.Filter = ebiten.FilterLinear
gotDst.DrawImage(src, op)
wantDst := ebiten.NewImage(w, h)
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(4.0/5.0, 4.0/5.0)
op.Filter = ebiten.FilterLinear
wantDst.DrawImage(l2, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := gotDst.At(i, j).(color.RGBA)
want := wantDst.At(i, j).(color.RGBA)
if !sameColors(got, want, 1) {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Disabled: it does not make sense to expect deterministic mipmap results (#909).
func Disabled_TestImageMipmapNegativeDet(t *testing.T) {
src, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := src.Bounds().Dx(), src.Bounds().Dy()
l1 := ebiten.NewImage(w/2, h/2)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(1/2.0, 1/2.0)
op.Filter = ebiten.FilterLinear
l1.DrawImage(src, op)
l1w, l1h := l1.Bounds().Dx(), l1.Bounds().Dy()
l2 := ebiten.NewImage(l1w/2, l1h/2)
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(1/2.0, 1/2.0)
op.Filter = ebiten.FilterLinear
l2.DrawImage(l1, op)
gotDst := ebiten.NewImage(w, h)
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(-1/5.0, -1/5.0)
op.GeoM.Translate(float64(w), float64(h))
op.Filter = ebiten.FilterLinear
gotDst.DrawImage(src, op)
wantDst := ebiten.NewImage(w, h)
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(-4.0/5.0, -4.0/5.0)
op.GeoM.Translate(float64(w), float64(h))
op.Filter = ebiten.FilterLinear
wantDst.DrawImage(l2, op)
allZero := true
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := gotDst.At(i, j).(color.RGBA)
want := wantDst.At(i, j).(color.RGBA)
if !sameColors(got, want, 1) {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
if got.A > 0 {
allZero = false
}
}
}
if allZero {
t.Errorf("the image must include non-zero values but not")
}
}
// Issue #710
func TestImageMipmapColor(t *testing.T) {
img0 := ebiten.NewImage(256, 256)
img1 := ebiten.NewImage(128, 128)
img1.Fill(color.White)
for i := 0; i < 8; i++ {
img0.Clear()
s := 1 - float64(i)/8
op := &ebiten.DrawImageOptions{}
op.Filter = ebiten.FilterLinear
op.GeoM.Scale(s, s)
op.ColorScale.Scale(1, 1, 0, 1)
img0.DrawImage(img1, op)
op.GeoM.Translate(128, 0)
op.ColorScale.Reset()
op.ColorScale.Scale(0, 1, 1, 1)
img0.DrawImage(img1, op)
want := color.RGBA{G: 0xff, B: 0xff, A: 0xff}
got := img0.At(128, 0)
if got != want {
t.Errorf("want: %v, got: %v", want, got)
}
}
}
// Issue #725
func TestImageMiamapAndDrawTriangle(t *testing.T) {
img0 := ebiten.NewImage(32, 32)
img1 := ebiten.NewImage(128, 128)
img2 := ebiten.NewImage(128, 128)
// Fill img1 red and create img1's mipmap
img1.Fill(color.RGBA{R: 0xff, A: 0xff})
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(0.25, 0.25)
op.Filter = ebiten.FilterLinear
img0.DrawImage(img1, op)
// Call DrawTriangle on img1 and fill it with green
img2.Fill(color.RGBA{G: 0xff, A: 0xff})
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 128,
DstY: 0,
SrcX: 128,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: 128,
SrcX: 0,
SrcY: 128,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 128,
DstY: 128,
SrcX: 128,
SrcY: 128,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
img1.DrawTriangles(vs, []uint16{0, 1, 2, 1, 2, 3}, img2, nil)
// Draw img1 (green) again. Confirm mipmap is correctly updated.
img0.Clear()
op = &ebiten.DrawImageOptions{}
op.GeoM.Scale(0.25, 0.25)
op.Filter = ebiten.FilterLinear
img0.DrawImage(img1, op)
w, h := img0.Bounds().Dx(), img0.Bounds().Dy()
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
c := img0.At(i, j).(color.RGBA)
if c.R != 0 {
t.Errorf("img0.At(%d, %d): red want %d got %d", i, j, 0, c.R)
}
}
}
}
func TestImageSubImageAt(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Fill(color.RGBA{R: 0xff, A: 0xff})
got := img.SubImage(image.Rect(1, 1, 16, 16)).At(0, 0).(color.RGBA)
want := color.RGBA{}
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
got = img.SubImage(image.Rect(1, 1, 16, 16)).At(1, 1).(color.RGBA)
want = color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageSubImageSize(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Fill(color.RGBA{R: 0xff, A: 0xff})
got := img.SubImage(image.Rect(1, 1, 16, 16)).Bounds().Dx()
want := 15
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageDrawImmediately(t *testing.T) {
const w, h = 16, 16
img0 := ebiten.NewImage(w, h)
img1 := ebiten.NewImage(w, h)
// Do not manipulate img0 here.
img0.Fill(color.RGBA{R: 0xff, A: 0xff})
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img0.At(i, j).(color.RGBA)
want := color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("img0.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
img0.DrawImage(img1, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img0.At(i, j).(color.RGBA)
want := color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("img0.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #669, #759
func TestImageLinearFilterGlitch(t *testing.T) {
const w, h = 200, 12
const scale = 1.2
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(int(math.Floor(w*scale)), h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := i + w*j
if j < 3 {
pix[4*idx] = 0xff
pix[4*idx+1] = 0xff
pix[4*idx+2] = 0xff
pix[4*idx+3] = 0xff
} else {
pix[4*idx] = 0
pix[4*idx+1] = 0
pix[4*idx+2] = 0
pix[4*idx+3] = 0xff
}
}
}
src.WritePixels(pix)
for _, f := range []ebiten.Filter{ebiten.FilterNearest, ebiten.FilterLinear} {
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(scale, 1)
op.Filter = f
dst.DrawImage(src, op)
for j := 1; j < h-1; j++ {
offset := int(math.Ceil(scale))
for i := offset; i < int(math.Floor(w*scale))-offset; i++ {
got := dst.At(i, j).(color.RGBA)
var want color.RGBA
if j < 3 {
want = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
} else {
want = color.RGBA{A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): filter: %d, got: %v, want: %v", i, j, f, got, want)
}
}
}
}
}
// Issue #1212
func TestImageLinearFilterGlitch2(t *testing.T) {
const w, h = 100, 100
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
idx := 0
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if i+j < 100 {
pix[4*idx] = 0
pix[4*idx+1] = 0
pix[4*idx+2] = 0
pix[4*idx+3] = 0xff
} else {
pix[4*idx] = 0xff
pix[4*idx+1] = 0xff
pix[4*idx+2] = 0xff
pix[4*idx+3] = 0xff
}
idx++
}
}
src.WritePixels(pix)
op := &ebiten.DrawImageOptions{}
op.Filter = ebiten.FilterLinear
dst.DrawImage(src, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
var want color.RGBA
if i+j < 100 {
want = color.RGBA{A: 0xff}
} else {
want = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageAddressRepeat(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
if 4 <= i && i < 8 && 4 <= j && j < 8 {
pix[idx] = byte(i-4) * 0x10
pix[idx+1] = byte(j-4) * 0x10
pix[idx+2] = 0
pix[idx+3] = 0xff
} else {
pix[idx] = 0
pix[idx+1] = 0
pix[idx+2] = 0xff
pix[idx+3] = 0xff
}
}
}
src.WritePixels(pix)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
op := &ebiten.DrawTrianglesOptions{}
op.Address = ebiten.AddressRepeat
dst.DrawTriangles(vs, is, src.SubImage(image.Rect(4, 4, 8, 8)).(*ebiten.Image), op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: byte(i%4) * 0x10, G: byte(j%4) * 0x10, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageAddressRepeatNegativePosition(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
if 4 <= i && i < 8 && 4 <= j && j < 8 {
pix[idx] = byte(i-4) * 0x10
pix[idx+1] = byte(j-4) * 0x10
pix[idx+2] = 0
pix[idx+3] = 0xff
} else {
pix[idx] = 0
pix[idx+1] = 0
pix[idx+2] = 0xff
pix[idx+3] = 0xff
}
}
}
src.WritePixels(pix)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: -w,
SrcY: -h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: 0,
SrcY: -h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: -w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
op := &ebiten.DrawTrianglesOptions{}
op.Address = ebiten.AddressRepeat
dst.DrawTriangles(vs, is, src.SubImage(image.Rect(4, 4, 8, 8)).(*ebiten.Image), op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: byte(i%4) * 0x10, G: byte(j%4) * 0x10, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageWritePixelsAfterClear(t *testing.T) {
const w, h = 256, 256
img := ebiten.NewImage(w, h)
img.WritePixels(make([]byte, 4*w*h))
// Clear used to call DrawImage to clear the image, which was the cause of crash. It is because after
// DrawImage is called, WritePixels for a region is forbidden.
//
// Now WritePixels was always called at Clear instead.
img.Clear()
img.WritePixels(make([]byte, 4*w*h))
// The test passes if this doesn't crash.
}
func TestImageSet(t *testing.T) {
type Pt struct {
X, Y int
}
const w, h = 16, 16
img := ebiten.NewImage(w, h)
colors := map[Pt]color.RGBA{
{1, 2}: {3, 4, 5, 6},
{7, 8}: {9, 10, 11, 12},
{13, 14}: {15, 16, 17, 18},
{-1, -1}: {19, 20, 21, 22},
}
for p, c := range colors {
img.Set(p.X, p.Y, c)
}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img.At(i, j).(color.RGBA)
var want color.RGBA
if c, ok := colors[Pt{i, j}]; ok {
want = c
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageSetAndDraw(t *testing.T) {
type Pt struct {
X, Y int
}
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
colors := map[Pt]color.RGBA{
{1, 2}: {3, 4, 5, 6},
{7, 8}: {9, 10, 11, 12},
{13, 14}: {15, 16, 17, 18},
}
for p, c := range colors {
src.Set(p.X, p.Y, c)
dst.Set(p.X+1, p.Y+1, c)
}
dst.DrawImage(src, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
var want color.RGBA
if c, ok := colors[Pt{i, j}]; ok {
want = c
}
if c, ok := colors[Pt{i - 1, j - 1}]; ok {
want = c
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
src.Clear()
dst.Clear()
for p, c := range colors {
src.Set(p.X, p.Y, c)
dst.Set(p.X+1, p.Y+1, c)
}
op := &ebiten.DrawImageOptions{}
op.GeoM.Translate(2, 2)
dst.DrawImage(src.SubImage(image.Rect(2, 2, w-2, h-2)).(*ebiten.Image), op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
var want color.RGBA
if 2 <= i && 2 <= j && i < w-2 && j < h-2 {
if c, ok := colors[Pt{i, j}]; ok {
want = c
}
}
if c, ok := colors[Pt{i - 1, j - 1}]; ok {
want = c
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageAlphaOnBlack(t *testing.T) {
const w, h = 16, 16
src0 := ebiten.NewImage(w, h)
src1 := ebiten.NewImage(w, h)
dst0 := ebiten.NewImage(w, h)
dst1 := ebiten.NewImage(w, h)
pix0 := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if (i/3)%2 == (j/3)%2 {
pix0[4*(i+j*w)] = 0xff
pix0[4*(i+j*w)+1] = 0xff
pix0[4*(i+j*w)+2] = 0xff
pix0[4*(i+j*w)+3] = 0xff
}
}
}
src0.WritePixels(pix0)
pix1 := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if (i/3)%2 == (j/3)%2 {
pix1[4*(i+j*w)] = 0xff
pix1[4*(i+j*w)+1] = 0xff
pix1[4*(i+j*w)+2] = 0xff
pix1[4*(i+j*w)+3] = 0xff
} else {
pix1[4*(i+j*w)] = 0
pix1[4*(i+j*w)+1] = 0
pix1[4*(i+j*w)+2] = 0
pix1[4*(i+j*w)+3] = 0xff
}
}
}
src1.WritePixels(pix1)
dst0.Fill(color.Black)
dst1.Fill(color.Black)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(0.5, 0.5)
op.Filter = ebiten.FilterLinear
dst0.DrawImage(src0, op)
dst1.DrawImage(src1, op)
gray := false
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst0.At(i, j)
want := dst1.At(i, j)
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
if r := got.(color.RGBA).R; 0 < r && r < 255 {
gray = true
}
}
}
if !gray {
t.Errorf("gray must be included in the results but not")
}
}
func TestImageDrawTrianglesWithSubImage(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if 4 <= i && i < 8 && 4 <= j && j < 8 {
pix[4*(i+j*w)] = 0xff
pix[4*(i+j*w)+1] = 0
pix[4*(i+j*w)+2] = 0
pix[4*(i+j*w)+3] = 0xff
} else {
pix[4*(i+j*w)] = 0
pix[4*(i+j*w)+1] = 0xff
pix[4*(i+j*w)+2] = 0
pix[4*(i+j*w)+3] = 0xff
}
}
}
src.WritePixels(pix)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
op := &ebiten.DrawTrianglesOptions{}
op.Address = ebiten.AddressClampToZero
dst.DrawTriangles(vs, is, src.SubImage(image.Rect(4, 4, 8, 8)).(*ebiten.Image), op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
var want color.RGBA
if 4 <= i && i < 8 && 4 <= j && j < 8 {
want = src.At(i, j).(color.RGBA)
}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #823
func TestImageAtAfterDisposingSubImage(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Set(0, 0, color.White)
img.SubImage(image.Rect(0, 0, 16, 16))
runtime.GC()
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
want64 := color.RGBA64{R: 0xffff, G: 0xffff, B: 0xffff, A: 0xffff}
got := img.At(0, 0)
if got != want {
t.Errorf("At(0,0) got: %v, want: %v", got, want)
}
got = img.RGBA64At(0, 0)
if got != want64 {
t.Errorf("RGBA64At(0,0) got: %v, want: %v", got, want)
}
img.Set(0, 1, color.White)
sub := img.SubImage(image.Rect(0, 0, 16, 16)).(*ebiten.Image)
sub.Dispose()
got = img.At(0, 1)
if got != want {
t.Errorf("At(0,1) got: %v, want: %v", got, want64)
}
got = img.RGBA64At(0, 1)
if got != want64 {
t.Errorf("RGBA64At(0,1) got: %v, want: %v", got, want64)
}
}
func TestImageAtAfterDeallocateSubImage(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Set(0, 0, color.White)
img.SubImage(image.Rect(0, 0, 16, 16))
runtime.GC()
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
want64 := color.RGBA64{R: 0xffff, G: 0xffff, B: 0xffff, A: 0xffff}
got := img.At(0, 0)
if got != want {
t.Errorf("At(0,0) got: %v, want: %v", got, want)
}
got = img.RGBA64At(0, 0)
if got != want64 {
t.Errorf("RGBA64At(0,0) got: %v, want: %v", got, want)
}
img.Set(0, 1, color.White)
sub := img.SubImage(image.Rect(0, 0, 16, 16)).(*ebiten.Image)
sub.Deallocate()
got = img.At(0, 1)
if got != want {
t.Errorf("At(0,1) got: %v, want: %v", got, want64)
}
got = img.RGBA64At(0, 1)
if got != want64 {
t.Errorf("RGBA64At(0,1) got: %v, want: %v", got, want64)
}
}
func TestImageSubImageSubImage(t *testing.T) {
img := ebiten.NewImage(16, 16)
img.Fill(color.White)
sub0 := img.SubImage(image.Rect(0, 0, 12, 12)).(*ebiten.Image)
sub1 := sub0.SubImage(image.Rect(4, 4, 16, 16)).(*ebiten.Image)
cases := []struct {
X int
Y int
Color color.RGBA
}{
{
X: 0,
Y: 0,
Color: color.RGBA{},
},
{
X: 4,
Y: 4,
Color: color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff},
},
{
X: 15,
Y: 15,
Color: color.RGBA{},
},
}
for _, c := range cases {
got := sub1.At(c.X, c.Y)
want := c.Color
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", c.X, c.Y, got, want)
}
}
}
// Issue #839
func TestImageTooSmallMipmap(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
src.Fill(color.White)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(1, 0.24)
op.Filter = ebiten.FilterLinear
dst.DrawImage(src.SubImage(image.Rect(5, 0, 6, 16)).(*ebiten.Image), op)
got := dst.At(0, 0).(color.RGBA)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageZeroSizedMipmap(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
op := &ebiten.DrawImageOptions{}
op.Filter = ebiten.FilterLinear
dst.DrawImage(src.SubImage(image.ZR).(*ebiten.Image), op)
}
// Issue #898
func TestImageFillingAndEdges(t *testing.T) {
const (
srcw, srch = 16, 16
dstw, dsth = 256, 16
)
src := ebiten.NewImage(srcw, srch)
dst := ebiten.NewImage(dstw, dsth)
src.Fill(color.White)
dst.Fill(color.Black)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(float64(dstw-2)/float64(srcw), float64(dsth-2)/float64(srch))
op.GeoM.Translate(1, 1)
dst.DrawImage(src, op)
for j := 0; j < dsth; j++ {
for i := 0; i < dstw; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if i == 0 || i == dstw-1 || j == 0 || j == dsth-1 {
want = color.RGBA{A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageDrawTrianglesAndMutateArgs(t *testing.T) {
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
clr := color.RGBA{R: 0xff, A: 0xff}
src.Fill(clr)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
dst.DrawTriangles(vs, is, src, nil)
vs[0].SrcX = w
vs[0].SrcY = h
is[5] = 0
for j := 0; j < w; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := clr
if got != want {
t.Errorf("dst.At(%d, %d): got %v, want %v", i, j, got, want)
}
}
}
}
func TestImageWritePixelsOnSubImage(t *testing.T) {
dst := ebiten.NewImage(17, 31)
dst.Fill(color.RGBA{R: 0xff, A: 0xff})
pix0 := make([]byte, 4*5*3)
idx := 0
for j := 0; j < 3; j++ {
for i := 0; i < 5; i++ {
pix0[4*idx] = 0
pix0[4*idx+1] = 0xff
pix0[4*idx+2] = 0
pix0[4*idx+3] = 0xff
idx++
}
}
r0 := image.Rect(4, 5, 9, 8)
dst.SubImage(r0).(*ebiten.Image).WritePixels(pix0)
pix1 := make([]byte, 4*5*3)
idx = 0
for j := 0; j < 3; j++ {
for i := 0; i < 5; i++ {
pix1[4*idx] = 0
pix1[4*idx+1] = 0
pix1[4*idx+2] = 0xff
pix1[4*idx+3] = 0xff
idx++
}
}
r1 := image.Rect(11, 10, 16, 13)
dst.SubImage(r1).(*ebiten.Image).WritePixels(pix1)
// Clear the pixels. This should not affect the result.
idx = 0
for j := 0; j < 3; j++ {
for i := 0; i < 5; i++ {
pix1[4*idx] = 0
pix1[4*idx+1] = 0
pix1[4*idx+2] = 0
pix1[4*idx+3] = 0
idx++
}
}
for j := 0; j < 31; j++ {
for i := 0; i < 17; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: 0xff, A: 0xff}
p := image.Pt(i, j)
switch {
case p.In(r0):
want = color.RGBA{G: 0xff, A: 0xff}
case p.In(r1):
want = color.RGBA{B: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageDrawTrianglesWithColorM(t *testing.T) {
const w, h = 16, 16
dst0 := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
src.Fill(color.White)
vs0 := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
op := &ebiten.DrawTrianglesOptions{}
op.ColorM.Scale(0.2, 0.4, 0.6, 0.8)
is := []uint16{0, 1, 2, 1, 2, 3}
dst0.DrawTriangles(vs0, is, src, op)
for _, format := range []ebiten.ColorScaleMode{
ebiten.ColorScaleModeStraightAlpha,
ebiten.ColorScaleModePremultipliedAlpha,
} {
format := format
t.Run(fmt.Sprintf("format%d", format), func(t *testing.T) {
var cr, cg, cb, ca float32
switch format {
case ebiten.ColorScaleModeStraightAlpha:
// The values are the same as ColorM.Scale
cr = 0.2
cg = 0.4
cb = 0.6
ca = 0.8
case ebiten.ColorScaleModePremultipliedAlpha:
cr = 0.2 * 0.8
cg = 0.4 * 0.8
cb = 0.6 * 0.8
ca = 0.8
}
vs1 := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: cr,
ColorG: cg,
ColorB: cb,
ColorA: ca,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: cr,
ColorG: cg,
ColorB: cb,
ColorA: ca,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: cr,
ColorG: cg,
ColorB: cb,
ColorA: ca,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: cr,
ColorG: cg,
ColorB: cb,
ColorA: ca,
},
}
dst1 := ebiten.NewImage(w, h)
op := &ebiten.DrawTrianglesOptions{}
op.ColorScaleMode = format
dst1.DrawTriangles(vs1, is, src, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst0.At(i, j)
want := dst1.At(i, j)
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
})
}
}
func TestImageDrawTrianglesInterpolatesColors(t *testing.T) {
const w, h = 3, 1
src := ebiten.NewImage(w, h)
src.Fill(color.White)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 0,
ColorB: 0,
ColorA: 0,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 0,
ColorG: 1,
ColorB: 0,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 0,
ColorB: 0,
ColorA: 0,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 0,
ColorG: 1,
ColorB: 0,
ColorA: 1,
},
}
for _, format := range []ebiten.ColorScaleMode{
ebiten.ColorScaleModeStraightAlpha,
ebiten.ColorScaleModePremultipliedAlpha,
} {
format := format
t.Run(fmt.Sprintf("format%d", format), func(t *testing.T) {
dst := ebiten.NewImage(w, h)
dst.Fill(color.RGBA{B: 0xff, A: 0xff})
op := &ebiten.DrawTrianglesOptions{}
op.ColorScaleMode = format
is := []uint16{0, 1, 2, 1, 2, 3}
dst.DrawTriangles(vs, is, src, op)
got := dst.At(1, 0).(color.RGBA)
// Correct color interpolation uses the alpha channel
// and notices that colors on the left side of the texture are fully transparent.
var want color.RGBA
switch format {
case ebiten.ColorScaleModeStraightAlpha:
want = color.RGBA{G: 0x80, B: 0x80, A: 0xff}
case ebiten.ColorScaleModePremultipliedAlpha:
want = color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0xff}
}
if !sameColors(got, want, 2) {
t.Errorf("At(1, 0): got: %v, want: %v", got, want)
}
})
}
}
func TestImageDrawTrianglesShaderInterpolatesValues(t *testing.T) {
const w, h = 3, 1
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
src.Fill(color.White)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 0,
ColorB: 0,
ColorA: 0,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 0,
ColorG: 1,
ColorB: 0,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 0,
ColorB: 0,
ColorA: 0,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 0,
ColorG: 1,
ColorB: 0,
ColorA: 1,
},
}
dst.Fill(color.RGBA{B: 0xff, A: 0xff})
op := &ebiten.DrawTrianglesShaderOptions{
Images: [4]*ebiten.Image{src, nil, nil, nil},
}
is := []uint16{0, 1, 2, 1, 2, 3}
shader, err := ebiten.NewShader([]byte(`
package main
func Fragment(dstPos vec4, srcPos vec2, color vec4) vec4 {
return color
}
`))
if err != nil {
t.Fatalf("could not compile shader: %v", err)
}
dst.DrawTrianglesShader(vs, is, shader, op)
got := dst.At(1, 0).(color.RGBA)
// Shaders get each color value interpolated independently.
want := color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0xff}
if !sameColors(got, want, 2) {
t.Errorf("At(1, 0): got: %v, want: %v", got, want)
}
}
// Issue #1137
func TestImageDrawOver(t *testing.T) {
const (
w = 320
h = 240
)
dst := ebiten.NewImage(w, h)
src := image.NewUniform(color.RGBA{R: 0xff, A: 0xff})
// This must not cause infinite-loop.
draw.Draw(dst, dst.Bounds(), src, image.ZP, draw.Over)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageDrawDisposedImage(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawImage must panic but not")
}
}()
dst := ebiten.NewImage(16, 16)
src := ebiten.NewImage(16, 16)
src.Dispose()
dst.DrawImage(src, nil)
}
func TestImageDrawDeallocatedImage(t *testing.T) {
dst := ebiten.NewImage(16, 16)
src := ebiten.NewImage(16, 16)
src.Deallocate()
// DrawImage must not panic.
dst.DrawImage(src, nil)
}
func TestImageDrawTrianglesDisposedImage(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawTriangles must panic but not")
}
}()
dst := ebiten.NewImage(16, 16)
src := ebiten.NewImage(16, 16)
src.Dispose()
vs := make([]ebiten.Vertex, 4)
is := []uint16{0, 1, 2, 1, 2, 3}
dst.DrawTriangles(vs, is, src, nil)
}
func TestImageDrawTrianglesDeallocateImage(t *testing.T) {
dst := ebiten.NewImage(16, 16)
src := ebiten.NewImage(16, 16)
src.Deallocate()
vs := make([]ebiten.Vertex, 4)
is := []uint16{0, 1, 2, 1, 2, 3}
// DrawTriangles must not panic.
dst.DrawTriangles(vs, is, src, nil)
}
// #1137
func BenchmarkImageDrawOver(b *testing.B) {
dst := ebiten.NewImage(16, 16)
src := image.NewUniform(color.Black)
for n := 0; n < b.N; n++ {
draw.Draw(dst, dst.Bounds(), src, image.ZP, draw.Over)
}
}
// Issue #1171
func TestImageFloatTranslate(t *testing.T) {
const w, h = 32, 32
for s := 2; s <= 8; s++ {
s := s
t.Run(fmt.Sprintf("scale%d", s), func(t *testing.T) {
check := func(src *ebiten.Image) {
dst := ebiten.NewImage(w*(s+1), h*(s+1))
dst.Fill(color.RGBA{R: 0xff, A: 0xff})
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(float64(s), float64(s))
op.GeoM.Translate(0, 0.501)
dst.DrawImage(src, op)
for j := 0; j < h*s+1; j++ {
for i := 0; i < w*s; i++ {
got := dst.At(i, j)
x := byte(0xff)
if j > 0 {
x = (byte(j) - 1) / byte(s)
}
want := color.RGBA{R: x, A: 0xff}
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
t.Run("image", func(t *testing.T) {
src := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
pix[4*(j*w+i)] = byte(j)
pix[4*(j*w+i)+3] = 0xff
}
}
src.WritePixels(pix)
check(src)
})
t.Run("subimage", func(t *testing.T) {
src := ebiten.NewImage(w*s, h*s)
pix := make([]byte, 4*(w*s)*(h*s))
for j := 0; j < h*s; j++ {
for i := 0; i < w*s; i++ {
pix[4*(j*(w*s)+i)] = byte(j)
pix[4*(j*(w*s)+i)+3] = 0xff
}
}
src.WritePixels(pix)
check(src.SubImage(image.Rect(0, 0, w, h)).(*ebiten.Image))
})
})
}
}
// Issue #1213
func TestImageColorMCopy(t *testing.T) {
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
for k := 0; k < 256; k++ {
op := &ebiten.DrawImageOptions{}
op.ColorM.Translate(1, 1, 1, float64(k)/0xff)
op.Blend = ebiten.BlendCopy
dst.DrawImage(src, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: byte(k), G: byte(k), B: byte(k), A: byte(k)}
if !sameColors(got, want, 1) {
t.Fatalf("dst.At(%d, %d), k: %d: got %v, want %v", i, j, k, got, want)
}
}
}
}
}
// TODO: Do we have to guarantee this behavior? See #1222
func TestImageWritePixelsAndModifyPixels(t *testing.T) {
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
pix[idx] = 0xff
pix[idx+1] = 0
pix[idx+2] = 0
pix[idx+3] = 0xff
}
}
src.WritePixels(pix)
// Modify pix after WritePixels
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
pix[idx] = 0
pix[idx+1] = 0xff
pix[idx+2] = 0
pix[idx+3] = 0xff
}
}
// Ensure that src's pixels are actually used
dst.DrawImage(src, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := src.At(i, j).(color.RGBA)
want := color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("src.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageCompositeModeMultiply(t *testing.T) {
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
dst.Fill(color.RGBA{R: 0x10, G: 0x20, B: 0x30, A: 0x40})
src.Fill(color.RGBA{R: 0x50, G: 0x60, B: 0x70, A: 0x80})
op := &ebiten.DrawImageOptions{}
op.CompositeMode = ebiten.CompositeModeMultiply
dst.DrawImage(src, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{
R: byte(math.Floor((0x10 / 255.0) * (0x50 / 255.0) * 255)),
G: byte(math.Floor((0x20 / 255.0) * (0x60 / 255.0) * 255)),
B: byte(math.Floor((0x30 / 255.0) * (0x70 / 255.0) * 255)),
A: byte(math.Floor((0x40 / 255.0) * (0x80 / 255.0) * 255)),
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #1269
func TestImageZeroTriangle(t *testing.T) {
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(1, 1)
vs := []ebiten.Vertex{}
is := []uint16{}
dst.DrawTriangles(vs, is, src, nil)
}
// Issue #1398
func TestImageDrawImageTooBigScale(t *testing.T) {
dst := ebiten.NewImage(1, 1)
src := ebiten.NewImage(1, 1)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(1e20, 1e20)
dst.DrawImage(src, op)
}
// Issue #1398
func TestImageDrawImageTooSmallScale(t *testing.T) {
dst := ebiten.NewImage(1, 1)
src := ebiten.NewImage(1, 1)
op := &ebiten.DrawImageOptions{}
op.Filter = ebiten.FilterLinear
op.GeoM.Scale(1e-10, 1e-10)
dst.DrawImage(src, op)
}
// Issue #1399
func TestImageDrawImageCannotAllocateImageForMipmap(t *testing.T) {
dst := ebiten.NewImage(1, 1)
src := ebiten.NewImage(maxImageSize, maxImageSize)
op := &ebiten.DrawImageOptions{}
op.GeoM.Scale(64, 64)
dst.DrawImage(src, op)
dst.At(0, 0)
}
func TestImageNewImageWithZeroSize(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawImage must panic but not")
}
}()
_ = ebiten.NewImage(0, 1)
}
func TestImageNewImageFromImageWithZeroSize(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawImage must panic but not")
}
}()
img := image.NewRGBA(image.Rect(0, 0, 0, 1))
_ = ebiten.NewImageFromImage(img)
}
func TestImageClip(t *testing.T) {
const (
w = 16
h = 16
)
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
dst.Fill(color.RGBA{R: 0xff, A: 0xff})
src.Fill(color.RGBA{G: 0xff, A: 0xff})
dst.SubImage(image.Rect(4, 5, 12, 14)).(*ebiten.Image).DrawImage(src, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: 0xff, A: 0xff}
if 4 <= i && i < 12 && 5 <= j && j < 14 {
want = color.RGBA{G: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #1691
func TestImageSubImageFill(t *testing.T) {
dst := ebiten.NewImage(3, 3).SubImage(image.Rect(1, 1, 2, 2)).(*ebiten.Image)
dst.Fill(color.White)
for j := 0; j < 3; j++ {
for i := 0; i < 3; i++ {
got := dst.At(i, j)
var want color.RGBA
if i == 1 && j == 1 {
want = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
dst = ebiten.NewImage(17, 31).SubImage(image.Rect(3, 4, 8, 10)).(*ebiten.Image)
dst.Fill(color.White)
for j := 0; j < 31; j++ {
for i := 0; i < 17; i++ {
got := dst.At(i, j)
var want color.RGBA
if 3 <= i && i < 8 && 4 <= j && j < 10 {
want = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageEvenOdd(t *testing.T) {
whiteImage := ebiten.NewImage(3, 3)
whiteImage.Fill(color.White)
emptySubImage := whiteImage.SubImage(image.Rect(1, 1, 2, 2)).(*ebiten.Image)
vs0 := []ebiten.Vertex{
{
DstX: 1, DstY: 1, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
{
DstX: 15, DstY: 1, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
{
DstX: 1, DstY: 15, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
{
DstX: 15, DstY: 15, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
}
is0 := []uint16{0, 1, 2, 1, 2, 3}
vs1 := []ebiten.Vertex{
{
DstX: 2, DstY: 2, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
{
DstX: 14, DstY: 2, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
{
DstX: 2, DstY: 14, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
{
DstX: 14, DstY: 14, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
}
is1 := []uint16{4, 5, 6, 5, 6, 7}
vs2 := []ebiten.Vertex{
{
DstX: 3, DstY: 3, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
{
DstX: 13, DstY: 3, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
{
DstX: 3, DstY: 13, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
{
DstX: 13, DstY: 13, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
}
is2 := []uint16{8, 9, 10, 9, 10, 11}
// Draw all the vertices once. The even-odd rule is applied for all the vertices once.
dst := ebiten.NewImage(16, 16)
op := &ebiten.DrawTrianglesOptions{
FillRule: ebiten.FillRuleEvenOdd,
}
dst.DrawTriangles(append(append(vs0, vs1...), vs2...), append(append(is0, is1...), is2...), emptySubImage, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst.At(i, j)
var want color.RGBA
switch {
case 3 <= i && i < 13 && 3 <= j && j < 13:
want = color.RGBA{B: 0xff, A: 0xff}
case 2 <= i && i < 14 && 2 <= j && j < 14:
want = color.RGBA{}
case 1 <= i && i < 15 && 1 <= j && j < 15:
want = color.RGBA{R: 0xff, A: 0xff}
default:
want = color.RGBA{}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
// Do the same thing but with a little shift. This confirms that the underlying stencil buffer is cleared correctly.
for i := range vs0 {
vs0[i].DstX++
vs0[i].DstY++
}
for i := range vs1 {
vs1[i].DstX++
vs1[i].DstY++
}
for i := range vs2 {
vs2[i].DstX++
vs2[i].DstY++
}
dst.Clear()
dst.DrawTriangles(append(append(vs0, vs1...), vs2...), append(append(is0, is1...), is2...), emptySubImage, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst.At(i, j)
var want color.RGBA
switch {
case 4 <= i && i < 14 && 4 <= j && j < 14:
want = color.RGBA{B: 0xff, A: 0xff}
case 3 <= i && i < 15 && 3 <= j && j < 15:
want = color.RGBA{}
case 2 <= i && i < 16 && 2 <= j && j < 16:
want = color.RGBA{R: 0xff, A: 0xff}
default:
want = color.RGBA{}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
// Do the same thing but with split DrawTriangle calls. This confirms that the even-odd rule is applied for one call.
for i := range vs0 {
vs0[i].DstX--
vs0[i].DstY--
}
for i := range vs1 {
vs1[i].DstX--
vs1[i].DstY--
}
for i := range vs2 {
vs2[i].DstX--
vs2[i].DstY--
}
dst.Clear()
// Use the first indices set.
dst.DrawTriangles(vs0, is0, emptySubImage, op)
dst.DrawTriangles(vs1, is0, emptySubImage, op)
dst.DrawTriangles(vs2, is0, emptySubImage, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst.At(i, j)
var want color.RGBA
switch {
case 3 <= i && i < 13 && 3 <= j && j < 13:
want = color.RGBA{B: 0xff, A: 0xff}
case 2 <= i && i < 14 && 2 <= j && j < 14:
want = color.RGBA{G: 0xff, A: 0xff}
case 1 <= i && i < 15 && 1 <= j && j < 15:
want = color.RGBA{R: 0xff, A: 0xff}
default:
want = color.RGBA{}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageFillRule(t *testing.T) {
for _, fillRule := range []ebiten.FillRule{ebiten.FillRuleFillAll, ebiten.FillRuleNonZero, ebiten.FillRuleEvenOdd} {
fillRule := fillRule
var name string
switch fillRule {
case ebiten.FillRuleFillAll:
name = "FillAll"
case ebiten.FillRuleNonZero:
name = "NonZero"
case ebiten.FillRuleEvenOdd:
name = "EvenOdd"
}
t.Run(name, func(t *testing.T) {
whiteImage := ebiten.NewImage(3, 3)
whiteImage.Fill(color.White)
emptySubImage := whiteImage.SubImage(image.Rect(1, 1, 2, 2)).(*ebiten.Image)
// The outside rectangle (clockwise)
vs0 := []ebiten.Vertex{
{
DstX: 1, DstY: 1, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
{
DstX: 15, DstY: 1, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
{
DstX: 15, DstY: 15, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
{
DstX: 1, DstY: 15, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 0, ColorB: 0, ColorA: 1,
},
}
is0 := []uint16{0, 1, 2, 2, 3, 0}
// An inside rectangle (clockwise)
vs1 := []ebiten.Vertex{
{
DstX: 2, DstY: 2, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
{
DstX: 7, DstY: 2, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
{
DstX: 7, DstY: 7, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
{
DstX: 2, DstY: 7, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 1, ColorB: 0, ColorA: 1,
},
}
is1 := []uint16{4, 5, 6, 6, 7, 4}
// An inside rectangle (counter-clockwise)
vs2 := []ebiten.Vertex{
{
DstX: 9, DstY: 9, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
{
DstX: 14, DstY: 9, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
{
DstX: 14, DstY: 14, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
{
DstX: 9, DstY: 14, SrcX: 1, SrcY: 1,
ColorR: 0, ColorG: 0, ColorB: 1, ColorA: 1,
},
}
is2 := []uint16{8, 11, 10, 10, 9, 8}
// Draw all the vertices once. The even-odd rule is applied for all the vertices once.
dst := ebiten.NewImage(16, 16)
op := &ebiten.DrawTrianglesOptions{
FillRule: fillRule,
}
dst.DrawTriangles(append(append(vs0, vs1...), vs2...), append(append(is0, is1...), is2...), emptySubImage, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst.At(i, j)
var want color.RGBA
switch {
case 2 <= i && i < 7 && 2 <= j && j < 7:
if fillRule != ebiten.FillRuleEvenOdd {
want = color.RGBA{G: 0xff, A: 0xff}
}
case 9 <= i && i < 14 && 9 <= j && j < 14:
if fillRule == ebiten.FillRuleFillAll {
want = color.RGBA{B: 0xff, A: 0xff}
}
case 1 <= i && i < 15 && 1 <= j && j < 15:
want = color.RGBA{R: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
// Do the same thing but with a little shift. This confirms that the underlying stencil buffer is cleared correctly.
for i := range vs0 {
vs0[i].DstX++
vs0[i].DstY++
}
for i := range vs1 {
vs1[i].DstX++
vs1[i].DstY++
}
for i := range vs2 {
vs2[i].DstX++
vs2[i].DstY++
}
dst.Clear()
dst.DrawTriangles(append(append(vs0, vs1...), vs2...), append(append(is0, is1...), is2...), emptySubImage, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst.At(i, j)
var want color.RGBA
switch {
case 3 <= i && i < 8 && 3 <= j && j < 8:
if fillRule != ebiten.FillRuleEvenOdd {
want = color.RGBA{G: 0xff, A: 0xff}
}
case 10 <= i && i < 15 && 10 <= j && j < 15:
if fillRule == ebiten.FillRuleFillAll {
want = color.RGBA{B: 0xff, A: 0xff}
}
case 2 <= i && i < 16 && 2 <= j && j < 16:
want = color.RGBA{R: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
// Do the same thing but with split DrawTriangle calls. This confirms that fill rules are applied for one call.
for i := range vs0 {
vs0[i].DstX--
vs0[i].DstY--
}
for i := range vs1 {
vs1[i].DstX--
vs1[i].DstY--
}
for i := range vs2 {
vs2[i].DstX--
vs2[i].DstY--
}
dst.Clear()
dst.DrawTriangles(vs0, []uint16{0, 1, 2, 2, 3, 0}, emptySubImage, op)
dst.DrawTriangles(vs1, []uint16{0, 1, 2, 2, 3, 0}, emptySubImage, op)
dst.DrawTriangles(vs2, []uint16{0, 3, 2, 2, 1, 0}, emptySubImage, op)
for j := 0; j < 16; j++ {
for i := 0; i < 16; i++ {
got := dst.At(i, j)
var want color.RGBA
switch {
case 2 <= i && i < 7 && 2 <= j && j < 7:
want = color.RGBA{G: 0xff, A: 0xff}
case 9 <= i && i < 14 && 9 <= j && j < 14:
want = color.RGBA{B: 0xff, A: 0xff}
case 1 <= i && i < 15 && 1 <= j && j < 15:
want = color.RGBA{R: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
})
}
}
// #1658
func BenchmarkColorMScale(b *testing.B) {
r := rand.Float64
dst := ebiten.NewImage(16, 16)
src := ebiten.NewImage(16, 16)
for n := 0; n < b.N; n++ {
op := &ebiten.DrawImageOptions{}
op.ColorM.Scale(r(), r(), r(), r())
dst.DrawImage(src, op)
}
}
func TestImageMoreIndicesThanMaxUint16(t *testing.T) {
const (
w = 16
h = 16
)
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
src.Fill(color.White)
op := &ebiten.DrawTrianglesOptions{}
vs := make([]ebiten.Vertex, 3)
is := make([]uint16, 65538)
dst.DrawTriangles(vs, is, src, op)
// The next draw call should work well (and this is likely batched).
vs = []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is = []uint16{0, 1, 2, 1, 2, 3}
dst.DrawTriangles(vs, is, src, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageMoreVerticesThanMaxUint16(t *testing.T) {
const (
w = 16
h = 16
)
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
src.Fill(color.White)
op := &ebiten.DrawTrianglesOptions{}
vs := make([]ebiten.Vertex, math.MaxUint16+1)
is := make([]uint16, 3)
dst.DrawTriangles(vs, is, src, op)
// The next draw call should work well (and this is likely batched).
vs = []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is = []uint16{0, 1, 2, 1, 2, 3}
dst.DrawTriangles(vs, is, src, op)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageNewImageFromEbitenImage(t *testing.T) {
const (
w = 16
h = 16
)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
pix[idx] = byte(i)
pix[idx+1] = byte(j)
pix[idx+2] = 0
pix[idx+3] = 0xff
}
}
img0 := ebiten.NewImage(w, h)
img0.WritePixels(pix)
img1 := ebiten.NewImageFromImage(img0)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img1.At(i, j)
want := color.RGBA{R: byte(i), G: byte(j), A: 0xff}
if got != want {
t.Errorf("img1.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
img2 := ebiten.NewImageFromImage(img0.SubImage(image.Rect(4, 4, 12, 12)))
for j := 0; j < h/2; j++ {
for i := 0; i < w/2; i++ {
got := img2.At(i, j)
want := color.RGBA{R: byte(i + 4), G: byte(j + 4), A: 0xff}
if got != want {
t.Errorf("img1.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageOptionsUnmanaged(t *testing.T) {
const (
w = 16
h = 16
)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
pix[idx] = byte(i)
pix[idx+1] = byte(j)
pix[idx+2] = 0
pix[idx+3] = 0xff
}
}
op := &ebiten.NewImageOptions{
Unmanaged: true,
}
img := ebiten.NewImageWithOptions(image.Rect(0, 0, w, h), op)
img.WritePixels(pix)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := img.At(i, j)
want := color.RGBA{R: byte(i), G: byte(j), A: 0xff}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageOptionsNegativeBoundsWritePixels(t *testing.T) {
const (
w = 16
h = 16
)
pix0 := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
pix0[idx] = byte(i)
pix0[idx+1] = byte(j)
pix0[idx+2] = 0
pix0[idx+3] = 0xff
}
}
const offset = -8
img := ebiten.NewImageWithOptions(image.Rect(offset, offset, w+offset, h+offset), nil)
img.WritePixels(pix0)
for j := offset; j < h+offset; j++ {
for i := offset; i < w+offset; i++ {
got := img.At(i, j)
want := color.RGBA{R: byte(i - offset), G: byte(j - offset), A: 0xff}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
pix1 := make([]byte, 4*(w/2)*(h/2))
for j := 0; j < h/2; j++ {
for i := 0; i < w/2; i++ {
idx := 4 * (i + j*w/2)
pix1[idx] = 0
pix1[idx+1] = 0
pix1[idx+2] = 0xff
pix1[idx+3] = 0xff
}
}
const offset2 = -4
sub := image.Rect(offset2, offset2, w/2+offset2, h/2+offset2)
img.SubImage(sub).(*ebiten.Image).WritePixels(pix1)
for j := offset; j < h+offset; j++ {
for i := offset; i < w+offset; i++ {
got := img.At(i, j)
want := color.RGBA{R: byte(i - offset), G: byte(j - offset), A: 0xff}
if image.Pt(i, j).In(sub) {
want = color.RGBA{B: 0xff, A: 0xff}
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageOptionsNegativeBoundsSet(t *testing.T) {
const (
w = 16
h = 16
)
pix0 := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (i + j*w)
pix0[idx] = byte(i)
pix0[idx+1] = byte(j)
pix0[idx+2] = 0
pix0[idx+3] = 0xff
}
}
const offset = -8
img := ebiten.NewImageWithOptions(image.Rect(offset, offset, w+offset, h+offset), nil)
img.WritePixels(pix0)
img.Set(-1, -2, color.RGBA{})
for j := offset; j < h+offset; j++ {
for i := offset; i < w+offset; i++ {
got := img.At(i, j)
want := color.RGBA{R: byte(i - offset), G: byte(j - offset), A: 0xff}
if i == -1 && j == -2 {
want = color.RGBA{}
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageOptionsNegativeBoundsDrawImage(t *testing.T) {
const (
w = 16
h = 16
offset = -8
)
dst := ebiten.NewImageWithOptions(image.Rect(offset, offset, w+offset, h+offset), nil)
src := ebiten.NewImageWithOptions(image.Rect(-1, -1, 1, 1), nil)
pix := make([]byte, 4*2*2)
for i := range pix {
pix[i] = 0xff
}
src.WritePixels(pix)
op := &ebiten.DrawImageOptions{}
op.GeoM.Translate(-1, -1)
op.GeoM.Scale(2, 3)
dst.DrawImage(src, op)
for j := offset; j < h+offset; j++ {
for i := offset; i < w+offset; i++ {
got := dst.At(i, j)
var want color.RGBA
if -2 <= i && i < 2 && -3 <= j && j < 3 {
want = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageOptionsNegativeBoundsDrawTriangles(t *testing.T) {
const (
w = 16
h = 16
offset = -8
)
dst := ebiten.NewImageWithOptions(image.Rect(offset, offset, w+offset, h+offset), nil)
src := ebiten.NewImageWithOptions(image.Rect(-1, -1, 1, 1), nil)
pix := make([]byte, 4*2*2)
for i := range pix {
pix[i] = 0xff
}
src.WritePixels(pix)
vs := []ebiten.Vertex{
{
DstX: -2,
DstY: -3,
SrcX: -1,
SrcY: -1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 2,
DstY: -3,
SrcX: 1,
SrcY: -1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: -2,
DstY: 3,
SrcX: -1,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 2,
DstY: 3,
SrcX: 1,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
dst.DrawTriangles(vs, is, src, nil)
for j := offset; j < h+offset; j++ {
for i := offset; i < w+offset; i++ {
got := dst.At(i, j)
var want color.RGBA
if -2 <= i && i < 2 && -3 <= j && j < 3 {
want = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
}
if got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageFromImageOptions(t *testing.T) {
r := image.Rect(-2, -3, 4, 5)
pix := make([]byte, 4*r.Dx()*r.Dy())
for i := range pix {
pix[i] = 0xff
}
src := &image.RGBA{
Pix: pix,
Stride: 4 * 2,
Rect: r,
}
op := &ebiten.NewImageFromImageOptions{
PreserveBounds: true,
}
img := ebiten.NewImageFromImageWithOptions(src, op)
if got, want := img.Bounds(), r; got != want {
t.Errorf("got: %v, want: %v", got, want)
}
for j := r.Min.Y; j < r.Max.Y; j++ {
for i := r.Min.X; i < r.Max.X; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageFromEbitenImageOptions(t *testing.T) {
r := image.Rect(-2, -3, 4, 5)
src := ebiten.NewImageWithOptions(r, nil)
pix := make([]byte, 4*r.Dx()*r.Dy())
for i := range pix {
pix[i] = 0xff
}
src.WritePixels(pix)
op := &ebiten.NewImageFromImageOptions{
PreserveBounds: true,
}
img := ebiten.NewImageFromImageWithOptions(src, op)
if got, want := img.Bounds(), r; got != want {
t.Errorf("got: %v, want: %v", got, want)
}
for j := r.Min.Y; j < r.Max.Y; j++ {
for i := r.Min.X; i < r.Max.X; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #2159
func TestImageOptionsFill(t *testing.T) {
r0 := image.Rect(-2, -3, 4, 5)
img := ebiten.NewImageWithOptions(r0, nil)
img.Fill(color.RGBA{R: 0xff, A: 0xff})
for j := r0.Min.Y; j < r0.Max.Y; j++ {
for i := r0.Min.X; i < r0.Max.X; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
r1 := image.Rect(-1, -2, 3, 4)
img.SubImage(r1).(*ebiten.Image).Fill(color.RGBA{G: 0xff, A: 0xff})
for j := r0.Min.Y; j < r0.Max.Y; j++ {
for i := r0.Min.X; i < r0.Max.X; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0xff, A: 0xff}
if image.Pt(i, j).In(r1) {
want = color.RGBA{G: 0xff, A: 0xff}
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #2159
func TestImageOptionsClear(t *testing.T) {
r0 := image.Rect(-2, -3, 4, 5)
img := ebiten.NewImageWithOptions(r0, nil)
img.Fill(color.RGBA{R: 0xff, A: 0xff})
img.Clear()
for j := r0.Min.Y; j < r0.Max.Y; j++ {
for i := r0.Min.X; i < r0.Max.X; i++ {
got := img.At(i, j)
want := color.RGBA{}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
img.Fill(color.RGBA{R: 0xff, A: 0xff})
r1 := image.Rect(-1, -2, 3, 4)
img.SubImage(r1).(*ebiten.Image).Clear()
for j := r0.Min.Y; j < r0.Max.Y; j++ {
for i := r0.Min.X; i < r0.Max.X; i++ {
got := img.At(i, j)
want := color.RGBA{R: 0xff, A: 0xff}
if image.Pt(i, j).In(r1) {
want = color.RGBA{}
}
if got != want {
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #2178
func TestImageTooManyDrawImage(t *testing.T) {
src := ebiten.NewImage(1, 1)
src.Fill(color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff})
const (
w = 256
h = 256
)
dst := ebiten.NewImage(w, h)
op := &ebiten.DrawImageOptions{}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
op.GeoM.Reset()
op.GeoM.Translate(float64(i), float64(j))
dst.DrawImage(src, op)
}
}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if got, want := dst.At(i, j), (color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}); got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageTooManyDrawImage2(t *testing.T) {
src := ebiten.NewImage(1, 1)
src.Fill(color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff})
const (
w = 512
h = 512
)
dst := ebiten.NewImage(w, h)
posToColor := func(i, j int) color.RGBA {
return color.RGBA{
R: byte(i),
G: byte(j),
B: 0xff,
A: 0xff,
}
}
op := &ebiten.DrawImageOptions{}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
op.GeoM.Reset()
op.GeoM.Translate(float64(i), float64(j))
op.ColorScale.Reset()
op.ColorScale.ScaleWithColor(posToColor(i, j))
dst.DrawImage(src, op)
}
}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if got, want := dst.At(i, j).(color.RGBA), posToColor(i, j); !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #2178
func TestImageTooManyDrawTriangles(t *testing.T) {
img := ebiten.NewImage(3, 3)
img.Fill(color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff})
src := img.SubImage(image.Rect(1, 1, 2, 2)).(*ebiten.Image)
const (
w = 128
h = 64
)
dst := ebiten.NewImage(w, h)
var vertices []ebiten.Vertex
var indices []uint16
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
n := uint16(len(vertices))
vertices = append(vertices,
ebiten.Vertex{
DstX: float32(i),
DstY: float32(j),
SrcX: 1,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
ebiten.Vertex{
DstX: float32(i) + 1,
DstY: float32(j),
SrcX: 2,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
ebiten.Vertex{
DstX: float32(i),
DstY: float32(j) + 1,
SrcX: 1,
SrcY: 2,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
ebiten.Vertex{
DstX: float32(i) + 1,
DstY: float32(j) + 1,
SrcX: 2,
SrcY: 2,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
)
indices = append(indices, n, n+1, n+2, n+1, n+2, n+3)
}
}
dst.DrawTriangles(vertices, indices, src, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
if got, want := dst.At(i, j), (color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}); got != want {
t.Errorf("dst.At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageSetOverSet(t *testing.T) {
img := ebiten.NewImage(1, 1)
img.Set(0, 0, color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff})
if got, want := img.At(0, 0), (color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
// Apply the change by 'Set' by calling DrawImage.
dummy := ebiten.NewImage(1, 1)
img.DrawImage(dummy, nil)
if got, want := img.At(0, 0), (color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
img.Set(0, 0, color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0x80})
if got, want := img.At(0, 0), (color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0x80}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
// Apply the change by 'Set' again.
img.DrawImage(dummy, nil)
if got, want := img.At(0, 0), (color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0x80}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
// Issue #2204
func TestImageTooManyConstantBuffersInDirectX(t *testing.T) {
src := ebiten.NewImage(3, 3)
src.Fill(color.White)
src = src.SubImage(image.Rect(1, 1, 2, 2)).(*ebiten.Image)
vs := []ebiten.Vertex{
{
DstX: 0, DstY: 0, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 1, ColorB: 1, ColorA: 1,
},
{
DstX: 16, DstY: 0, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 1, ColorB: 1, ColorA: 1,
},
{
DstX: 0, DstY: 16, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 1, ColorB: 1, ColorA: 1,
},
{
DstX: 16, DstY: 16, SrcX: 1, SrcY: 1,
ColorR: 1, ColorG: 1, ColorB: 1, ColorA: 1,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
dst0 := ebiten.NewImage(16, 16)
dst1 := ebiten.NewImage(16, 16)
op := &ebiten.DrawTrianglesOptions{
FillRule: ebiten.FillRuleEvenOdd,
}
for i := 0; i < 100; i++ {
dst0.DrawTriangles(vs, is, src, op)
dst1.DrawTriangles(vs, is, src, op)
}
if got, want := dst0.At(0, 0), (color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
if got, want := dst1.At(0, 0), (color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageColorMAndScale(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
src.Fill(color.RGBA{R: 0x80, G: 0x80, B: 0x80, A: 0x80})
vs := []ebiten.Vertex{
{
SrcX: 0,
SrcY: 0,
DstX: 0,
DstY: 0,
ColorR: 0.5,
ColorG: 0.25,
ColorB: 0.5,
ColorA: 0.75,
},
{
SrcX: w,
SrcY: 0,
DstX: w,
DstY: 0,
ColorR: 0.5,
ColorG: 0.25,
ColorB: 0.5,
ColorA: 0.75,
},
{
SrcX: 0,
SrcY: h,
DstX: 0,
DstY: h,
ColorR: 0.5,
ColorG: 0.25,
ColorB: 0.5,
ColorA: 0.75,
},
{
SrcX: w,
SrcY: h,
DstX: w,
DstY: h,
ColorR: 0.5,
ColorG: 0.25,
ColorB: 0.5,
ColorA: 0.75,
},
}
is := []uint16{0, 1, 2, 1, 2, 3}
for _, format := range []ebiten.ColorScaleMode{
ebiten.ColorScaleModeStraightAlpha,
ebiten.ColorScaleModePremultipliedAlpha,
} {
format := format
t.Run(fmt.Sprintf("format%d", format), func(t *testing.T) {
dst := ebiten.NewImage(w, h)
op := &ebiten.DrawTrianglesOptions{}
op.ColorM.Translate(0.25, 0.25, 0.25, 0)
op.ColorScaleMode = format
dst.DrawTriangles(vs, is, src, op)
got := dst.At(0, 0).(color.RGBA)
alphaBeforeScale := 0.5
var want color.RGBA
switch format {
case ebiten.ColorScaleModeStraightAlpha:
want = color.RGBA{
R: byte(math.Floor(0xff * (0.5/alphaBeforeScale + 0.25) * alphaBeforeScale * 0.5 * 0.75)),
G: byte(math.Floor(0xff * (0.5/alphaBeforeScale + 0.25) * alphaBeforeScale * 0.25 * 0.75)),
B: byte(math.Floor(0xff * (0.5/alphaBeforeScale + 0.25) * alphaBeforeScale * 0.5 * 0.75)),
A: byte(math.Floor(0xff * alphaBeforeScale * 0.75)),
}
case ebiten.ColorScaleModePremultipliedAlpha:
want = color.RGBA{
R: byte(math.Floor(0xff * (0.5/alphaBeforeScale + 0.25) * alphaBeforeScale * 0.5)),
G: byte(math.Floor(0xff * (0.5/alphaBeforeScale + 0.25) * alphaBeforeScale * 0.25)),
B: byte(math.Floor(0xff * (0.5/alphaBeforeScale + 0.25) * alphaBeforeScale * 0.5)),
A: byte(math.Floor(0xff * alphaBeforeScale * 0.75)),
}
}
if !sameColors(got, want, 2) {
t.Errorf("got: %v, want: %v", got, want)
}
})
}
}
func TestImageBlendOperation(t *testing.T) {
const w, h = 16, 1
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
dstColor := func(i int) (byte, byte, byte, byte) {
return byte(4 * i * 17), byte(4*i*17 + 1), byte(4*i*17 + 2), byte(4*i*17 + 3)
}
srcColor := func(i int) (byte, byte, byte, byte) {
return byte(4 * i * 13), byte(4*i*13 + 1), byte(4*i*13 + 2), byte(4*i*13 + 3)
}
clamp := func(x int) byte {
if x > 255 {
return 255
}
if x < 0 {
return 0
}
return byte(x)
}
dstPix := make([]byte, 4*w*h)
for i := 0; i < w; i++ {
r, g, b, a := dstColor(i)
dstPix[4*i] = r
dstPix[4*i+1] = g
dstPix[4*i+2] = b
dstPix[4*i+3] = a
}
srcPix := make([]byte, 4*w*h)
for i := 0; i < w; i++ {
r, g, b, a := srcColor(i)
srcPix[4*i] = r
srcPix[4*i+1] = g
srcPix[4*i+2] = b
srcPix[4*i+3] = a
}
src.WritePixels(srcPix)
operations := []ebiten.BlendOperation{
ebiten.BlendOperationAdd,
ebiten.BlendOperationSubtract,
ebiten.BlendOperationReverseSubtract,
}
for _, rgbOp := range operations {
for _, alphaOp := range operations {
// Reset the destination state.
dst.WritePixels(dstPix)
op := &ebiten.DrawImageOptions{}
op.Blend = ebiten.Blend{
BlendFactorSourceRGB: ebiten.BlendFactorOne,
BlendFactorSourceAlpha: ebiten.BlendFactorOne,
BlendFactorDestinationRGB: ebiten.BlendFactorOne,
BlendFactorDestinationAlpha: ebiten.BlendFactorOne,
BlendOperationRGB: rgbOp,
BlendOperationAlpha: alphaOp,
}
dst.DrawImage(src, op)
for i := 0; i < w; i++ {
got := dst.At(i, 0).(color.RGBA)
sr, sg, sb, sa := srcColor(i)
dr, dg, db, da := dstColor(i)
var want color.RGBA
switch rgbOp {
case ebiten.BlendOperationAdd:
want.R = clamp(int(sr) + int(dr))
want.G = clamp(int(sg) + int(dg))
want.B = clamp(int(sb) + int(db))
case ebiten.BlendOperationSubtract:
want.R = clamp(int(sr) - int(dr))
want.G = clamp(int(sg) - int(dg))
want.B = clamp(int(sb) - int(db))
case ebiten.BlendOperationReverseSubtract:
want.R = clamp(int(dr) - int(sr))
want.G = clamp(int(dg) - int(sg))
want.B = clamp(int(db) - int(sb))
}
switch alphaOp {
case ebiten.BlendOperationAdd:
want.A = clamp(int(sa) + int(da))
case ebiten.BlendOperationSubtract:
want.A = clamp(int(sa) - int(da))
case ebiten.BlendOperationReverseSubtract:
want.A = clamp(int(da) - int(sa))
}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, 0): operations: %d, %d: got: %v, want: %v", i, rgbOp, alphaOp, got, want)
}
}
}
}
}
func TestImageBlendOperationMinAndMax(t *testing.T) {
const w, h = 16, 1
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
dstColor := func(i int) (byte, byte, byte, byte) {
return byte(4 * i * 17), byte(4*i*17 + 1), byte(4*i*17 + 2), byte(4*i*17 + 3)
}
srcColor := func(i int) (byte, byte, byte, byte) {
return byte(4 * i * 13), byte(4*i*13 + 1), byte(4*i*13 + 2), byte(4*i*13 + 3)
}
dstPix := make([]byte, 4*w*h)
for i := 0; i < w; i++ {
r, g, b, a := dstColor(i)
dstPix[4*i] = r
dstPix[4*i+1] = g
dstPix[4*i+2] = b
dstPix[4*i+3] = a
}
srcPix := make([]byte, 4*w*h)
for i := 0; i < w; i++ {
r, g, b, a := srcColor(i)
srcPix[4*i] = r
srcPix[4*i+1] = g
srcPix[4*i+2] = b
srcPix[4*i+3] = a
}
src.WritePixels(srcPix)
operations := []ebiten.BlendOperation{
ebiten.BlendOperationMin,
ebiten.BlendOperationMax,
}
for _, rgbOp := range operations {
for _, alphaOp := range operations {
// Reset the destination state.
dst.WritePixels(dstPix)
op := &ebiten.DrawImageOptions{}
// Use the default blend factors, and confirm that the factors are ignored.
op.Blend = ebiten.Blend{
BlendFactorSourceRGB: ebiten.BlendFactorDefault,
BlendFactorSourceAlpha: ebiten.BlendFactorDefault,
BlendFactorDestinationRGB: ebiten.BlendFactorDefault,
BlendFactorDestinationAlpha: ebiten.BlendFactorDefault,
BlendOperationRGB: rgbOp,
BlendOperationAlpha: alphaOp,
}
dst.DrawImage(src, op)
for i := 0; i < w; i++ {
got := dst.At(i, 0).(color.RGBA)
sr, sg, sb, sa := srcColor(i)
dr, dg, db, da := dstColor(i)
var want color.RGBA
switch rgbOp {
case ebiten.BlendOperationMin:
want.R = min(sr, dr)
want.G = min(sg, dg)
want.B = min(sb, db)
case ebiten.BlendOperationMax:
want.R = max(sr, dr)
want.G = max(sg, dg)
want.B = max(sb, db)
}
switch alphaOp {
case ebiten.BlendOperationMin:
want.A = min(sa, da)
case ebiten.BlendOperationMax:
want.A = max(sa, da)
}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, 0): operations: %d, %d: got: %v, want: %v", i, rgbOp, alphaOp, got, want)
}
}
}
}
}
func TestImageBlendFactor(t *testing.T) {
if skipTooSlowTests(t) {
return
}
const w, h = 16, 1
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
dstColor := func(i int) (byte, byte, byte, byte) {
return byte(4 * i * 17), byte(4*i*17 + 1), byte(4*i*17 + 2), byte(4*i*17 + 3)
}
srcColor := func(i int) (byte, byte, byte, byte) {
return byte(4 * i * 13), byte(4*i*13 + 1), byte(4*i*13 + 2), byte(4*i*13 + 3)
}
colorToFloats := func(r, g, b, a byte) (float64, float64, float64, float64) {
return float64(r) / 0xff, float64(g) / 0xff, float64(b) / 0xff, float64(a) / 0xff
}
clamp := func(x int) byte {
if x > 255 {
return 255
}
if x < 0 {
return 0
}
return byte(x)
}
dstPix := make([]byte, 4*w*h)
for i := 0; i < w; i++ {
r, g, b, a := dstColor(i)
dstPix[4*i] = r
dstPix[4*i+1] = g
dstPix[4*i+2] = b
dstPix[4*i+3] = a
}
srcPix := make([]byte, 4*w*h)
for i := 0; i < w; i++ {
r, g, b, a := srcColor(i)
srcPix[4*i] = r
srcPix[4*i+1] = g
srcPix[4*i+2] = b
srcPix[4*i+3] = a
}
src.WritePixels(srcPix)
factors := []ebiten.BlendFactor{
ebiten.BlendFactorZero,
ebiten.BlendFactorOne,
ebiten.BlendFactorSourceColor,
ebiten.BlendFactorOneMinusSourceColor,
ebiten.BlendFactorSourceAlpha,
ebiten.BlendFactorOneMinusSourceAlpha,
ebiten.BlendFactorDestinationColor,
ebiten.BlendFactorOneMinusDestinationColor,
ebiten.BlendFactorDestinationAlpha,
ebiten.BlendFactorOneMinusDestinationAlpha,
}
for _, srcRGBFactor := range factors {
for _, srcAlphaFactor := range factors {
for _, dstRGBFactor := range factors {
for _, dstAlphaFactor := range factors {
// Reset the destination state.
dst.WritePixels(dstPix)
op := &ebiten.DrawImageOptions{}
op.Blend = ebiten.Blend{
BlendFactorSourceRGB: srcRGBFactor,
BlendFactorSourceAlpha: srcAlphaFactor,
BlendFactorDestinationRGB: dstRGBFactor,
BlendFactorDestinationAlpha: dstAlphaFactor,
BlendOperationRGB: ebiten.BlendOperationAdd,
BlendOperationAlpha: ebiten.BlendOperationAdd,
}
dst.DrawImage(src, op)
for i := 0; i < w; i++ {
got := dst.At(i, 0).(color.RGBA)
sr, sg, sb, sa := colorToFloats(srcColor(i))
dr, dg, db, da := colorToFloats(dstColor(i))
var r, g, b, a float64
switch srcRGBFactor {
case ebiten.BlendFactorZero:
r += 0 * sr
g += 0 * sg
b += 0 * sb
case ebiten.BlendFactorOne:
r += 1 * sr
g += 1 * sg
b += 1 * sb
case ebiten.BlendFactorSourceColor:
r += sr * sr
g += sg * sg
b += sb * sb
case ebiten.BlendFactorOneMinusSourceColor:
r += (1 - sr) * sr
g += (1 - sg) * sg
b += (1 - sb) * sb
case ebiten.BlendFactorSourceAlpha:
r += sa * sr
g += sa * sg
b += sa * sb
case ebiten.BlendFactorOneMinusSourceAlpha:
r += (1 - sa) * sr
g += (1 - sa) * sg
b += (1 - sa) * sb
case ebiten.BlendFactorDestinationColor:
r += dr * sr
g += dg * sg
b += db * sb
case ebiten.BlendFactorOneMinusDestinationColor:
r += (1 - dr) * sr
g += (1 - dg) * sg
b += (1 - db) * sb
case ebiten.BlendFactorDestinationAlpha:
r += da * sr
g += da * sg
b += da * sb
case ebiten.BlendFactorOneMinusDestinationAlpha:
r += (1 - da) * sr
g += (1 - da) * sg
b += (1 - da) * sb
}
switch srcAlphaFactor {
case ebiten.BlendFactorZero:
a += 0 * sa
case ebiten.BlendFactorOne:
a += 1 * sa
case ebiten.BlendFactorSourceColor, ebiten.BlendFactorSourceAlpha:
a += sa * sa
case ebiten.BlendFactorOneMinusSourceColor, ebiten.BlendFactorOneMinusSourceAlpha:
a += (1 - sa) * sa
case ebiten.BlendFactorDestinationColor, ebiten.BlendFactorDestinationAlpha:
a += da * sa
case ebiten.BlendFactorOneMinusDestinationColor, ebiten.BlendFactorOneMinusDestinationAlpha:
a += (1 - da) * sa
}
switch dstRGBFactor {
case ebiten.BlendFactorZero:
r += 0 * dr
g += 0 * dg
b += 0 * db
case ebiten.BlendFactorOne:
r += 1 * dr
g += 1 * dg
b += 1 * db
case ebiten.BlendFactorSourceColor:
r += sr * dr
g += sg * dg
b += sb * db
case ebiten.BlendFactorOneMinusSourceColor:
r += (1 - sr) * dr
g += (1 - sg) * dg
b += (1 - sb) * db
case ebiten.BlendFactorSourceAlpha:
r += sa * dr
g += sa * dg
b += sa * db
case ebiten.BlendFactorOneMinusSourceAlpha:
r += (1 - sa) * dr
g += (1 - sa) * dg
b += (1 - sa) * db
case ebiten.BlendFactorDestinationColor:
r += dr * dr
g += dg * dg
b += db * db
case ebiten.BlendFactorOneMinusDestinationColor:
r += (1 - dr) * dr
g += (1 - dg) * dg
b += (1 - db) * db
case ebiten.BlendFactorDestinationAlpha:
r += da * dr
g += da * dg
b += da * db
case ebiten.BlendFactorOneMinusDestinationAlpha:
r += (1 - da) * dr
g += (1 - da) * dg
b += (1 - da) * db
}
switch dstAlphaFactor {
case ebiten.BlendFactorZero:
a += 0 * da
case ebiten.BlendFactorOne:
a += 1 * da
case ebiten.BlendFactorSourceColor, ebiten.BlendFactorSourceAlpha:
a += sa * da
case ebiten.BlendFactorOneMinusSourceColor, ebiten.BlendFactorOneMinusSourceAlpha:
a += (1 - sa) * da
case ebiten.BlendFactorDestinationColor, ebiten.BlendFactorDestinationAlpha:
a += da * da
case ebiten.BlendFactorOneMinusDestinationColor, ebiten.BlendFactorOneMinusDestinationAlpha:
a += (1 - da) * da
}
want := color.RGBA{
R: clamp(int(r * 0xff)),
G: clamp(int(g * 0xff)),
B: clamp(int(b * 0xff)),
A: clamp(int(a * 0xff)),
}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, 0): factors: %d, %d, %d, %d: got: %v, want: %v", i, srcRGBFactor, srcAlphaFactor, dstRGBFactor, dstAlphaFactor, got, want)
}
}
}
}
}
}
}
func TestImageAntiAlias(t *testing.T) {
// This value depends on internal/ui.bigOffscreenScale. Sync this.
const bigOffscreenScale = 2
const w, h = 272, 208
dst0 := ebiten.NewImage(w, h)
dst1 := ebiten.NewImage(w, h)
tmp := ebiten.NewImage(w*bigOffscreenScale, h*bigOffscreenScale)
src := ebiten.NewImage(3, 3)
src.Fill(color.RGBA{R: 0x24, G: 0x3f, B: 0x6a, A: 0x88})
for _, blend := range []ebiten.Blend{
{}, // Default
ebiten.BlendClear,
ebiten.BlendCopy,
ebiten.BlendSourceOver,
ebiten.BlendDestinationOver,
ebiten.BlendXor,
ebiten.BlendLighter,
} {
rnd := rand.New(rand.NewPCG(0, 0))
max := func(x, y, z byte) byte {
if x >= y && x >= z {
return x
}
if y >= x && y >= z {
return y
}
return z
}
dstPix := make([]byte, 4*w*h)
for i := 0; i < w*h; i++ {
n := rnd.Int()
r, g, b := byte(n), byte(n>>8), byte(n>>16)
a := max(r, g, b)
dstPix[4*i] = r
dstPix[4*i+1] = g
dstPix[4*i+2] = b
dstPix[4*i+3] = a
}
dst0.WritePixels(dstPix)
dst1.WritePixels(dstPix)
tmp.Clear()
// Create an actual result.
op := &ebiten.DrawTrianglesOptions{}
op.Blend = blend
op.AntiAlias = true
vs0 := []ebiten.Vertex{
{
DstX: w / 4,
DstY: h / 4,
SrcX: 1,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 2 * w / 4,
DstY: h / 4,
SrcX: 2,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w / 4,
DstY: 2 * h / 4,
SrcX: 1,
SrcY: 2,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint16{0, 1, 2}
dst0.DrawTriangles(vs0, is, src, op)
vs1 := []ebiten.Vertex{
{
DstX: 2 * w / 4,
DstY: 3 * h / 4,
SrcX: 1,
SrcY: 2,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 3 * w / 4,
DstY: 2 * h / 4,
SrcX: 2,
SrcY: 1,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 3 * w / 4,
DstY: 3 * h / 4,
SrcX: 2,
SrcY: 2,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
dst0.DrawTriangles(vs1, is, src, op)
// Create an expected result.
// Copy an enlarged destination image to the offscreen.
opCopy := &ebiten.DrawImageOptions{}
opCopy.GeoM.Scale(bigOffscreenScale, bigOffscreenScale)
opCopy.Blend = ebiten.BlendCopy
tmp.DrawImage(dst1, opCopy)
// Render the vertices onto the offscreen.
for i := range vs0 {
vs0[i].DstX *= 2
vs0[i].DstY *= 2
}
for i := range vs1 {
vs1[i].DstX *= 2
vs1[i].DstY *= 2
}
op = &ebiten.DrawTrianglesOptions{}
op.Blend = blend
tmp.DrawTriangles(vs0, is, src, op)
tmp.DrawTriangles(vs1, is, src, op)
// Render a shrunk offscreen image onto the destination.
opShrink := &ebiten.DrawImageOptions{}
opShrink.GeoM.Scale(1.0/bigOffscreenScale, 1.0/bigOffscreenScale)
opShrink.Filter = ebiten.FilterLinear
opShrink.Blend = ebiten.BlendCopy
dst1.DrawImage(tmp, opShrink)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst0.At(i, j).(color.RGBA)
want := dst1.At(i, j).(color.RGBA)
if !sameColors(got, want, 2) {
t.Errorf("At(%d, %d), blend: %v, got: %v, want: %v", i, j, blend, got, want)
}
}
}
}
}
func TestImageColorMScale(t *testing.T) {
const w, h = 16, 16
dst0 := ebiten.NewImage(w, h)
dst1 := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
src.Fill(color.RGBA{R: 0x24, G: 0x3f, B: 0x6a, A: 0x88})
// As the ColorM is a diagonal matrix, a built-in shader for a color matrix is NOT used.
op := &ebiten.DrawImageOptions{}
op.ColorM.Scale(0.3, 0.4, 0.5, 0.6)
dst0.DrawImage(src, op)
// As the ColorM is not a diagonal matrix, a built-in shader for a color matrix is used.
op = &ebiten.DrawImageOptions{}
op.ColorM.Scale(0.3, 0.4, 0.5, 0.6)
op.ColorM.Translate(0, 0, 0, 1e-4)
dst1.DrawImage(src, op)
got := dst0.At(0, 0)
want := dst1.At(0, 0)
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageColorScaleAndColorM(t *testing.T) {
const w, h = 16, 16
dst0 := ebiten.NewImage(w, h)
dst1 := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
src.Fill(color.RGBA{R: 0x24, G: 0x3f, B: 0x6a, A: 0x88})
// ColorScale is applied to premultiplied-alpha colors.
op := &ebiten.DrawImageOptions{}
op.ColorScale.Scale(0.3*0.6, 0.4*0.6, 0.5*0.6, 0.6)
dst0.DrawImage(src, op)
// ColorM.Scale is applied to straight-alpha colors.
op = &ebiten.DrawImageOptions{}
op.ColorM.Scale(0.3, 0.4, 0.5, 0.6)
dst1.DrawImage(src, op)
got := dst0.At(0, 0)
want := dst1.At(0, 0)
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
// Issue #2428
func TestImageSetAndSubImage(t *testing.T) {
const w, h = 16, 16
img := ebiten.NewImage(w, h)
img.Set(1, 1, color.RGBA{R: 0xff, A: 0xff})
got := img.SubImage(image.Rect(0, 0, w, h)).At(1, 1).(color.RGBA)
want := color.RGBA{R: 0xff, A: 0xff}
if got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
// Issue #2611
func TestImageDrawTrianglesWithGreaterIndexThanVerticesCount(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawTriangles must panic but not")
}
}()
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
src := ebiten.NewImage(w, h)
vs := make([]ebiten.Vertex, 4)
is := []uint16{0, 1, 2, 1, 2, 4}
dst.DrawTriangles(vs, is, src, nil)
}
// Issue #2611
func TestImageDrawTrianglesShaderWithGreaterIndexThanVerticesCount(t *testing.T) {
defer func() {
if r := recover(); r == nil {
t.Errorf("DrawTrianglesShader must panic but not")
}
}()
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
vs := make([]ebiten.Vertex, 4)
is := []uint16{0, 1, 2, 1, 2, 4}
shader, err := ebiten.NewShader([]byte(`
package main
func Fragment(dstPos vec4, srcPos vec2, color vec4) vec4 {
return color
}
`))
if err != nil {
t.Fatalf("could not compile shader: %v", err)
}
dst.DrawTrianglesShader(vs, is, shader, nil)
}
// Issue #2733
func TestImageGeoMAfterDraw(t *testing.T) {
src := ebiten.NewImage(1, 1)
dst := ebiten.NewImageWithOptions(image.Rect(-1, -1, 0, 0), nil)
op0 := &ebiten.DrawImageOptions{}
dst.DrawImage(src, op0)
if x, y := op0.GeoM.Apply(0, 0); x != 0 || y != 0 {
t.Errorf("got: (%0.2f, %0.2f), want: (0, 0)", x, y)
}
s, err := ebiten.NewShader([]byte(`//kage:unit pixels
package main
func Fragment(dstPos vec4, srcPos vec2, color vec4) vec4 {
return vec4(1)
}
`))
if err != nil {
t.Fatal(err)
}
op1 := &ebiten.DrawRectShaderOptions{}
dst.DrawRectShader(1, 1, s, op1)
if x, y := op1.GeoM.Apply(0, 0); x != 0 || y != 0 {
t.Errorf("got: (%0.2f, %0.2f), want: (0, 0)", x, y)
}
}
func TestImageWritePixelAndDispose(t *testing.T) {
const (
w = 16
h = 16
)
img := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for i := range pix {
pix[i] = 0xff
}
img.WritePixels(pix)
img.Dispose()
// Confirm that any pixel information is invalidated after Dispose is called.
if got, want := img.At(0, 0), (color.RGBA{}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageWritePixelAndDeallocate(t *testing.T) {
const (
w = 16
h = 16
)
img := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for i := range pix {
pix[i] = 0xff
}
img.WritePixels(pix)
img.Deallocate()
// Confirm that any pixel information is cleared after Deallocate is called.
if got, want := img.At(0, 0), (color.RGBA{}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
}
func TestImageDrawImageAfterDeallocation(t *testing.T) {
src, _, err := openEbitenImage()
if err != nil {
t.Fatal(err)
return
}
w, h := src.Bounds().Dx(), src.Bounds().Dy()
dst := ebiten.NewImage(w, h)
dst.DrawImage(src, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := src.At(i, j)
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
// Even after deallocating the image, the image is still available.
dst.Deallocate()
dst.DrawImage(src, nil)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := src.At(i, j)
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
// Issue #2798
func TestImageInvalidPremultipliedAlphaColor(t *testing.T) {
// This test checks the rendering result when Set and WritePixels use an invalid premultiplied alpha color.
// The result values are kept and not clamped.
const (
w = 16
h = 16
)
dst := ebiten.NewImage(w, h)
dst.Set(0, 0, color.RGBA{R: 0xff, G: 0xc0, B: 0x80, A: 0x40})
dst.Set(0, 1, color.RGBA{R: 0xff, G: 0xc0, B: 0x80, A: 0x00})
if got, want := dst.At(0, 0).(color.RGBA), (color.RGBA{R: 0xff, G: 0xc0, B: 0x80, A: 0x40}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
if got, want := dst.At(0, 1).(color.RGBA), (color.RGBA{R: 0xff, G: 0xc0, B: 0x80, A: 0x00}); got != want {
t.Errorf("got: %v, want: %v", got, want)
}
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
pix[4*(j*16+i)] = byte(i)
pix[4*(j*16+i)+1] = byte(j)
pix[4*(j*16+i)+2] = 0x80
pix[4*(j*16+i)+3] = byte(i - j)
}
}
dst.WritePixels(pix)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j)
want := color.RGBA{R: byte(i), G: byte(j), B: 0x80, A: byte(i - j)}
if got != want {
t.Errorf("At(%d, %d): got: %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageDrawTriangles32(t *testing.T) {
const w, h = 16, 16
src := ebiten.NewImage(w, h)
dst := ebiten.NewImage(w, h)
pix := make([]byte, 4*w*h)
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
idx := 4 * (j*w + i)
pix[idx] = byte(i) * 0x10
pix[idx+1] = byte(j) * 0x10
pix[idx+2] = 0xff
pix[idx+3] = 0xff
}
}
src.WritePixels(pix)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint32{0, 1, 2, 1, 2, 3}
op := &ebiten.DrawTrianglesOptions{}
dst.DrawTriangles32(vs, is, src, op)
// Even if the indices are modified, this should not affect the rendering result.
for i := range is {
is[i] = 0
}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: byte(i) * 0x10, G: byte(j) * 0x10, B: 0xff, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
}
func TestImageDrawTrianglesShader32(t *testing.T) {
const w, h = 16, 16
dst := ebiten.NewImage(w, h)
vs := []ebiten.Vertex{
{
DstX: 0,
DstY: 0,
SrcX: 0,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: 0,
SrcX: w,
SrcY: 0,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: 0,
DstY: h,
SrcX: 0,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
{
DstX: w,
DstY: h,
SrcX: w,
SrcY: h,
ColorR: 1,
ColorG: 1,
ColorB: 1,
ColorA: 1,
},
}
is := []uint32{0, 1, 2, 1, 2, 3}
op := &ebiten.DrawTrianglesShaderOptions{}
shader, err := ebiten.NewShader([]byte(`//kage:unit pixels
package main
func Fragment(dstPos vec4, srcPos vec2, color vec4) vec4 {
// imageSrcNOrigin is actually not necessary here.
// As no source image is bounded, the source's origin position is (0, 0).
// However, let's use this function for readability.
p := srcPos - imageSrc0Origin()
return vec4(floor(p.x) / 16, floor(p.y) / 16, 1, 1)
}
`))
if err != nil {
t.Fatal(err)
}
dst.DrawTrianglesShader32(vs, is, shader, op)
// Even if the indices are modified, this should not affect the rendering result.
for i := range is {
is[i] = 0
}
for j := 0; j < h; j++ {
for i := 0; i < w; i++ {
got := dst.At(i, j).(color.RGBA)
want := color.RGBA{R: byte(i) * 0x10, G: byte(j) * 0x10, B: 0xff, A: 0xff}
if !sameColors(got, want, 1) {
t.Errorf("dst.At(%d, %d): got %v, want: %v", i, j, got, want)
}
}
}
}

Опубликовать ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://api.gitlife.ru/oschina-mirror/mirrors-ebiten.git
git@api.gitlife.ru:oschina-mirror/mirrors-ebiten.git
oschina-mirror
mirrors-ebiten
mirrors-ebiten
main