Merge "Shader & Program cleanup, Part 2: Stop calling render thread methods from...
[platform/core/uifw/dali-core.git] / dali / internal / event / images / image-impl.cpp
1 /*
2  * Copyright (c) 2014 Samsung Electronics Co., Ltd.
3  *
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
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
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.
15  *
16  */
17
18 // CLASS HEADER
19 #include <dali/internal/event/images/image-impl.h>
20
21 // INTERNAL INCLUDES
22 #include <dali/public-api/common/dali-common.h>
23 #include <dali/public-api/object/type-registry.h>
24
25 #include <dali/integration-api/debug.h>
26 #include <dali/internal/event/resources/resource-ticket.h>
27 #include <dali/internal/event/common/thread-local-storage.h>
28 #include <dali/internal/event/resources/resource-client.h>
29 #include <dali/internal/event/images/image-factory.h>
30 #include <dali/internal/event/images/nine-patch-image-impl.h>
31 #include <dali/internal/event/common/stage-impl.h>
32
33 using namespace Dali::Integration;
34
35 namespace Dali
36 {
37
38 namespace Internal
39 {
40
41 namespace
42 {
43
44 BaseHandle CreateImage()
45 {
46   ImagePtr image = Image::New();
47   return Dali::Image(image.Get());
48 }
49
50 TypeRegistration mType( typeid(Dali::Image), typeid(Dali::BaseHandle), CreateImage );
51
52 Dali::SignalConnectorType signalConnector1(mType, Dali::Image::SIGNAL_IMAGE_LOADING_FINISHED,    &Image::DoConnectSignal);
53 Dali::SignalConnectorType signalConnector2(mType, Dali::Image::SIGNAL_IMAGE_UPLOADED,            &Image::DoConnectSignal);
54
55 }
56
57 Image::Image( LoadPolicy loadPol, ReleasePolicy releasePol )
58 : mImageFactory(ThreadLocalStorage::Get().GetImageFactory()),
59   mWidth(0),
60   mHeight(0),
61   mConnectionCount(0),
62   mLoadPolicy(loadPol),
63   mReleasePolicy(releasePol)
64 {
65 }
66
67 ImagePtr Image::New()
68 {
69   ImagePtr image = new Image;
70   image->Initialize();
71   return image;
72 }
73
74 ImagePtr Image::New( const std::string& filename, const Dali::ImageAttributes& attributes, LoadPolicy loadPol, ReleasePolicy releasePol )
75 {
76   ImagePtr image;
77   if( IsNinePatchFileName(filename) )
78   {
79     image = NinePatchImage::New( filename, attributes, loadPol, releasePol );
80   }
81   else
82   {
83     image = new Image( loadPol, releasePol );
84     image->Initialize();
85
86     // consider the requested size as natural size, 0 means we don't (yet) know it
87     image->mWidth = attributes.GetWidth();
88     image->mHeight = attributes.GetHeight();
89     image->mRequest = image->mImageFactory.RegisterRequest( filename, &attributes );
90
91     if( Dali::Image::Immediate == loadPol )
92     {
93       // Trigger loading of the image on a as soon as it can be done
94       image->mTicket = image->mImageFactory.Load( image->mRequest.Get() );
95       image->mTicket->AddObserver( *image );
96     }
97   }
98   DALI_LOG_SET_OBJECT_STRING( image, filename );
99
100   return image;
101 }
102
103 ImagePtr Image::New( NativeImage& nativeImg )
104 {
105   ImagePtr image = new Image;
106   image->Initialize();
107
108   ResourceClient &resourceClient = ThreadLocalStorage::Get().GetResourceClient();
109
110   image->mWidth  = nativeImg.GetWidth();
111   image->mHeight = nativeImg.GetHeight();
112
113   const ResourceTicketPtr& ticket = resourceClient.AddNativeImage( nativeImg );
114   DALI_ASSERT_DEBUG( dynamic_cast<ImageTicket*>( ticket.Get() ) && "Resource ticket not ImageTicket subclass for image resource.\n" );
115   image->mTicket = static_cast<ImageTicket*>(ticket.Get());
116   image->mTicket->AddObserver( *image );
117
118   return image;
119 }
120
121 Image::~Image()
122 {
123   if( mTicket )
124   {
125     mTicket->RemoveObserver( *this );
126     if( Stage::IsInstalled() )
127     {
128       mImageFactory.ReleaseTicket( mTicket.Get() );
129     }
130   }
131
132   if( Stage::IsInstalled() )
133   {
134     UnregisterObject();
135   }
136 }
137
138 bool Image::DoConnectSignal( BaseObject* object, ConnectionTrackerInterface* tracker, const std::string& signalName, FunctorDelegate* functor )
139 {
140   bool connected( true );
141   DALI_ASSERT_DEBUG( dynamic_cast<Image*>( object ) && "Resource ticket not ImageTicket subclass for image resource.\n" );
142   Image* image = static_cast<Image*>(object);
143
144   if( Dali::Image::SIGNAL_IMAGE_LOADING_FINISHED == signalName )
145   {
146     image->LoadingFinishedSignal().Connect( tracker, functor );
147   }
148   else if(Dali::Image::SIGNAL_IMAGE_UPLOADED == signalName)
149   {
150     image->UploadedSignal().Connect( tracker, functor );
151   }
152   else
153   {
154     // signalName does not match any signal
155     connected = false;
156   }
157
158   return connected;
159 }
160
161 ResourceId Image::GetResourceId() const
162 {
163   ResourceId ret = mTicket ? mTicket->GetId() : 0;
164
165   return ret;
166 }
167
168 const Dali::ImageAttributes& Image::GetAttributes() const
169 {
170   if( mTicket )
171   {
172     return mImageFactory.GetActualAttributes( mTicket );
173   }
174   else
175   {
176     return mImageFactory.GetRequestAttributes( mRequest );
177   }
178 }
179
180 const std::string& Image::GetFilename() const
181 {
182   return mImageFactory.GetRequestPath( mRequest );
183 }
184
185 void Image::Reload()
186 {
187   if ( mRequest )
188   {
189     ResourceTicketPtr ticket = mImageFactory.Reload( mRequest.Get() );
190     SetTicket( ticket.Get() );
191   }
192 }
193
194 void Image::ResourceLoadingFailed(const ResourceTicket& ticket)
195 {
196   mLoadingFinishedV2.Emit( Dali::Image( this ) );
197 }
198
199 void Image::ResourceLoadingSucceeded(const ResourceTicket& ticket)
200 {
201   mLoadingFinishedV2.Emit( Dali::Image( this ) );
202 }
203
204 void Image::ResourceUploaded(const ResourceTicket& ticket)
205 {
206   mUploadedV2.Emit( Dali::Image( this ) );
207 }
208
209 void Image::ResourceSavingSucceeded( const ResourceTicket& ticket )
210 {
211   // do nothing
212 }
213
214 void Image::ResourceSavingFailed( const ResourceTicket& ticket )
215 {
216   // do nothing
217 }
218
219 unsigned int Image::GetWidth() const
220 {
221   // if width is 0, it means we've not yet loaded the image
222   if( 0u == mWidth )
223   {
224     Size size;
225     mImageFactory.GetImageSize( mRequest, mTicket, size );
226     mWidth = size.width;
227     // The app will probably ask for the height immediately, so don't waste the synchronous file IO that ImageFactory may have just done:
228     DALI_ASSERT_DEBUG( 0 == mHeight || unsigned(size.height) == mHeight );
229     if( 0 == mHeight )
230     {
231       mHeight = size.height;
232     }
233   }
234   return mWidth;
235 }
236
237 unsigned int Image::GetHeight() const
238 {
239   if( 0u == mHeight )
240   {
241     Size size;
242     mImageFactory.GetImageSize( mRequest, mTicket, size );
243     mHeight = size.height;
244     DALI_ASSERT_DEBUG( 0 == mWidth || unsigned(size.width) == mWidth );
245     if( 0 == mWidth )
246     {
247       mWidth = size.width;
248     }
249   }
250   return mHeight;
251 }
252
253 Vector2 Image::GetNaturalSize() const
254 {
255   Vector2 naturalSize(mWidth, mHeight);
256   if( 0u == mWidth || 0u == mHeight )
257   {
258     mImageFactory.GetImageSize( mRequest, mTicket, naturalSize );
259     mWidth = naturalSize.width;
260     mHeight = naturalSize.height;
261   }
262   return naturalSize;
263 }
264
265 void Image::Connect()
266 {
267   ++mConnectionCount;
268
269   if( mConnectionCount == 1 )
270   {
271     // ticket was thrown away when related actors went offstage or image loading on demand
272     if( !mTicket )
273     {
274       ResourceTicketPtr newTicket = mImageFactory.Load( mRequest.Get() );
275       SetTicket( newTicket.Get() );
276     }
277   }
278 }
279
280 void Image::Disconnect()
281 {
282   if( !mTicket )
283   {
284     return;
285   }
286
287   DALI_ASSERT_DEBUG( mConnectionCount > 0 );
288   --mConnectionCount;
289   if( mConnectionCount == 0 && mReleasePolicy == Dali::Image::Unused )
290   {
291     // release image memory when it's not visible anymore (decrease ref. count of texture)
292     SetTicket( NULL );
293   }
294 }
295
296 void Image::Initialize()
297 {
298   RegisterObject();
299 }
300
301 void Image::SetTicket( ResourceTicket* ticket )
302 {
303   if( ticket == mTicket.Get() )
304   {
305     return;
306   }
307
308   if( mTicket )
309   {
310     mTicket->RemoveObserver( *this );
311     mImageFactory.ReleaseTicket( mTicket.Get() );
312   }
313
314   if( ticket )
315   {
316     mTicket.Reset( ticket );
317     mTicket->AddObserver( *this );
318   }
319   else
320   {
321     mTicket.Reset();
322   }
323 }
324
325 bool Image::IsNinePatchFileName( const std::string& filename )
326 {
327   bool match = false;
328
329   std::string::const_reverse_iterator iter = filename.rbegin();
330   enum { SUFFIX, HASH, HASH_DOT, DONE } state = SUFFIX;
331   while(iter < filename.rend())
332   {
333     switch(state)
334     {
335       case SUFFIX:
336       {
337         if(*iter == '.')
338         {
339           state = HASH;
340         }
341         else if(!isalnum(*iter))
342         {
343           state = DONE;
344         }
345       }
346       break;
347       case HASH:
348       {
349         if( *iter == '#' || *iter == '9' )
350         {
351           state = HASH_DOT;
352         }
353         else
354         {
355           state = DONE;
356         }
357       }
358       break;
359       case HASH_DOT:
360       {
361         if(*iter == '.')
362         {
363           match = true;
364         }
365         state = DONE; // Stop testing characters
366       }
367       break;
368       case DONE:
369       {
370       }
371       break;
372     }
373
374     // Satisfy prevent
375     if( state == DONE )
376     {
377       break;
378     }
379
380     ++iter;
381   }
382   return match;
383 }
384
385
386 } // namespace Internal
387
388 } // namespace Dali