2 * Copyright (c) 2015 Samsung Electronics Co., Ltd.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
19 #include <dali/internal/event/images/resource-image-impl.h>
22 #include <dali/public-api/common/dali-common.h>
23 #include <dali/public-api/object/type-registry.h>
24 #include <dali/integration-api/debug.h>
25 #include <dali/internal/event/common/thread-local-storage.h>
26 #include <dali/internal/event/images/image-factory.h>
27 #include <dali/internal/event/images/nine-patch-image-impl.h>
28 #include <dali/internal/event/common/stage-impl.h>
30 using namespace Dali::Integration;
41 BaseHandle CreateImage()
43 ImagePtr image = ResourceImage::New();
44 return Dali::Image(image.Get());
47 TypeRegistration mType( typeid(Dali::ResourceImage), typeid(Dali::Image), CreateImage );
49 Dali::SignalConnectorType signalConnector1(mType, Dali::ResourceImage::SIGNAL_IMAGE_LOADING_FINISHED, &ResourceImage::DoConnectSignal);
53 ResourceImage::ResourceImage( LoadPolicy loadPol, ReleasePolicy releasePol )
54 : Image( releasePol ),
55 mImageFactory( ThreadLocalStorage::Get().GetImageFactory() ),
60 ResourceImagePtr ResourceImage::New()
62 ResourceImagePtr image = new ResourceImage;
67 ResourceImagePtr ResourceImage::New( const std::string& url, const Dali::ImageAttributes& attributes, LoadPolicy loadPol, ReleasePolicy releasePol )
69 ResourceImagePtr image;
70 if( IsNinePatch( url ) )
72 image = NinePatchImage::New( url, attributes, releasePol );
76 image = new ResourceImage( loadPol, releasePol );
79 // consider the requested size as natural size, 0 means we don't (yet) know it
80 image->mWidth = attributes.GetWidth();
81 image->mHeight = attributes.GetHeight();
82 image->mRequest = image->mImageFactory.RegisterRequest( url, &attributes );
84 if( Dali::ResourceImage::IMMEDIATE == loadPol )
86 // Trigger loading of the image on a as soon as it can be done
87 image->mTicket = image->mImageFactory.Load( *image->mRequest.Get() );
88 image->mTicket->AddObserver( *image );
91 DALI_LOG_SET_OBJECT_STRING( image, url );
96 ResourceImage::~ResourceImage()
100 mTicket->RemoveObserver( *this );
101 if( Stage::IsInstalled() )
103 mImageFactory.ReleaseTicket( mTicket.Get() );
109 bool ResourceImage::DoConnectSignal( BaseObject* object, ConnectionTrackerInterface* tracker, const std::string& signalName, FunctorDelegate* functor )
111 bool connected( true );
112 DALI_ASSERT_DEBUG( dynamic_cast<ResourceImage*>( object ) && "Resource ticket not ImageTicket subclass for image resource.\n" );
113 ResourceImage* image = static_cast<ResourceImage*>(object);
115 if( Dali::ResourceImage::SIGNAL_IMAGE_LOADING_FINISHED == signalName )
117 image->LoadingFinishedSignal().Connect( tracker, functor );
119 else if(Dali::ResourceImage::SIGNAL_IMAGE_UPLOADED == signalName)
121 image->UploadedSignal().Connect( tracker, functor );
125 // signalName does not match any signal
132 const Dali::ImageAttributes& ResourceImage::GetAttributes() const
136 return mImageFactory.GetActualAttributes( mTicket );
140 return mImageFactory.GetRequestAttributes( mRequest );
144 const std::string& ResourceImage::GetUrl() const
146 return mImageFactory.GetRequestPath( mRequest );
149 void ResourceImage::Reload()
153 ResourceTicketPtr ticket = mImageFactory.Reload( *mRequest.Get() );
154 SetTicket( ticket.Get() );
158 unsigned int ResourceImage::GetWidth() const
160 // if width is 0, it means we've not yet loaded the image
164 mImageFactory.GetImageSize( mRequest, mTicket, size );
166 // The app will probably ask for the height immediately, so don't waste the synchronous file IO that ImageFactory may have just done:
167 DALI_ASSERT_DEBUG( 0 == mHeight || unsigned(size.height) == mHeight );
170 mHeight = size.height;
176 unsigned int ResourceImage::GetHeight() const
181 mImageFactory.GetImageSize( mRequest, mTicket, size );
182 mHeight = size.height;
183 DALI_ASSERT_DEBUG( 0 == mWidth || unsigned(size.width) == mWidth );
192 Vector2 ResourceImage::GetNaturalSize() const
194 Vector2 naturalSize(mWidth, mHeight);
195 if( 0u == mWidth || 0u == mHeight )
197 mImageFactory.GetImageSize( mRequest, mTicket, naturalSize );
198 mWidth = naturalSize.width;
199 mHeight = naturalSize.height;
204 void ResourceImage::ResourceLoadingFailed(const ResourceTicket& ticket)
206 mLoadingFinished.Emit( Dali::ResourceImage( this ) );
209 void ResourceImage::ResourceLoadingSucceeded(const ResourceTicket& ticket)
211 mLoadingFinished.Emit( Dali::ResourceImage( this ) );
214 void ResourceImage::Connect()
218 if( mConnectionCount == 1 )
220 // ticket was thrown away when related actors went offstage or image loading on demand
223 DALI_ASSERT_DEBUG( mRequest.Get() );
224 ResourceTicketPtr newTicket = mImageFactory.Load( *mRequest.Get() );
225 SetTicket( newTicket.Get() );
230 void ResourceImage::Disconnect()
237 DALI_ASSERT_DEBUG( mConnectionCount > 0 );
239 if( mConnectionCount == 0 && mReleasePolicy == Dali::ResourceImage::UNUSED )
241 // release image memory when it's not visible anymore (decrease ref. count of texture)
246 bool ResourceImage::IsNinePatch( const std::string& url )
250 std::string::const_reverse_iterator iter = url.rbegin();
251 enum { SUFFIX, HASH, HASH_DOT, DONE } state = SUFFIX;
252 while(iter < url.rend())
262 else if(!isalnum(*iter))
270 if( *iter == '#' || *iter == '9' )
286 state = DONE; // Stop testing characters
306 void ResourceImage::SetTicket( ResourceTicket* ticket )
308 if( ticket == mTicket.Get() )
315 mTicket->RemoveObserver( *this );
316 mImageFactory.ReleaseTicket( mTicket.Get() );
321 mTicket.Reset( ticket );
322 mTicket->AddObserver( *this );
330 } // namespace Internal