X-Git-Url: http://review.tizen.org/git/?a=blobdiff_plain;f=automated-tests%2Fsrc%2Fdali-toolkit%2Futc-Dali-ImageView.cpp;h=60d155179e0abcb10e3ea9a701938e2c334e83b0;hb=bcd234fc96035a85ab1a116c4ef8dc68ce3885cf;hp=823c2c20c67809fbf21ca50624467c2d00e731f8;hpb=ded68bd82415ca96ae7ec9926cdd3cda8bc05707;p=platform%2Fcore%2Fuifw%2Fdali-toolkit.git diff --git a/automated-tests/src/dali-toolkit/utc-Dali-ImageView.cpp b/automated-tests/src/dali-toolkit/utc-Dali-ImageView.cpp index 823c2c2..60d1551 100644 --- a/automated-tests/src/dali-toolkit/utc-Dali-ImageView.cpp +++ b/automated-tests/src/dali-toolkit/utc-Dali-ImageView.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2022 Samsung Electronics Co., Ltd. + * Copyright (c) 2024 Samsung Electronics Co., Ltd. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,6 +23,7 @@ #include #include +#include #include "dummy-control.h" #include @@ -33,6 +34,7 @@ #include #include #include +#include #include #include #include @@ -58,6 +60,7 @@ namespace const char* TEST_IMAGE_FILE_NAME = "gallery_image_01.jpg"; const char* TEST_IMAGE_FILE_NAME2 = "gallery_image_02.jpg"; +// resolution: 1024*1024 const char* TEST_IMAGE_1 = TEST_RESOURCE_DIR "/TB-gloss.png"; const char* TEST_IMAGE_2 = TEST_RESOURCE_DIR "/tb-norm.png"; @@ -68,6 +71,8 @@ const char* TEST_BROKEN_IMAGE_L = TEST_RESOURCE_DIR "/broken_l.9.png"; const char* TEST_BROKEN_IMAGE_01 = TEST_RESOURCE_DIR "/button-up.9.png"; const char* TEST_BROKEN_IMAGE_02 = TEST_RESOURCE_DIR "/heartsframe.9.png"; +const char* TEST_INVALID_NPATCH_FILE_NAME_01 = "invalid1.9.png"; + // resolution: 34*34, pixel format: RGBA8888 static const char* gImage_34_RGBA = TEST_RESOURCE_DIR "/icon-edit.png"; // resolution: 600*600, pixel format: RGB888 @@ -76,7 +81,11 @@ static const char* gImage_600_RGB = TEST_RESOURCE_DIR "/test-image-600.jpg"; // resolution: 50*50, frame count: 4, frame delay: 0.2 second for each frame const char* TEST_GIF_FILE_NAME = TEST_RESOURCE_DIR "/anim.gif"; -const char* TEST_VECTOR_IMAGE_FILE_NAME = TEST_RESOURCE_DIR "/insta_camera.json"; +const char* TEST_SVG_FILE_NAME = TEST_RESOURCE_DIR "/svg1.svg"; +const char* TEST_ANIMATED_VECTOR_IMAGE_FILE_NAME = TEST_RESOURCE_DIR "/insta_camera.json"; +const char* TEST_WEBP_FILE_NAME = TEST_RESOURCE_DIR "/dali-logo.webp"; + +const char* TEST_OVERWRITABLE_IMAGE_FILE_NAME = TEST_RESOURCE_DIR "/overwritable-image.jpg"; void TestUrl(ImageView imageView, const std::string url) { @@ -87,6 +96,38 @@ void TestUrl(ImageView imageView, const std::string url) DALI_TEST_EQUALS(urlActual, url, TEST_LOCATION); } +void OverwriteImage(const char* sourceFilename) +{ + FILE* fpOut = fopen(TEST_OVERWRITABLE_IMAGE_FILE_NAME, "wb"); + DALI_TEST_CHECK(fpOut); + if(fpOut) + { + FILE* fpIn = fopen(sourceFilename, "rb"); + if(fpIn) + { + fseek(fpIn, 0, SEEK_END); + size_t size = ftell(fpIn); + + tet_printf("Open %s success! file size : %zu byte\n", sourceFilename, size); + Dali::Vector data; + data.Resize(size); + fseek(fpIn, 0, SEEK_SET); + size_t realSize = fread(data.Begin(), sizeof(uint8_t), size, fpIn); + fclose(fpIn); + data.Resize(realSize); + + // Overwrite + fwrite(data.Begin(), sizeof(uint8_t), size, fpOut); + } + else + { + tet_printf("Open %s failed! write invalid\n", sourceFilename); + fprintf(fpOut, "invalid\n"); + } + fclose(fpOut); + } +} + } // namespace int UtcDaliImageViewNewP(void) @@ -569,16 +610,8 @@ int UtcDaliImageViewAsyncLoadingWithAtlasing(void) // loading is not started if the actor is offScene application.GetScene().Add(imageView); - application.SendNotification(); - application.Render(16); - application.Render(16); - application.SendNotification(); imageView.SetProperty(Dali::Actor::Property::LAYOUT_DIRECTION, Dali::LayoutDirection::RIGHT_TO_LEFT); - application.SendNotification(); - application.Render(16); - application.Render(16); - application.SendNotification(); // loading started, this waits for the loader thread for max 30 seconds DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); @@ -616,10 +649,6 @@ int UtcDaliImageViewAsyncLoadingWithAtlasing02(void) imageView.SetProperty(ImageView::Property::IMAGE, asyncLoadingMap); application.GetScene().Add(imageView); - application.SendNotification(); - application.Render(16); - application.Render(16); - application.SendNotification(); // loading started, this waits for the loader thread for max 30 seconds DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); @@ -781,9 +810,25 @@ int UtcDaliImageViewAsyncLoadingEncodedBufferWithAtlasing(void) Property::Map imageMap; imageMap[ImageVisual::Property::URL] = url.GetUrl(); + imageMap[ImageVisual::Property::DESIRED_HEIGHT] = 600; + imageMap[ImageVisual::Property::DESIRED_WIDTH] = 600; + imageMap[ImageVisual::Property::ATLASING] = true; + + // No atlasing with big image + ImageView imageView_bigdesired = ImageView::New(); + imageView_bigdesired.SetProperty(ImageView::Property::IMAGE, imageMap); + imageView_bigdesired.SetProperty(Toolkit::Control::Property::PADDING, Extents(10u, 10u, 10u, 10u)); + + imageMap[ImageVisual::Property::DESIRED_HEIGHT] = 0; + imageMap[ImageVisual::Property::DESIRED_WIDTH] = 0; + + // No atlasing with zero desired size + ImageView imageView_nodesired = ImageView::New(); + imageView_nodesired.SetProperty(ImageView::Property::IMAGE, imageMap); + imageView_nodesired.SetProperty(Toolkit::Control::Property::PADDING, Extents(10u, 10u, 10u, 10u)); + imageMap[ImageVisual::Property::DESIRED_HEIGHT] = 34; imageMap[ImageVisual::Property::DESIRED_WIDTH] = 34; - imageMap[ImageVisual::Property::ATLASING] = true; ImageView imageView = ImageView::New(); imageView.SetProperty(ImageView::Property::IMAGE, imageMap); @@ -791,10 +836,15 @@ int UtcDaliImageViewAsyncLoadingEncodedBufferWithAtlasing(void) // By default, Aysnc loading is used // loading is not started if the actor is offScene - application.GetScene().Add(imageView); + application.GetScene().Add(imageView_bigdesired); + application.GetScene().Add(imageView_nodesired); application.SendNotification(); application.Render(16); + + // loading started, this waits for the loader thread for max 30 seconds + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.Render(16); application.SendNotification(); @@ -803,10 +853,6 @@ int UtcDaliImageViewAsyncLoadingEncodedBufferWithAtlasing(void) imageView.SetProperty(ImageView::Property::IMAGE, imageMap); imageView.SetProperty(Dali::Actor::Property::LAYOUT_DIRECTION, Dali::LayoutDirection::RIGHT_TO_LEFT); - application.SendNotification(); - application.Render(16); - application.Render(16); - application.SendNotification(); // loading started, this waits for the loader thread for max 30 seconds DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); @@ -845,6 +891,7 @@ int UtcDaliImageViewSyncLoadingEncodedBuffer(void) // Sync loading is used Property::Map syncLoadingMap; syncLoadingMap["url"] = url.GetUrl(); + syncLoadingMap["alphaMaskUrl"] = gImage_34_RGBA; syncLoadingMap["desiredHeight"] = 34; syncLoadingMap["desiredWidth"] = 34; syncLoadingMap["synchronousLoading"] = true; @@ -866,6 +913,126 @@ int UtcDaliImageViewSyncLoadingEncodedBuffer(void) END_TEST; } +int UtcDaliImageViewEncodedBufferWithSvg(void) +{ + ToolkitTestApplication application; + TestGlAbstraction& gl = application.GetGlAbstraction(); + const std::vector& textures = gl.GetBoundTextures(); + size_t numTextures = textures.size(); + + // Get encoded raw-buffer svg image and generate url + EncodedImageBuffer buffer = ConvertFileToEncodedImageBuffer(TEST_SVG_FILE_NAME, EncodedImageBuffer::ImageType::VECTOR_IMAGE); + ImageUrl url = Toolkit::Image::GenerateUrl(buffer); + + // Async loading, no atlasing for big size image + ImageView imageView = ImageView::New(url.GetUrl()); + + // By default, Aysnc loading is used + application.GetScene().Add(imageView); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100, 100)); + imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::CENTER); + + application.SendNotification(); + application.Render(16); + + // Load svg image + rasterize. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + application.SendNotification(); + + const std::vector& textures2 = gl.GetBoundTextures(); + DALI_TEST_GREATER(textures2.size(), numTextures, TEST_LOCATION); + + // Remove visual, for line coverage. + imageView.Unparent(); + application.SendNotification(); + application.Render(16); + + END_TEST; +} + +int UtcDaliImageViewEncodedBufferWithAnimatedVectorImage(void) +{ + ToolkitTestApplication application; + TestGlAbstraction& gl = application.GetGlAbstraction(); + const std::vector& textures = gl.GetBoundTextures(); + size_t numTextures = textures.size(); + + // Get encoded raw-buffer lottie image and generate url + EncodedImageBuffer buffer = ConvertFileToEncodedImageBuffer(TEST_ANIMATED_VECTOR_IMAGE_FILE_NAME, EncodedImageBuffer::ImageType::ANIMATED_VECTOR_IMAGE); + ImageUrl url = Toolkit::Image::GenerateUrl(buffer); + + // Async loading, no atlasing for big size image + ImageView imageView = ImageView::New(url.GetUrl()); + + // By default, Aysnc loading is used + application.GetScene().Add(imageView); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100, 100)); + imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::CENTER); + + application.SendNotification(); + application.Render(16); + + // Load lottie image is sync. Only wait rasterize. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + application.SendNotification(); + + const std::vector& textures2 = gl.GetBoundTextures(); + DALI_TEST_GREATER(textures2.size(), numTextures, TEST_LOCATION); + + // Remove visual, for line coverage. + imageView.Unparent(); + application.SendNotification(); + application.Render(16); + + END_TEST; +} + +int UtcDaliImageViewEncodedBufferWithInvalidImageType(void) +{ + ToolkitTestApplication application; + TestGlAbstraction& gl = application.GetGlAbstraction(); + const std::vector& textures = gl.GetBoundTextures(); + size_t numTextures = textures.size(); + + // Get encoded raw-buffer jpg image with invalid image type, and generate url + EncodedImageBuffer buffer = ConvertFileToEncodedImageBuffer(gImage_34_RGBA, static_cast(-1)); + ImageUrl url = Toolkit::Image::GenerateUrl(buffer); + + // Async loading, no atlasing for big size image + ImageView imageView = ImageView::New(url.GetUrl()); + + // By default, Aysnc loading is used + application.GetScene().Add(imageView); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100, 100)); + imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::CENTER); + + application.SendNotification(); + application.Render(16); + + // Load image + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + application.SendNotification(); + + const std::vector& textures2 = gl.GetBoundTextures(); + DALI_TEST_GREATER(textures2.size(), numTextures, TEST_LOCATION); + + // Remove visual, for line coverage. + imageView.Unparent(); + application.SendNotification(); + application.Render(16); + + END_TEST; +} + int UtcDaliImageViewAddedTexture(void) { ToolkitTestApplication application; @@ -1025,6 +1192,23 @@ void ResourceReadySignal(Control control) gResourceReadySignalFired = true; } +void OnResourceReadySignalSVG(Control control) +{ + // Check whether Image Visual transforms on ImageVieiw::OnRelayout() + Toolkit::Internal::Control& controlImpl = Toolkit::Internal::GetImplementation(control); + Toolkit::Visual::Base imageVisual = DevelControl::GetVisual(controlImpl, ImageView::Property::IMAGE); + Property::Map resultMap; + imageVisual.CreatePropertyMap(resultMap); + + Property::Value* transformValue = resultMap.Find(Visual::Property::TRANSFORM); + DALI_TEST_CHECK(transformValue); + Property::Map* retMap = transformValue->GetMap(); + DALI_TEST_CHECK(retMap); + + // Fitting mode should not be applied at this point + DALI_TEST_EQUALS(retMap->Find(Visual::Transform::Property::SIZE)->Get(), Vector2::ZERO, TEST_LOCATION); +} + int UtcDaliImageViewCheckResourceReady(void) { ToolkitTestApplication application; @@ -1247,6 +1431,74 @@ int UtcDaliImageViewReplaceImage(void) END_TEST; } +int UtcDaliImageViewReloadAlphaMaskImage(void) +{ + ToolkitTestApplication application; + + gResourceReadySignalFired = false; + + ImageView dummy = ImageView::New(); + ImageView imageView = ImageView::New(); + Property::Map propertyMap; + + // To keep alpha mask cached, scene on some dummy image. + // Note : If we don't cache alpha mask image, the reference count of mask image become zero. + // In this case, we might need to wait mask image loading, which is not neccesary & can be changed behavior. + propertyMap[ImageVisual::Property::URL] = gImage_600_RGB; + propertyMap[ImageVisual::Property::ALPHA_MASK_URL] = TEST_BROKEN_IMAGE_DEFAULT; + dummy.SetProperty(ImageView::Property::IMAGE, propertyMap); + + application.GetScene().Add(dummy); + + application.SendNotification(); + application.Render(16); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(3), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + propertyMap.Clear(); + propertyMap[ImageVisual::Property::URL] = gImage_34_RGBA; + propertyMap[ImageVisual::Property::ALPHA_MASK_URL] = TEST_BROKEN_IMAGE_DEFAULT; + imageView.SetProperty(ImageView::Property::IMAGE, propertyMap); + + DALI_TEST_EQUALS(imageView.IsResourceReady(), false, TEST_LOCATION); + + imageView.ResourceReadySignal().Connect(&ResourceReadySignal); + + application.GetScene().Add(imageView); + + application.SendNotification(); + application.Render(16); + + // Load image and use cached mask. Now we try to apply masking. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + DALI_TEST_EQUALS(gResourceReadySignalFired, false, TEST_LOCATION); + + // Cancel apply masking. + imageView.Unparent(); + + application.SendNotification(); + application.Render(16); + + // Reload same image again. + application.GetScene().Add(imageView); + + application.SendNotification(); + application.Render(16); + + // Finish apply masking. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + + END_TEST; +} + void OnRelayoutOverride(Size size) { gNaturalSize = size; // Size Relayout is using @@ -2359,7 +2611,7 @@ int UtcDaliImageViewFittingModesWithAnimatedVectorImageVisual(void) ImageView imageView = ImageView::New(); Property::Map imageMap; imageMap.Add(Toolkit::Visual::Property::TYPE, DevelVisual::ANIMATED_VECTOR_IMAGE); - imageMap.Add(Toolkit::ImageVisual::Property::URL, TEST_VECTOR_IMAGE_FILE_NAME); // 249x169 image + imageMap.Add(Toolkit::ImageVisual::Property::URL, TEST_ANIMATED_VECTOR_IMAGE_FILE_NAME); // 249x169 image imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, imageMap); imageView.SetProperty(Actor::Property::SIZE, Vector2(600, 600)); @@ -2599,10 +2851,6 @@ void ReloadImage(ImageView imageView) void ResourceFailedReload(Control control) { gFailCounter++; - if(gFailCounter < MAX_RETRIES) - { - ReloadImage(ImageView::DownCast(control)); - } } } // namespace @@ -2621,17 +2869,14 @@ int UtcDaliImageViewReloadFailedOnResourceReadySignal(void) // loading started, this waits for the loader thread to complete DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); - application.SendNotification(); - DALI_TEST_EQUALS(gFailCounter, 1, TEST_LOCATION); + ReloadImage(imageView); DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); - application.SendNotification(); - DALI_TEST_EQUALS(gFailCounter, 2, TEST_LOCATION); + ReloadImage(imageView); DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); - application.SendNotification(); DALI_TEST_EQUALS(gFailCounter, 3, TEST_LOCATION); END_TEST; @@ -2642,96 +2887,190 @@ int UtcDaliImageViewLoadRemoteSVG(void) tet_infoline("Test load from a remote server."); ToolkitTestApplication application; - Toolkit::ImageView imageView; - imageView = Toolkit::ImageView::New(); - imageView.SetImage("https://dev.w3.org/SVG/tools/svgweb/samples/svg-files/check.svg"); - // Victor. Temporary (or permanent?) update as the url above seems not to work from time to time ... - // imageView.SetImage("https://upload.wikimedia.org/wikipedia/commons/thumb/0/02/SVG_logo.svg/64px-SVG_logo.svg.png"); - imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::TOP_LEFT); - imageView.SetProperty(Actor::Property::ANCHOR_POINT, AnchorPoint::TOP_LEFT); - imageView.SetProperty(Actor::Property::SIZE, Vector2(300, 300)); - imageView.SetProperty(Actor::Property::POSITION, Vector3(150.0f, 150.0f, 0.0f)); - application.GetScene().Add(imageView); + { + Toolkit::ImageView imageView; + imageView = Toolkit::ImageView::New(); + imageView.SetImage("https://dev.w3.org/SVG/tools/svgweb/samples/svg-files/check.svg"); + imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::TOP_LEFT); + imageView.SetProperty(Actor::Property::ANCHOR_POINT, AnchorPoint::TOP_LEFT); + imageView.SetProperty(Actor::Property::SIZE, Vector2(300, 300)); + imageView.SetProperty(Actor::Property::POSITION, Vector3(150.0f, 150.0f, 0.0f)); - DALI_TEST_CHECK(imageView); + application.GetScene().Add(imageView); - DALI_TEST_EQUALS(imageView.GetRendererCount(), 0u, TEST_LOCATION); + DALI_TEST_CHECK(imageView); - application.SendNotification(); + DALI_TEST_EQUALS(imageView.GetRendererCount(), 0u, TEST_LOCATION); - DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); - application.SendNotification(); - application.Render(); + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); - DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + application.SendNotification(); + application.Render(); - END_TEST; -} + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + } -int UtcDaliImageViewSyncSVGLoading(void) -{ - ToolkitTestApplication application; + // Without size set + { + Toolkit::ImageView imageView; + imageView = Toolkit::ImageView::New(); + imageView.SetImage("https://dev.w3.org/SVG/tools/svgweb/samples/svg-files/check.svg"); + imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::TOP_LEFT); + imageView.SetProperty(Actor::Property::ANCHOR_POINT, AnchorPoint::TOP_LEFT); + imageView.SetProperty(Actor::Property::POSITION, Vector3(150.0f, 150.0f, 0.0f)); - tet_infoline("ImageView Testing SVG image sync loading"); + application.GetScene().Add(imageView); - // Sync loading, automatic atlasing for small size image - { - TraceCallStack& callStack = application.GetGlAbstraction().GetTextureTrace(); - callStack.Reset(); - callStack.Enable(true); + DALI_TEST_CHECK(imageView); - ImageView imageView = ImageView::New(); + DALI_TEST_EQUALS(imageView.GetRendererCount(), 0u, TEST_LOCATION); - // Sync loading is used - Property::Map syncLoadingMap; - syncLoadingMap.Insert(Toolkit::Visual::Property::TYPE, Toolkit::Visual::IMAGE); - syncLoadingMap.Insert(Toolkit::ImageVisual::Property::URL, TEST_RESOURCE_DIR "/svg1.svg"); - syncLoadingMap.Insert(Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING, true); - imageView.SetProperty(ImageView::Property::IMAGE, syncLoadingMap); + application.SendNotification(); - application.GetScene().Add(imageView); - DALI_TEST_CHECK(imageView); + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); application.SendNotification(); - application.Render(16); - Vector3 naturalSize = imageView.GetNaturalSize(); + application.Render(); - DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); - DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); } + END_TEST; } -int UtcDaliImageViewAsyncSVGLoading(void) +int UtcDaliImageViewLoadRemoteLottie(void) { + tet_infoline("Test load from a remote server. (Note we don't support real download now. Just for line coverage)"); + ToolkitTestApplication application; - tet_infoline("ImageView Testing SVG image async loading"); + { + Toolkit::ImageView imageView; + imageView = Toolkit::ImageView::New(); + imageView.SetImage("https://lottie.json"); + imageView.SetProperty(Actor::Property::PARENT_ORIGIN, ParentOrigin::TOP_LEFT); + imageView.SetProperty(Actor::Property::ANCHOR_POINT, AnchorPoint::TOP_LEFT); + imageView.SetProperty(Actor::Property::SIZE, Vector2(300, 300)); + imageView.SetProperty(Actor::Property::POSITION, Vector3(150.0f, 150.0f, 0.0f)); + + application.GetScene().Add(imageView); + + DALI_TEST_CHECK(imageView); + + application.SendNotification(); + application.Render(); + + // Do not check anything for here. + } + + END_TEST; +} + +int UtcDaliImageViewSyncSVGLoading(void) +{ + ToolkitTestApplication application; + + tet_infoline("ImageView Testing SVG image sync loading"); - // Sync loading, automatic atlasing for small size image { - TraceCallStack& callStack = application.GetGlAbstraction().GetTextureTrace(); - callStack.Reset(); - callStack.Enable(true); + ImageView imageView = ImageView::New(); + + // Sync loading is used + Property::Map syncLoadingMap; + syncLoadingMap.Insert(Toolkit::Visual::Property::TYPE, Toolkit::Visual::IMAGE); + syncLoadingMap.Insert(Toolkit::ImageVisual::Property::URL, TEST_RESOURCE_DIR "/svg1.svg"); + syncLoadingMap.Insert(Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING, true); + imageView.SetProperty(ImageView::Property::IMAGE, syncLoadingMap); + + application.GetScene().Add(imageView); + DALI_TEST_CHECK(imageView); + + application.SendNotification(); + Vector3 naturalSize = imageView.GetNaturalSize(); + + DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); + DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); + } + END_TEST; +} +int UtcDaliImageViewSyncSVGLoading02(void) +{ + ToolkitTestApplication application; + + tet_infoline("ImageView Testing SVG image sync loading"); + + { ImageView imageView = ImageView::New(); // Sync loading is used Property::Map syncLoadingMap; syncLoadingMap.Insert(Toolkit::Visual::Property::TYPE, Toolkit::Visual::IMAGE); syncLoadingMap.Insert(Toolkit::ImageVisual::Property::URL, TEST_RESOURCE_DIR "/svg1.svg"); - syncLoadingMap.Insert(Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING, false); + syncLoadingMap.Insert(Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING, true); + syncLoadingMap.Insert(DevelVisual::Property::VISUAL_FITTING_MODE, Toolkit::DevelVisual::FIT_KEEP_ASPECT_RATIO); imageView.SetProperty(ImageView::Property::IMAGE, syncLoadingMap); + imageView.ResourceReadySignal().Connect(&OnResourceReadySignalSVG); application.GetScene().Add(imageView); DALI_TEST_CHECK(imageView); application.SendNotification(); - application.Render(16); + application.Render(); + + // Check whether Image Visual transforms on ImageVieiw::OnRelayout() + Toolkit::Internal::Control& controlImpl = Toolkit::Internal::GetImplementation(imageView); + Toolkit::Visual::Base imageVisual = DevelControl::GetVisual(controlImpl, ImageView::Property::IMAGE); + Property::Map resultMap; + imageVisual.CreatePropertyMap(resultMap); + + Property::Value* transformValue = resultMap.Find(Visual::Property::TRANSFORM); + DALI_TEST_CHECK(transformValue); + Property::Map* retMap = transformValue->GetMap(); + DALI_TEST_CHECK(retMap); + + // Image Visual should be positioned depending on ImageView's padding + DALI_TEST_EQUALS(retMap->Find(Visual::Transform::Property::SIZE)->Get(), Vector2(100, 100), TEST_LOCATION); + Vector3 naturalSize = imageView.GetNaturalSize(); + DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); + DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); + } + END_TEST; +} + +int UtcDaliImageViewAsyncSVGLoading(void) +{ + ToolkitTestApplication application; + + tet_infoline("ImageView Testing SVG image async loading"); + + { + ImageView imageView = ImageView::New(); + + // Async loading is used - default value of SYNCHRONOUS_LOADING is false. + Property::Map propertyMap; + propertyMap.Insert(Toolkit::Visual::Property::TYPE, Toolkit::Visual::IMAGE); + propertyMap.Insert(Toolkit::ImageVisual::Property::URL, TEST_RESOURCE_DIR "/svg1.svg"); + imageView.SetProperty(ImageView::Property::IMAGE, propertyMap); + + application.GetScene().Add(imageView); + DALI_TEST_CHECK(imageView); + + application.SendNotification(); + + // Wait for loading + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + Vector3 naturalSize = imageView.GetNaturalSize(); DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); } @@ -2744,12 +3083,8 @@ int UtcDaliImageViewSVGLoadingSyncSetInvalidValue(void) tet_infoline("ImageView Testing SVG image async loading"); - // Sync loading, automatic atlasing for small size image + // Sync loading { - TraceCallStack& callStack = application.GetGlAbstraction().GetTextureTrace(); - callStack.Reset(); - callStack.Enable(true); - ImageView imageView = ImageView::New(); // Sync loading is used @@ -2768,7 +3103,13 @@ int UtcDaliImageViewSVGLoadingSyncSetInvalidValue(void) DALI_TEST_CHECK(imageView); application.SendNotification(); + + // Wait for loading + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); application.Render(16); + Vector3 naturalSize = imageView.GetNaturalSize(); DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); @@ -2784,12 +3125,16 @@ int UtcDaliImageViewSVGLoadingSyncSetInvalidValue(void) END_TEST; } -int UtcDaliImageViewSvgLoadingFailure(void) +int UtcDaliImageViewSvgLoadingFailureLocalFile(void) { - ToolkitTestApplication application; - // Local svg file - invalid file path { + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + gResourceReadySignalFired = false; ImageView imageView = ImageView::New(TEST_RESOURCE_DIR "/foo.svg"); @@ -2801,16 +3146,67 @@ int UtcDaliImageViewSvgLoadingFailure(void) application.GetScene().Add(imageView); application.SendNotification(); + + // loading started, this waits for the loader thread - load + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + // Should be shown a broken image + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(textureTrace.FindMethod("BindTexture"), true, TEST_LOCATION); + } + + // Local svg file - invalid file path without size set + { + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + + gResourceReadySignalFired = false; + textureTrace.Reset(); + + ImageView imageView = ImageView::New(TEST_RESOURCE_DIR "/foo.svg"); + imageView.ResourceReadySignal().Connect(&ResourceReadySignal); + + DALI_TEST_EQUALS(imageView.IsResourceReady(), false, TEST_LOCATION); + + application.GetScene().Add(imageView); + + application.SendNotification(); + + // loading started, this waits for the loader thread - load & rasterize + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); application.Render(16); DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + // Should be shown a broken image + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(textureTrace.FindMethod("BindTexture"), true, TEST_LOCATION); } // Local svg file - invalid file { + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + gResourceReadySignalFired = false; + textureTrace.Reset(); ImageView imageView = ImageView::New(TEST_RESOURCE_DIR "/invalid.svg"); imageView.SetProperty(Actor::Property::SIZE, Vector2(200.f, 200.f)); @@ -2821,18 +3217,38 @@ int UtcDaliImageViewSvgLoadingFailure(void) application.GetScene().Add(imageView); application.SendNotification(); + + // loading started, this waits for the loader thread - load & rasterize + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); application.Render(16); DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + // Should be shown a broken image + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(textureTrace.FindMethod("BindTexture"), true, TEST_LOCATION); } + END_TEST; +} + +int UtcDaliImageViewSvgLoadingFailureRemoteFile01(void) +{ // Remote svg file { + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + gResourceReadySignalFired = false; - ImageView imageView = ImageView::New("https://bar.org/foobar.svg"); + ImageView imageView = ImageView::New("https://127.0.0.1/foobar.svg"); imageView.SetProperty(Actor::Property::SIZE, Vector2(200.f, 200.f)); imageView.ResourceReadySignal().Connect(&ResourceReadySignal); @@ -2842,7 +3258,46 @@ int UtcDaliImageViewSvgLoadingFailure(void) application.SendNotification(); - // loading started, this waits for the loader thread + // loading started, this waits for the loader thread - load & rasterize + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + // Should be shown a broken image + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(textureTrace.FindMethod("BindTexture"), true, TEST_LOCATION); + } + + END_TEST; +} + +int UtcDaliImageViewSvgLoadingFailureRemoteFile02(void) +{ + // Remote svg file without size set + { + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + + gResourceReadySignalFired = false; + + ImageView imageView = ImageView::New("https://127.0.0.1/foobar.svg"); + imageView.ResourceReadySignal().Connect(&ResourceReadySignal); + + DALI_TEST_EQUALS(imageView.IsResourceReady(), false, TEST_LOCATION); + + application.GetScene().Add(imageView); + + application.SendNotification(); + + // loading started, this waits for the loader thread - load & rasterize DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); application.SendNotification(); @@ -2851,6 +3306,10 @@ int UtcDaliImageViewSvgLoadingFailure(void) DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + // Should be shown a broken image + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(textureTrace.FindMethod("BindTexture"), true, TEST_LOCATION); } END_TEST; @@ -2862,7 +3321,11 @@ int UtcDaliImageViewSvgRasterizationFailure(void) gResourceReadySignalFired = false; - ImageView imageView = ImageView::New(TEST_RESOURCE_DIR "/svg1.svg"); + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + + ImageView imageView = ImageView::New(TEST_RESOURCE_DIR "/invalid1.svg"); imageView.SetProperty(Actor::Property::SIZE, Vector2(200.f, 200.f)); imageView.ResourceReadySignal().Connect(&ResourceReadySignal); @@ -2872,91 +3335,232 @@ int UtcDaliImageViewSvgRasterizationFailure(void) application.SendNotification(); - // loading started, this waits for the loader thread - DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); application.SendNotification(); application.Render(16); DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); - DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::READY, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); - // Reset flag - gResourceReadySignalFired = false; + // Should be shown a broken image + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(textureTrace.FindMethod("BindTexture"), true, TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewSvgChageSize(void) +{ + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + + ImageView imageView = ImageView::New(TEST_SVG_FILE_NAME); + application.GetScene().Add(imageView); + + application.SendNotification(); + + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); - // Change size - imageView.SetProperty(Actor::Property::SIZE, Vector2(0.f, 0.f)); + application.SendNotification(); + application.Render(16); + + DALI_TEST_EQUALS(Test::VectorImageRenderer::GetLoadCount(), 1, TEST_LOCATION); + + // Change actor size, then rasterization should be done again + imageView.SetProperty(Actor::Property::SIZE, Vector2(200.f, 200.f)); application.SendNotification(); - // rasterization started, this waits for the rasterize thread + // Wait for rasterization DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); application.SendNotification(); application.Render(16); - DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); - DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); - // Fail to rasterize because the size is 0. - DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + // We should not load the file again. + DALI_TEST_EQUALS(Test::VectorImageRenderer::GetLoadCount(), 1, TEST_LOCATION); END_TEST; } -int UtcDaliImageViewTVGLoading(void) +int UtcDaliImageViewSvgAtlasing(void) { ToolkitTestApplication application; - tet_infoline("ImageView Testing TVG image loading"); + TraceCallStack& callStack = application.GetGlAbstraction().GetTextureTrace(); + callStack.Reset(); + callStack.Enable(true); - { - ImageView imageView = ImageView::New(); + Property::Map propertyMap; + propertyMap["url"] = TEST_SVG_FILE_NAME; + propertyMap["atlasing"] = true; - imageView.SetImage(TEST_RESOURCE_DIR "/test.tvg"); + ImageView imageView = ImageView::New(); + imageView.SetProperty(ImageView::Property::IMAGE, propertyMap); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.f, 100.f)); + application.GetScene().Add(imageView); - application.GetScene().Add(imageView); - DALI_TEST_CHECK(imageView); - Vector3 naturalSize = imageView.GetNaturalSize(); + application.SendNotification(); - DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); - DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); - } - END_TEST; -} -int UtcDaliImageViewImageLoadFailure01(void) -{ - ToolkitTestApplication application; + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); - Toolkit::StyleManager styleManager = Toolkit::StyleManager::Get(); - DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::SMALL, TEST_BROKEN_IMAGE_S); - DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::NORMAL, TEST_BROKEN_IMAGE_M); - DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::LARGE, TEST_BROKEN_IMAGE_L); + application.SendNotification(); + application.Render(16); - std::string brokenUrl; - brokenUrl = DevelStyleManager::GetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::SMALL); - DALI_TEST_EQUALS(TEST_BROKEN_IMAGE_S, brokenUrl, TEST_LOCATION); + // use atlas + TraceCallStack::NamedParams params1; + params1["width"] << 100; + params1["height"] << 100; + DALI_TEST_EQUALS(callStack.FindMethodAndParams("TexSubImage2D", params1), true, TEST_LOCATION); - brokenUrl = DevelStyleManager::GetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::NORMAL); - DALI_TEST_EQUALS(TEST_BROKEN_IMAGE_M, brokenUrl, TEST_LOCATION); + imageView.SetProperty(Actor::Property::SIZE, Vector2(600.f, 600.f)); - brokenUrl = DevelStyleManager::GetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::LARGE); - DALI_TEST_EQUALS(TEST_BROKEN_IMAGE_L, brokenUrl, TEST_LOCATION); + application.SendNotification(); - ImageView imageView = ImageView::New("invalidUrl.png"); - imageView.SetProperty(Actor::Property::SIZE, Vector2(200.f, 200.f)); + // Wait for rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + callStack.Reset(); - application.GetScene().Add(imageView); application.SendNotification(); application.Render(16); - // loading started, this waits for the loader thread - DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + // not use atlas + TraceCallStack::NamedParams params2; + params2["width"] << 600; + params2["height"] << 600; + DALI_TEST_EQUALS(callStack.FindMethodAndParams("TexImage2D", params2), true, TEST_LOCATION); END_TEST; } -int UtcDaliImageViewImageLoadFailure02(void) +int UtcDaliImageViewTVGLoading(void) +{ + ToolkitTestApplication application; + + tet_infoline("ImageView Testing TVG image loading"); + + { + ImageView imageView = ImageView::New(TEST_RESOURCE_DIR "/test.tvg"); + application.GetScene().Add(imageView); + DALI_TEST_CHECK(imageView); + + application.SendNotification(); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + Vector3 naturalSize = imageView.GetNaturalSize(); + + DALI_TEST_EQUALS(naturalSize.width, 100.0f, TEST_LOCATION); + DALI_TEST_EQUALS(naturalSize.height, 100.0f, TEST_LOCATION); + } + END_TEST; +} + +int UtcDaliImageViewSvgDesiredSize01(void) +{ + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + + int desiredWidth = 100, desiredHeight = 150; + ImageView imageView = ImageView::New(TEST_SVG_FILE_NAME, ImageDimensions(desiredWidth, desiredHeight)); + + application.GetScene().Add(imageView); + + application.SendNotification(); + + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + { + std::stringstream out; + out << GL_TEXTURE_2D << ", " << 0u << ", " << desiredWidth << ", " << desiredHeight; + DALI_TEST_CHECK(textureTrace.FindMethodAndParams("TexImage2D", out.str().c_str())); + } + + END_TEST; +} + +int UtcDaliImageViewSvgDesiredSize02(void) +{ + ToolkitTestApplication application; + + TestGlAbstraction& gl = application.GetGlAbstraction(); + TraceCallStack& textureTrace = gl.GetTextureTrace(); + textureTrace.Enable(true); + + int desiredWidth = 150, desiredHeight = 100; + ImageView imageView = ImageView::New(); + imageView[ImageView::Property::IMAGE] = Property::Map().Add("url", TEST_SVG_FILE_NAME).Add("desiredWidth", desiredWidth).Add("desiredHeight", desiredHeight); + application.GetScene().Add(imageView); + + application.SendNotification(); + + // Wait for loading & rasterization + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + { + std::stringstream out; + out << GL_TEXTURE_2D << ", " << 0u << ", " << desiredWidth << ", " << desiredHeight; + DALI_TEST_CHECK(textureTrace.FindMethodAndParams("TexImage2D", out.str().c_str())); + } + + END_TEST; +} + +int UtcDaliImageViewImageLoadFailure01(void) +{ + ToolkitTestApplication application; + + Toolkit::StyleManager styleManager = Toolkit::StyleManager::Get(); + DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::SMALL, TEST_BROKEN_IMAGE_S); + DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::NORMAL, TEST_BROKEN_IMAGE_M); + DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::LARGE, TEST_BROKEN_IMAGE_L); + + std::string brokenUrl; + brokenUrl = DevelStyleManager::GetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::SMALL); + DALI_TEST_EQUALS(TEST_BROKEN_IMAGE_S, brokenUrl, TEST_LOCATION); + + brokenUrl = DevelStyleManager::GetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::NORMAL); + DALI_TEST_EQUALS(TEST_BROKEN_IMAGE_M, brokenUrl, TEST_LOCATION); + + brokenUrl = DevelStyleManager::GetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::LARGE); + DALI_TEST_EQUALS(TEST_BROKEN_IMAGE_L, brokenUrl, TEST_LOCATION); + + ImageView imageView = ImageView::New("invalidUrl.png"); + imageView.SetProperty(Actor::Property::SIZE, Vector2(200.f, 200.f)); + + application.GetScene().Add(imageView); + application.SendNotification(); + application.Render(16); + + // loading started, this waits for the loader thread + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewImageLoadFailure02(void) { ToolkitTestApplication application; @@ -3085,7 +3689,7 @@ void OnResourceReadySignal01(Control control) else if(control.GetVisualResourceStatus(ImageView::Property::IMAGE) == Visual::ResourceStatus::FAILED) { // Make the resource ready immediately - control[ImageView::Property::IMAGE] = TEST_RESOURCE_DIR "/svg1.svg"; + control[ImageView::Property::IMAGE] = gImage_600_RGB; } } @@ -3103,6 +3707,7 @@ void OnResourceReadySignal02(Control control) ImageView gImageView1; ImageView gImageView2; ImageView gImageView3; +ImageView gImageView4; void OnResourceReadySignal03(Control control) { @@ -3128,6 +3733,256 @@ void OnResourceReadySignal03(Control control) gResourceReadySignalCounter++; } +void OnSimpleResourceReadySignal(Control control) +{ + // simply increate counter + gResourceReadySignalCounter++; +} + +int gResourceReadySignal04ComesOrder = 0; + +void OnResourceReadySignal04(Control control) +{ + gResourceReadySignalCounter++; + tet_printf("rc %d\n", gResourceReadySignalCounter); + if(gResourceReadySignalCounter == 1) + { + auto scene = gImageView1.GetParent(); + + // Request load something + // We hope this request result is return later than gImageView2. + gImageView3 = ImageView::New(TEST_IMAGE_1); + gImageView3.ResourceReadySignal().Connect(&OnResourceReadySignal04); + scene.Add(gImageView3); + gImageView4 = ImageView::New(TEST_IMAGE_2); + gImageView4.ResourceReadySignal().Connect(&OnResourceReadySignal04); + scene.Add(gImageView4); + + if(control == gImageView1) + { + gResourceReadySignal04ComesOrder = 1; + } + else + { + gResourceReadySignal04ComesOrder = 2; + } + } + if(gResourceReadySignalCounter == 2) + { + if(gResourceReadySignal04ComesOrder == 1 && control == gImageView2) + { + // Scene off first one. + gImageView1.Unparent(); + + // Scene off second one. + gImageView2.Unparent(); + } + else if(gResourceReadySignal04ComesOrder == 2 && control == gImageView1) + { + // Scene off first one. + gImageView2.Unparent(); + + // Scene off second one. + gImageView1.Unparent(); + } + else + { + // We can't check that this utc fail case. just pass always when we come here. + gResourceReadySignal04ComesOrder = -1; + } + + // If we don't seperate index of FreeList area + // and if we don't queue remove during obversing, + // cache index become something invalid data. + // In this case, some strange observer can be called. + // For example, gImageView4.LoadComplete will be called. + } +} + +void OnResourceReadySignal05(Control control) +{ + gResourceReadySignalCounter++; + + // Request load with same image + // NOTE : The url must not be same as gImageView1 + const int viewCount = 4; + for(int i = 0; i < viewCount; ++i) + { + gImageView1.Add(ImageView::New("invalid2.jpg")); + } +} + +int gResourceReadySignal06ComesOrder = 0; + +void OnResourceReadySignal06(Control control) +{ + gResourceReadySignalCounter++; + if(gResourceReadySignalCounter == 1) + { + auto scene = gImageView1.GetParent(); + + // Request load something + // We hope this request result is return later than gImageView2. + + Property::Map map1; + map1[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + map1[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = TEST_BROKEN_IMAGE_DEFAULT; + + gImageView3 = ImageView::New(); + gImageView3.SetProperty(Toolkit::ImageView::Property::IMAGE, map1); + gImageView3.ResourceReadySignal().Connect(&OnResourceReadySignal06); + + Property::Map map2; + map2[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_2; + map2[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = TEST_BROKEN_IMAGE_S; + gImageView4 = ImageView::New(); + gImageView4.SetProperty(Toolkit::ImageView::Property::IMAGE, map2); + gImageView4.ResourceReadySignal().Connect(&OnResourceReadySignal06); + + if(control == gImageView1) + { + gResourceReadySignal06ComesOrder = 1; + } + else + { + gResourceReadySignal06ComesOrder = 2; + } + } + if(gResourceReadySignalCounter == 2) + { + if(gResourceReadySignal06ComesOrder == 1 && control == gImageView2) + { + // Scene off first one. + gImageView1.Unparent(); + + // Scene off second one. + gImageView2.Unparent(); + } + else if(gResourceReadySignal06ComesOrder == 2 && control == gImageView1) + { + // Scene off first one. + gImageView2.Unparent(); + + // Scene off second one. + gImageView1.Unparent(); + } + else + { + // We can't check that this utc fail case. just pass always when we come here. + gResourceReadySignal06ComesOrder = -1; + } + + // If we don't seperate index of FreeList area + // and if we don't queue remove during obversing, + // cache index become something invalid data. + // In this case, some strange observer can be called. + // For example, gImageView4.LoadComplete will be called. + } +} + +void OnResourceReadySignal07(Control control) +{ + gResourceReadySignalCounter++; + // Load masked image + tet_printf("rc %d %d\n", gResourceReadySignalCounter, static_cast(gImageView2)); + + if(!gImageView2) + { + auto scene = gImageView1.GetParent(); + + Property::Map map1; + map1[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + map1[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = TEST_BROKEN_IMAGE_DEFAULT; + + gImageView2 = ImageView::New(); + gImageView2.SetProperty(Toolkit::ImageView::Property::IMAGE, map1); + gImageView2.ResourceReadySignal().Connect(&OnResourceReadySignal07); + + scene.Add(gImageView2); + } +} + +void OnResourceReadySignal08(Control control) +{ + gResourceReadySignalCounter++; + + if(gImageView1) + { + gImageView1.Unparent(); + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Unparent(); + gImageView2.Reset(); + } +} + +std::size_t gResourceReadySignal09Emitted = false; + +void OnResourceReadySignal09(Control control) +{ + gResourceReadySignalCounter++; + + if(gImageView1 && !gResourceReadySignal09Emitted) + { + gResourceReadySignal09Emitted = true; + gImageView1.ResourceReadySignal().Disconnect(&OnResourceReadySignal09); + + // Try to load cached invalid nine patch image. It will request load now. + gImageView1.SetImage(TEST_INVALID_NPATCH_FILE_NAME_01); + gImageView2.SetImage(TEST_INVALID_NPATCH_FILE_NAME_01); + + // Destroy all visuals immediatly. + gImageView1.Unparent(); + gImageView1.Reset(); + gImageView2.Unparent(); + gImageView2.Reset(); + } +} +constexpr int gResourceReadySignal10MaxCounter = 5; + +void OnResourceReadySignal10(Control control) +{ + gResourceReadySignalCounter++; + + tet_printf("OnResourceReadySignal10 comes!\n"); + if(gResourceReadySignalCounter < gResourceReadySignal10MaxCounter) + { + tet_printf("OnResourceReadySignal10 Set image\n"); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, gImage_34_RGBA); + tet_printf("OnResourceReadySignal10 Set image done\n"); + } +} + +void OnResourceReadySignal11(Control control) +{ + gResourceReadySignalCounter++; + + if(!gImageView2) + { + auto scene = gImageView1.GetParent(); + + // Try to load animated image visual here which is already cached, and then ignore forcely. + + Property::Map map1; + map1[Toolkit::ImageVisual::Property::URL] = TEST_GIF_FILE_NAME; + + gImageView2 = ImageView::New(); + gImageView2.SetProperty(Toolkit::ImageView::Property::IMAGE, map1); + + gImageView3 = ImageView::New(); + gImageView3.SetProperty(Toolkit::ImageView::Property::IMAGE, map1); + + scene.Add(gImageView2); + gImageView2.Unparent(); + + scene.Add(gImageView3); + gImageView3.Unparent(); + gImageView3.Reset(); // Destroy visual + } +} + } // namespace int UtcDaliImageViewSetImageOnResourceReadySignal01(void) @@ -3152,6 +4007,15 @@ int UtcDaliImageViewSetImageOnResourceReadySignal01(void) DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + // Create a new ImageView to cache the image + ImageView imageView1 = ImageView::New(gImage_600_RGB); + application.GetScene().Add(imageView1); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + // Reset count gResourceReadySignalCounter = 0; @@ -3228,3 +4092,1539 @@ int UtcDaliImageViewSetImageOnResourceReadySignal03(void) END_TEST; } + +int UtcDaliImageViewOnResourceReadySignalWithBrokenAlphaMask01(void) +{ + tet_infoline("Test signal handler when image / mask image is broken."); + + ToolkitTestApplication application; + + auto TestResourceReadyUrl = [&application](int eventTriggerCount, bool isSynchronous, const std::string& url, const std::string& mask, const char* location) { + gResourceReadySignalCounter = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = url; + if(!mask.empty()) + { + map[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = mask; + } + map[Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING] = isSynchronous; + + ImageView imageView = ImageView::New(); + imageView[Toolkit::ImageView::Property::IMAGE] = map; + imageView[Actor::Property::SIZE] = Vector2(100.0f, 200.0f); + imageView.ResourceReadySignal().Connect(&OnSimpleResourceReadySignal); + + application.GetScene().Add(imageView); + application.SendNotification(); + application.Render(); + + if(!isSynchronous) + { + // Wait for loading + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(eventTriggerCount), true, location); + } + tet_printf("test %s [sync:%d] signal fired\n", url.c_str(), isSynchronous ? 1 : 0); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, location); + + imageView.Unparent(); + }; + + for(int synchronous = 0; synchronous <= 1; synchronous++) + { + tet_printf("Test normal case (sync:%d)\n", synchronous); + TestResourceReadyUrl(1, synchronous, gImage_600_RGB, "", TEST_LOCATION); + TestResourceReadyUrl(3, synchronous, gImage_600_RGB, gImage_34_RGBA, TEST_LOCATION); // 3 event trigger required : 2 image load + 1 apply mask + + tet_printf("Test broken image case (sync:%d)\n", synchronous); + TestResourceReadyUrl(1, synchronous, "invalid.jpg", "", TEST_LOCATION); + TestResourceReadyUrl(2, synchronous, "invalid.jpg", gImage_34_RGBA, TEST_LOCATION); + + tet_printf("Test broken mask image case (sync:%d)\n", synchronous); + TestResourceReadyUrl(2, synchronous, gImage_600_RGB, "invalid.png", TEST_LOCATION); + + tet_printf("Test broken both image, mask image case (sync:%d)\n", synchronous); + TestResourceReadyUrl(2, synchronous, "invalid.jpg", "invalid.png", TEST_LOCATION); + } + + END_TEST; +} + +int UtcDaliImageViewOnResourceReadySignalWithBrokenAlphaMask02(void) +{ + tet_infoline("Test signal handler when image try to use cached-and-broken mask image."); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + auto TestBrokenMaskResourceReadyUrl = [&application](const std::string& url, const char* location) { + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = url; + // Use invalid mask url + map[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = "invalid.png"; + + ImageView imageView = ImageView::New(); + imageView[Toolkit::ImageView::Property::IMAGE] = map; + imageView[Actor::Property::SIZE] = Vector2(100.0f, 200.0f); + imageView.ResourceReadySignal().Connect(&OnSimpleResourceReadySignal); + + application.GetScene().Add(imageView); + + // Don't unparent imageView, for keep the cache. + }; + + // Use more than 4 images (The number of LocalImageLoadThread) + const std::vector testUrlList = {gImage_34_RGBA, gImage_600_RGB, "invalid.jpg" /* invalid url */, TEST_IMAGE_1, TEST_IMAGE_2, TEST_BROKEN_IMAGE_DEFAULT}; + + int expectResourceReadySignalCounter = 0; + + for(auto& url : testUrlList) + { + TestBrokenMaskResourceReadyUrl(url, TEST_LOCATION); + expectResourceReadySignalCounter++; + } + + // Remain 1 signal due to we use #URL + 1 mask image. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(expectResourceReadySignalCounter + 1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(); + DALI_TEST_EQUALS(gResourceReadySignalCounter, expectResourceReadySignalCounter, TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewCheckVariousCaseSendOnResourceReadySignal(void) +{ + tet_infoline("Test signal handler various case."); + + auto TestResourceReadyUrl = [](int eventTriggerCount, bool isSynchronous, bool loadSuccess, const std::string& url, const std::string& mask, const char* location) { + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = url; + if(!mask.empty()) + { + map[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = mask; + } + map[Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING] = isSynchronous; + + ImageView imageView = ImageView::New(); + imageView[Toolkit::ImageView::Property::IMAGE] = map; + imageView[Actor::Property::SIZE] = Vector2(100.0f, 200.0f); + imageView.ResourceReadySignal().Connect(&OnSimpleResourceReadySignal); + + application.GetScene().Add(imageView); + application.SendNotification(); + application.Render(); + + // Wait for loading + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(eventTriggerCount), true, location); + + tet_printf("test %s [sync:%d] signal fired\n", url.c_str(), isSynchronous ? 1 : 0); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, location); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(Toolkit::ImageView::Property::IMAGE), loadSuccess ? Toolkit::Visual::ResourceStatus::READY : Toolkit::Visual::ResourceStatus::FAILED, location); + + imageView.Unparent(); + }; + + auto TestAuxiliaryResourceReadyUrl = [](bool isSynchronous, bool loadSuccess, const std::string& url, const std::string& auxiliaryUrl, const char* location) { + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = url; + map[Toolkit::DevelImageVisual::Property::AUXILIARY_IMAGE] = auxiliaryUrl; + map[Toolkit::DevelImageVisual::Property::AUXILIARY_IMAGE_ALPHA] = 0.5f; + map[Toolkit::ImageVisual::Property::SYNCHRONOUS_LOADING] = isSynchronous; + + ImageView imageView = ImageView::New(); + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + imageView.ResourceReadySignal().Connect(&OnSimpleResourceReadySignal); + application.GetScene().Add(imageView); + + application.SendNotification(); + application.Render(); + + if(!isSynchronous) + { + // Wait for loading + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, location); + } + + tet_printf("test %s [sync:%d] signal fired\n", url.c_str(), isSynchronous ? 1 : 0); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, location); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(Toolkit::ImageView::Property::IMAGE), loadSuccess ? Toolkit::Visual::ResourceStatus::READY : Toolkit::Visual::ResourceStatus::FAILED, TEST_LOCATION); + + imageView.Unparent(); + }; + + // Case 1 : asynchronous loading + tet_printf("Test invalid single simple image Asynchronous\n"); + + // Test normal case + TestResourceReadyUrl(1, 0, 1, gImage_600_RGB, "", TEST_LOCATION); + TestResourceReadyUrl(2, 0, 1, TEST_SVG_FILE_NAME, "", TEST_LOCATION); // load & rasterize + TestResourceReadyUrl(1, 0, 1, TEST_BROKEN_IMAGE_L, "", TEST_LOCATION); + + TestResourceReadyUrl(2, 0, 1, TEST_GIF_FILE_NAME, "", TEST_LOCATION); // 2 image loading - batch size + TestResourceReadyUrl(2, 0, 1, TEST_ANIMATED_VECTOR_IMAGE_FILE_NAME, "", TEST_LOCATION); // load & rasterize + + TestResourceReadyUrl(3, 0, 1, gImage_600_RGB, gImage_34_RGBA, TEST_LOCATION); // 2 image loading + 1 applymask + + TestAuxiliaryResourceReadyUrl(0, 1, TEST_BROKEN_IMAGE_L, gImage_34_RGBA, TEST_LOCATION); + + // Test broken case + TestResourceReadyUrl(1, 0, 0, "invalid.jpg", "", TEST_LOCATION); + TestResourceReadyUrl(1, 0, 0, "invalid.svg", "", TEST_LOCATION); + TestResourceReadyUrl(1, 0, 0, "invalid.9.png", "", TEST_LOCATION); + TestResourceReadyUrl(1, 0, 0, "invalid.gif", "", TEST_LOCATION); // 1 image loading + TestResourceReadyUrl(1, 0, 0, "invalid.json", "", TEST_LOCATION); // 0 rasterize + + TestResourceReadyUrl(2, 0, 0, "invalid.jpg", "invalid.png", TEST_LOCATION); // 2 image loading + TestResourceReadyUrl(2, 0, 1, gImage_600_RGB, "invalid.png", TEST_LOCATION); // 2 image loading + TestResourceReadyUrl(2, 0, 0, "invalid.jpg", gImage_34_RGBA, TEST_LOCATION); // 2 image loading + + TestAuxiliaryResourceReadyUrl(0, 0, "invalid.9.png", "invalid.png", TEST_LOCATION); + TestAuxiliaryResourceReadyUrl(0, 1, TEST_BROKEN_IMAGE_L, "invalid.png", TEST_LOCATION); + TestAuxiliaryResourceReadyUrl(0, 0, "invalid.9.png", gImage_34_RGBA, TEST_LOCATION); + + // Case 2 : Synchronous loading + tet_printf("Test invalid single simple image Synchronous\n"); + + // Test normal case + TestResourceReadyUrl(0, 1, 1, gImage_600_RGB, "", TEST_LOCATION); + TestResourceReadyUrl(0, 1, 1, TEST_SVG_FILE_NAME, "", TEST_LOCATION); // synchronous rasterize + TestResourceReadyUrl(0, 1, 1, TEST_BROKEN_IMAGE_L, "", TEST_LOCATION); + + TestResourceReadyUrl(1, 1, 1, TEST_GIF_FILE_NAME, "", TEST_LOCATION); // first frame image loading sync + second frame image loading async + + TestResourceReadyUrl(0, 1, 1, gImage_600_RGB, gImage_34_RGBA, TEST_LOCATION); + + TestAuxiliaryResourceReadyUrl(1, 1, TEST_BROKEN_IMAGE_L, gImage_34_RGBA, TEST_LOCATION); + + // Test broken case + TestResourceReadyUrl(0, 1, 0, "invalid.jpg", "", TEST_LOCATION); + TestResourceReadyUrl(0, 1, 0, "invalid.svg", "", TEST_LOCATION); + TestResourceReadyUrl(0, 1, 0, "invalid.9.png", "", TEST_LOCATION); + TestResourceReadyUrl(0, 1, 0, "invalid.gif", "", TEST_LOCATION); + + TestResourceReadyUrl(0, 1, 0, "invalid.jpg", "invalid.png", TEST_LOCATION); + TestResourceReadyUrl(0, 1, 1, gImage_600_RGB, "invalid.png", TEST_LOCATION); + TestResourceReadyUrl(0, 1, 0, "invalid.jpg", gImage_34_RGBA, TEST_LOCATION); + + TestAuxiliaryResourceReadyUrl(1, 0, "invalid.9.png", "invalid.png", TEST_LOCATION); + TestAuxiliaryResourceReadyUrl(1, 1, TEST_BROKEN_IMAGE_L, "invalid.png", TEST_LOCATION); + TestAuxiliaryResourceReadyUrl(1, 0, "invalid.9.png", gImage_34_RGBA, TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal04(void) +{ + tet_infoline("Test texturemanager's remove queue works well within signal handler."); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + gResourceReadySignal04ComesOrder = 0; + + gImageView1 = ImageView::New("invalid.jpg"); // request invalid image, to make loading failed fast. + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal04); + application.GetScene().Add(gImageView1); + + gImageView2 = ImageView::New("invalid.png"); // request invalid image, to make loading failed fast. + gImageView2.ResourceReadySignal().Connect(&OnResourceReadySignal04); + application.GetScene().Add(gImageView2); + + application.SendNotification(); + application.Render(); + + tet_infoline("Try to load 2 invalid image"); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 2, TEST_LOCATION); + + tet_infoline("load done"); + + // We can test this UTC only if gImageView1 and gImageView2 loaded done. + if(gResourceReadySignal04ComesOrder == -1) + { + tet_infoline("Bad news.. gImageView3 or gImageView4 loaded faster than others. just skip this UTC"); + } + else + { + // gImageView3 and gImageView4 load must not be successed yet. + DALI_TEST_EQUALS(gImageView3.GetRendererCount(), 0u, TEST_LOCATION); + DALI_TEST_EQUALS(gImageView4.GetRendererCount(), 0u, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + tet_infoline("Try to load 2 valid image"); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 4, TEST_LOCATION); + + tet_infoline("load done"); + + // gImageView3 and gImageView4 load must be successed now. + DALI_TEST_EQUALS(gImageView3.GetRendererAt(0).GetTextures().GetTextureCount(), 1u, TEST_LOCATION); + DALI_TEST_EQUALS(gImageView4.GetRendererAt(0).GetTextures().GetTextureCount(), 1u, TEST_LOCATION); + } + + END_TEST; +} +int UtcDaliImageViewSetImageOnResourceReadySignal05(void) +{ + tet_infoline("Test multiple views with same image during ResourceReady load the image only 1 times"); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + gImageView1 = ImageView::New("invalid.jpg"); // request invalid image, to make loading failed fast. + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal05); + application.GetScene().Add(gImageView1); + + application.SendNotification(); + application.Render(); + + tet_infoline("Try to load 1 invalid.jpg image"); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, TEST_LOCATION); + + tet_infoline("Try to load 1 invalid2.jpg image"); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + tet_infoline("Now we don't have any image to be loaded. Check event thread trigger failed."); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1, 1), false, TEST_LOCATION); + + gImageView1.Unparent(); + gImageView1.Reset(); + + END_TEST; +} +int UtcDaliImageViewSetImageOnResourceReadySignal06(void) +{ + tet_infoline("Test texturemanager's remove image & mask queue works well within signal handler."); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + gResourceReadySignal06ComesOrder = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = "invalid.jpg"; + map[Toolkit::ImageVisual::Property::ALPHA_MASK_URL] = "invalid.png"; + + gImageView1 = ImageView::New(); // request invalid image, to make loading failed fast. + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal06); + application.GetScene().Add(gImageView1); + + gImageView2 = ImageView::New(); // request invalid image, to make loading failed fast. + gImageView2.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView2.ResourceReadySignal().Connect(&OnResourceReadySignal06); + application.GetScene().Add(gImageView2); + + application.SendNotification(); + application.Render(); + + tet_infoline("Try to load 2 invalid image"); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 2, TEST_LOCATION); + + tet_infoline("load done"); + + // We can test this UTC only if gImageView1 and gImageView2 loaded done. + if(gResourceReadySignal06ComesOrder == -1) + { + tet_infoline("Bad news.. gImageView3 or gImageView4 loaded faster than others. just skip this UTC"); + } + else + { + // gImageView3 and gImageView4 load must not be successed yet. + DALI_TEST_EQUALS(gImageView3.GetRendererCount(), 0u, TEST_LOCATION); + DALI_TEST_EQUALS(gImageView4.GetRendererCount(), 0u, TEST_LOCATION); + + application.GetScene().Add(gImageView3); + application.GetScene().Add(gImageView4); + application.SendNotification(); + application.Render(); + + tet_infoline("Try to load 2 valid image"); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 2, TEST_LOCATION); + + tet_infoline("Note that resource ready should not come now."); + tet_infoline("Try to load remained 2 valid image + apply masking"); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(4), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 4, TEST_LOCATION); + + tet_infoline("Check all resource ready comes now."); + } + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal07(void) +{ + tet_infoline("Test texturemanager's remove image & mask queue works well within signal handler 02."); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal07); + application.GetScene().Add(gImageView1); + + application.SendNotification(); + application.Render(); + + // Load gImageView1 + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, TEST_LOCATION); + + tet_infoline("load image1 done"); + + // Load gImageView2 and mask + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, TEST_LOCATION); + + // gImageView2 mask apply done + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 2, TEST_LOCATION); + + tet_infoline("load image2 done"); + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal08(void) +{ + tet_infoline("Test remove npatch images during resource ready"); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = TEST_BROKEN_IMAGE_M; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + + // Case 1 : Remove all images during resource ready. + try + { + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal08); + application.GetScene().Add(gImageView1); + + application.SendNotification(); + application.Render(); + + // Load gImageView1 + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + + // Clear cache. + application.SendNotification(); + application.Render(); + + gResourceReadySignalCounter = 0; + + // Case 2 : Remove all images when we use cached resource. + try + { + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView1.ResourceReadySignal().Connect(&OnSimpleResourceReadySignal); + application.GetScene().Add(gImageView1); + + application.SendNotification(); + application.Render(); + + // Load gImageView1 + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + gImageView2 = ImageView::New(); + gImageView2.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView2.ResourceReadySignal().Connect(&OnResourceReadySignal08); + application.GetScene().Add(gImageView2); + DALI_TEST_EQUALS(gResourceReadySignalCounter, 2, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal09(void) +{ + tet_infoline("Test load invalid npatch images during invalid resource ready"); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = TEST_INVALID_NPATCH_FILE_NAME_01; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + if(gImageView3) + { + gImageView3.Reset(); + } + + // Dummy view with npatch image + ImageView dummyView = ImageView::New(TEST_BROKEN_IMAGE_M); + application.GetScene().Add(dummyView); + + application.SendNotification(); + application.Render(); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(); + + // Case 1 : Reload images during resource ready. + try + { + gResourceReadySignal09Emitted = false; + + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal09); + application.GetScene().Add(gImageView1); + + gImageView2 = ImageView::New(); + application.GetScene().Add(gImageView2); + + // Load TEST_INVALID_NPATCH_FILE_NAME_01 + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + // Load TEST_INVALID_NPATCH_FILE_NAME_01 one more times. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + + // Clear cache. + application.SendNotification(); + application.Render(); + + gResourceReadySignalCounter = 0; + + // Case 2 : Remove all images when we use cached resource. + try + { + gResourceReadySignal09Emitted = false; + + gImageView3 = ImageView::New(); + gImageView3.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView3.ResourceReadySignal().Connect(&OnSimpleResourceReadySignal); + application.GetScene().Add(gImageView3); + + gImageView2 = ImageView::New(); + application.GetScene().Add(gImageView2); + + // Load gImageView2 + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal09); + application.GetScene().Add(gImageView1); + + // Load gImageView1 + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + // Load TEST_INVALID_NPATCH_FILE_NAME_01 one more times. + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + if(gImageView3) + { + gImageView3.Reset(); + } + + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal10(void) +{ + tet_infoline("Test ResourceReady signal comes more than 2 times."); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + + // Dummy view to cache image. + ImageView dummyView = ImageView::New(gImage_34_RGBA); + application.GetScene().Add(dummyView); + + application.SendNotification(); + application.Render(); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(); + + try + { + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, gImage_34_RGBA); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal10); + application.GetScene().Add(gImageView1); // It will call resourceReady signal 1 time. + + tet_printf("ResourceReady called %d times\n", gResourceReadySignalCounter); + + DALI_TEST_GREATER(gResourceReadySignal10MaxCounter, gResourceReadySignalCounter, TEST_LOCATION); // Check whether resource ready call too much. + + for(int i = 0; i < gResourceReadySignal10MaxCounter; ++i) + { + tet_printf("RunIdles\n"); + // Executes the idle callbacks. + application.RunIdles(); + application.SendNotification(); + application.Render(); + tet_printf("RunIdles done\n"); + } + tet_printf("ResourceReady called %d times\n", gResourceReadySignalCounter); + + DALI_TEST_EQUALS(gResourceReadySignalCounter, gResourceReadySignal10MaxCounter, TEST_LOCATION); + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + + // Clear cache. + application.SendNotification(); + application.Render(); + + gResourceReadySignalCounter = 0; + + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal10WhenAddIdleFailed(void) +{ + tet_infoline("Test ResourceReady signal comes more than 2 times, but do not call again if AddIdle failed"); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + + // Dummy view to cache image. + ImageView dummyView = ImageView::New(gImage_34_RGBA); + application.GetScene().Add(dummyView); + + application.SendNotification(); + application.Render(); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(); + + // Make AddIdle failed. + ToolkitApplication::ADD_IDLE_SUCCESS = false; + + try + { + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, gImage_34_RGBA); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal10); + application.GetScene().Add(gImageView1); // It will call resourceReady signal 1 time. + + tet_printf("ResourceReady called %d times\n", gResourceReadySignalCounter); + + DALI_TEST_GREATER(gResourceReadySignal10MaxCounter, gResourceReadySignalCounter, TEST_LOCATION); // Check whether resource ready call too much. + + for(int i = 0; i < gResourceReadySignal10MaxCounter; ++i) + { + tet_printf("RunIdles\n"); + // Executes the idle callbacks. + application.RunIdles(); + application.SendNotification(); + application.Render(); + tet_printf("RunIdles done\n"); + } + tet_printf("ResourceReady called %d times\n", gResourceReadySignalCounter); + + DALI_TEST_GREATER(gResourceReadySignal10MaxCounter, gResourceReadySignalCounter, TEST_LOCATION); // Check whether resource ready not called multiple times. + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + + ToolkitApplication::ADD_IDLE_SUCCESS = true; + + // Clear cache. + application.SendNotification(); + application.Render(); + + gResourceReadySignalCounter = 0; + + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + + END_TEST; +} + +int UtcDaliImageViewSetImageOnResourceReadySignal11(void) +{ + tet_infoline("Test ResourceReady Add AnimatedImageVisual and then Remove immediately."); + + ToolkitTestApplication application; + + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + if(gImageView3) + { + gImageView3.Reset(); + } + + try + { + gImageView1 = ImageView::New(); + gImageView1.SetProperty(Toolkit::ImageView::Property::IMAGE, TEST_GIF_FILE_NAME); + gImageView1.ResourceReadySignal().Connect(&OnResourceReadySignal11); + application.GetScene().Add(gImageView1); // It will call resourceReady signal 1 time. + + tet_printf("ResourceReady called %d times\n", gResourceReadySignalCounter); + + DALI_TEST_EQUALS(gResourceReadySignalCounter, 0, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + // Load gImageView1 + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + tet_printf("ResourceReady called %d times\n", gResourceReadySignalCounter); + + DALI_TEST_EQUALS(gResourceReadySignalCounter, 1, TEST_LOCATION); + + DALI_TEST_CHECK(true); + } + catch(...) + { + // Exception should not happened + DALI_TEST_CHECK(false); + } + + // Clear cache. + application.SendNotification(); + application.Render(); + + gResourceReadySignalCounter = 0; + + // Clear image view for clear test + + if(gImageView1) + { + gImageView1.Reset(); + } + if(gImageView2) + { + gImageView2.Reset(); + } + if(gImageView3) + { + gImageView3.Reset(); + } + + END_TEST; +} + +int UtcDaliImageViewUseSameUrlWithAnimatedImageVisual(void) +{ + tet_infoline("Test multiple views with same image in animated image visual"); + ToolkitTestApplication application; + + gImageView1 = ImageView::New(TEST_WEBP_FILE_NAME); + application.GetScene().Add(gImageView1); + + tet_infoline("Remove imageView and Create new imageView with same url"); + application.GetScene().Remove(gImageView1); + gImageView2 = ImageView::New(TEST_WEBP_FILE_NAME); + application.GetScene().Add(gImageView2); + + application.SendNotification(); + application.Render(); + + tet_infoline("Check the ImageView load image successfully"); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + END_TEST; +} + +int UtcDaliImageViewNpatchImageCacheTest01(void) +{ + tet_infoline("Test npatch image cached"); + + ToolkitTestApplication application; + ImageView imageView[2]; + + auto& gl = application.GetGlAbstraction(); + gl.EnableTextureCallTrace(true); + auto& textureCallStack = gl.GetTextureTrace(); + textureCallStack.Enable(true); + textureCallStack.EnableLogging(true); + + auto TestNPatch = [&](int index, const std::string& nPatchImageUrl, const char* location) { + if(imageView[index]) + { + imageView[index].Unparent(); + } + + // Ensure remove npatch cache if required. + application.SendNotification(); + application.Render(); + + imageView[index] = ImageView::New(nPatchImageUrl); + imageView[index].SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + application.GetScene().Add(imageView[index]); + }; + + TestNPatch(0, TEST_BROKEN_IMAGE_M, TEST_LOCATION); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + tet_printf("trace : \n%s\n", textureCallStack.GetTraceString().c_str()); + + // Check we gen only 1 textures + DALI_TEST_EQUALS(textureCallStack.CountMethod("GenTextures"), 1, TEST_LOCATION); + textureCallStack.Reset(); + + // Let we use cached textures + for(int i = 0; i < 10; i++) + { + TestNPatch(1, TEST_BROKEN_IMAGE_M, TEST_LOCATION); + TestNPatch(0, TEST_BROKEN_IMAGE_M, TEST_LOCATION); + } + + application.SendNotification(); + application.Render(); + // Check we use cached npatch data so we don't generate new texture textures + DALI_TEST_EQUALS(textureCallStack.CountMethod("GenTextures"), 0, TEST_LOCATION); + + // Clear all cached + imageView[0].Unparent(); + imageView[0].Reset(); + imageView[1].Unparent(); + imageView[1].Reset(); + + application.SendNotification(); + application.Render(); + + textureCallStack.Reset(); + // Let we use deference textures + TestNPatch(1, TEST_BROKEN_IMAGE_S, TEST_LOCATION); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + // Try to load multiple times. + for(int i = 0; i < 3; i++) + { + TestNPatch(0, TEST_BROKEN_IMAGE_M, TEST_LOCATION); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + TestNPatch(1, TEST_BROKEN_IMAGE_S, TEST_LOCATION); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + } + application.SendNotification(); + application.Render(); + + imageView[0].Unparent(); + imageView[0].Reset(); + imageView[1].Unparent(); + imageView[1].Reset(); + + application.SendNotification(); + application.Render(); + + // Check memory leak + DALI_TEST_EQUALS(textureCallStack.CountMethod("GenTextures"), textureCallStack.CountMethod("DeleteTextures"), TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewNpatchImageCacheTest02(void) +{ + tet_infoline("Test npatch image cached with border"); + + ToolkitTestApplication application; + ImageView imageView[2]; + + auto& gl = application.GetGlAbstraction(); + gl.EnableTextureCallTrace(true); + auto& textureCallStack = gl.GetTextureTrace(); + textureCallStack.Enable(true); + textureCallStack.EnableLogging(true); + + auto TestBorderImage = [&](int index, const std::string& normalImageUrl, const Rect border, const char* location) { + Property::Map map; + map[Toolkit::Visual::Property::TYPE] = Toolkit::Visual::N_PATCH; + map[Toolkit::ImageVisual::Property::URL] = normalImageUrl; + map[Toolkit::ImageVisual::Property::BORDER] = border; + + imageView[index] = ImageView::New(); + imageView[index].SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView[index].SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + application.GetScene().Add(imageView[index]); + }; + + TestBorderImage(0, gImage_34_RGBA, Rect(0, 0, 0, 0), TEST_LOCATION); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(); + + tet_printf("trace : \n%s\n", textureCallStack.GetTraceString().c_str()); + + // Check we gen only 1 textures + DALI_TEST_EQUALS(textureCallStack.CountMethod("GenTextures"), 1, TEST_LOCATION); + textureCallStack.Reset(); + + // Let we use cached textures + for(int i = 0; i < 10; i++) + { + TestBorderImage(0, gImage_34_RGBA, Rect(i, i + 1, i + 2, i + 3), TEST_LOCATION); + TestBorderImage(1, gImage_34_RGBA, Rect(i + 1, i, i + 3, i + 2), TEST_LOCATION); + } + + application.SendNotification(); + application.Render(); + + // Check we use cached npatch data so we don't generate new texture textures + DALI_TEST_EQUALS(textureCallStack.CountMethod("GenTextures"), 0, TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewPlaceholderImage01(void) +{ + tet_infoline("Test imageView use placeholder image"); + + ToolkitTestApplication application; + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + + ImageView imageView = ImageView::New(); + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + application.GetScene().Add(imageView); + + Property::Value value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + std::string url; + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url.empty()); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + + application.SendNotification(); + application.Render(); + + value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url == gImage_34_RGBA); + + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + // Replace Image test + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + map[ImageView::Property::PLACEHOLDER_IMAGE] = gImage_34_RGBA; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_2; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + // Replace Image test2 + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_2; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(900); + + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + END_TEST; +} + +int UtcDaliImageViewPlaceholderImage02(void) +{ + tet_infoline("Test imageView use placeholder image without resource ready"); + + ToolkitTestApplication application; + + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + + ImageView imageView = ImageView::New(); + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + DALI_TEST_EQUALS(imageView.IsResourceReady(), false, TEST_LOCATION); + imageView.ResourceReadySignal().Connect(&ResourceReadySignal); + application.GetScene().Add(imageView); + + Property::Value value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + std::string url; + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url.empty()); + + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + application.Render(); + + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url == gImage_34_RGBA); + + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + + gResourceReadySignalFired = false; + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(gResourceReadySignalFired, false, TEST_LOCATION); + + END_TEST; +} + +int UtcDaliImageViewTransitionEffect01(void) +{ + tet_infoline("Test imageView use transition effect"); + + ToolkitTestApplication application; + Property::Map map; + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + map[Toolkit::Visual::Property::OPACITY] = 0.9f; + + ImageView imageView = ImageView::New(); + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + application.GetScene().Add(imageView); + + Property::Value value = imageView.GetProperty(ImageView::Property::ENABLE_TRANSITION_EFFECT); + bool transition; + DALI_TEST_CHECK(value.Get(transition)); + DALI_TEST_CHECK(transition == false); + imageView.SetProperty(Toolkit::ImageView::Property::ENABLE_TRANSITION_EFFECT, true); + + application.SendNotification(); + application.Render(); + + value = imageView.GetProperty(ImageView::Property::ENABLE_TRANSITION_EFFECT); + DALI_TEST_CHECK(value.Get(transition)); + DALI_TEST_CHECK(transition == true); + + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(); + + // Test transition effect with placeholder + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, ""); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + // Test transition effect without placeholder + map[Toolkit::ImageVisual::Property::URL] = TEST_IMAGE_1; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = ""; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + map[Toolkit::ImageVisual::Property::URL] = gImage_600_RGB; + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + imageView.SetImage(TEST_IMAGE_1); + application.SendNotification(); + application.Render(); + + imageView.SetImage(gImage_600_RGB); + application.SendNotification(); + application.Render(9000); + + imageView.SetImage(""); + application.SendNotification(); + application.Render(); + + imageView.SetImage(TEST_IMAGE_1); + application.SendNotification(); + application.Render(); + + // Clear all cached + imageView.Unparent(); + imageView.Reset(); + + END_TEST; +} + +int UtcDaliImageViewTransitionEffect02(void) +{ + tet_infoline("Test imageView use transition effect with replace image"); + + ToolkitTestApplication application; + + Property::Map map; + + ImageView imageView = ImageView::New(); + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + application.GetScene().Add(imageView); + + Property::Value value; + value = imageView.GetProperty(ImageView::Property::ENABLE_TRANSITION_EFFECT); + bool transition; + DALI_TEST_CHECK(value.Get(transition)); + DALI_TEST_CHECK(transition == false); + imageView.SetProperty(Toolkit::ImageView::Property::ENABLE_TRANSITION_EFFECT, true); + + value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + std::string url; + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url.empty()); + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + application.Render(); + + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, ""); + application.SendNotification(); + application.Render(); + + imageView.SetProperty(Toolkit::ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + application.SendNotification(); + application.Render(); + + value = imageView.GetProperty(ImageView::Property::ENABLE_TRANSITION_EFFECT); + DALI_TEST_CHECK(value.Get(transition)); + DALI_TEST_CHECK(transition == true); + + value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url == gImage_34_RGBA); + + imageView.SetProperty(Toolkit::ImageView::Property::IMAGE, map); + application.SendNotification(); + application.Render(); + + // Clear all cached + imageView.Unparent(); + imageView.Reset(); + + END_TEST; +} + +int UtcDaliImageViewTransitionEffect03(void) +{ + tet_infoline("Test imageView use transition effect with placeholder"); + + ToolkitTestApplication application; + Property::Map map; + + ImageView imageView = ImageView::New(); + imageView.SetImage(""); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.0f, 200.0f)); + imageView.SetProperty(ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + imageView.SetProperty(ImageView::Property::ENABLE_TRANSITION_EFFECT, true); + application.GetScene().Add(imageView); + + //DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + application.SendNotification(); + application.Render(16); + + tet_infoline("(1)"); + imageView.SetImage(gImage_600_RGB); + imageView.SetProperty(ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(2), true, TEST_LOCATION); + application.SendNotification(); + application.Render(16); + + Property::Value value; + value = imageView.GetProperty(ImageView::Property::ENABLE_TRANSITION_EFFECT); + bool transition; + DALI_TEST_CHECK(value.Get(transition)); + DALI_TEST_CHECK(transition == true); + + value = imageView.GetProperty(ImageView::Property::PLACEHOLDER_IMAGE); + std::string url; + DALI_TEST_CHECK(value.Get(url)); + DALI_TEST_CHECK(url == gImage_34_RGBA); + + imageView.SetImage(""); + application.SendNotification(); + application.Render(16); + + imageView.SetImage(TEST_IMAGE_1); + imageView.SetProperty(ImageView::Property::PLACEHOLDER_IMAGE, gImage_34_RGBA); + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + application.SendNotification(); + application.Render(16); + + // Clear all cached + imageView.Unparent(); + imageView.Reset(); + + END_TEST; +} + +int UtcDaliImageViewImageLoadFailureAndReload01(void) +{ + tet_infoline("Try to load invalid image first, and then reload after that image valid."); + ToolkitTestApplication application; + + gResourceReadySignalFired = false; + + // Make overwritable image invalid first. + OverwriteImage(""); + + ImageView imageView = ImageView::New(TEST_OVERWRITABLE_IMAGE_FILE_NAME); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.f, 100.f)); + imageView.ResourceReadySignal().Connect(&ResourceReadySignal); + + application.GetScene().Add(imageView); + application.SendNotification(); + application.Render(16); + + // loading started, this waits for the loader thread + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + gResourceReadySignalFired = false; + + // Make overwritable image valid now. + OverwriteImage(gImage_34_RGBA); + + // Reload the image + Property::Map attributes; + DevelControl::DoAction(imageView, ImageView::Property::IMAGE, DevelImageVisual::Action::RELOAD, attributes); + application.SendNotification(); + application.Render(16); + + // loading started, this waits for the loader thread + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::READY, TEST_LOCATION); + + // Make overwritable image invalid end of test (for clean). + OverwriteImage(""); + + gResourceReadySignalFired = false; + + END_TEST; +} + +int UtcDaliImageViewImageLoadFailureAndReload02(void) +{ + tet_infoline("Try to load invalid image first, and then reload after that image valid."); + tet_infoline("This case, broken image was n-patch. So we should check whether Geometry / Shader changed after reload"); + ToolkitTestApplication application; + + Toolkit::StyleManager styleManager = Toolkit::StyleManager::Get(); + DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::SMALL, TEST_BROKEN_IMAGE_S); + DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::NORMAL, TEST_BROKEN_IMAGE_M); + DevelStyleManager::SetBrokenImageUrl(styleManager, DevelStyleManager::BrokenImageType::LARGE, TEST_BROKEN_IMAGE_L); + + gResourceReadySignalFired = false; + + // Make overwritable image invalid first. + OverwriteImage(""); + + ImageView imageView = ImageView::New(TEST_OVERWRITABLE_IMAGE_FILE_NAME); + imageView.SetProperty(Actor::Property::SIZE, Vector2(100.f, 100.f)); + imageView.ResourceReadySignal().Connect(&ResourceReadySignal); + + application.GetScene().Add(imageView); + application.SendNotification(); + application.Render(16); + + // loading started, this waits for the loader thread + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::FAILED, TEST_LOCATION); + + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + Geometry brokenGeometry = imageView.GetRendererAt(0u).GetGeometry(); + Shader brokenShader = imageView.GetRendererAt(0u).GetShader(); + DALI_TEST_CHECK(brokenGeometry); + DALI_TEST_CHECK(brokenShader); + + gResourceReadySignalFired = false; + + // Make overwritable image valid now. + OverwriteImage(gImage_34_RGBA); + + // Reload the image + Property::Map attributes; + DevelControl::DoAction(imageView, ImageView::Property::IMAGE, DevelImageVisual::Action::RELOAD, attributes); + application.SendNotification(); + application.Render(16); + + // loading started, this waits for the loader thread + DALI_TEST_EQUALS(Test::WaitForEventThreadTrigger(1), true, TEST_LOCATION); + + DALI_TEST_EQUALS(gResourceReadySignalFired, true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.IsResourceReady(), true, TEST_LOCATION); + DALI_TEST_EQUALS(imageView.GetVisualResourceStatus(ImageView::Property::IMAGE), Visual::ResourceStatus::READY, TEST_LOCATION); + + // Check whether we don't use n-patch shader and geometry in this case. + DALI_TEST_EQUALS(imageView.GetRendererCount(), 1u, TEST_LOCATION); + DALI_TEST_CHECK(brokenGeometry != imageView.GetRendererAt(0u).GetGeometry()); + DALI_TEST_CHECK(brokenShader != imageView.GetRendererAt(0u).GetShader()); + + // Make overwritable image invalid end of test (for clean). + OverwriteImage(""); + + gResourceReadySignalFired = false; + + END_TEST; +} \ No newline at end of file