[dali_1.0.10] Merge branch 'tizen'
[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   }
228   return mWidth;
229 }
230
231 unsigned int Image::GetHeight() const
232 {
233   if( 0u == mHeight )
234   {
235     Size size;
236     mImageFactory.GetImageSize( mRequest, mTicket, size );
237     mHeight = size.height;
238   }
239   return mHeight;
240 }
241
242 Vector2 Image::GetNaturalSize() const
243 {
244   Vector2 naturalSize(mWidth, mHeight);
245   if( 0u == mWidth || 0u == mHeight )
246   {
247     mImageFactory.GetImageSize( mRequest, mTicket, naturalSize );
248     mWidth = naturalSize.width;
249     mHeight = naturalSize.height;
250   }
251   return naturalSize;
252 }
253
254 void Image::Connect()
255 {
256   ++mConnectionCount;
257
258   if( mConnectionCount == 1 )
259   {
260     // ticket was thrown away when related actors went offstage or image loading on demand
261     if( !mTicket )
262     {
263       ResourceTicketPtr newTicket = mImageFactory.Load( mRequest.Get() );
264       SetTicket( newTicket.Get() );
265     }
266   }
267 }
268
269 void Image::Disconnect()
270 {
271   if( !mTicket )
272   {
273     return;
274   }
275
276   DALI_ASSERT_DEBUG( mConnectionCount > 0 );
277   --mConnectionCount;
278   if( mConnectionCount == 0 && mReleasePolicy == Dali::Image::Unused )
279   {
280     // release image memory when it's not visible anymore (decrease ref. count of texture)
281     SetTicket( NULL );
282   }
283 }
284
285 void Image::Initialize()
286 {
287   RegisterObject();
288 }
289
290 void Image::SetTicket( ResourceTicket* ticket )
291 {
292   if( ticket == mTicket.Get() )
293   {
294     return;
295   }
296
297   if( mTicket )
298   {
299     mTicket->RemoveObserver( *this );
300     mImageFactory.ReleaseTicket( mTicket.Get() );
301   }
302
303   if( ticket )
304   {
305     mTicket.Reset( ticket );
306     mTicket->AddObserver( *this );
307   }
308   else
309   {
310     mTicket.Reset();
311   }
312 }
313
314 bool Image::IsNinePatchFileName( const std::string& filename )
315 {
316   bool match = false;
317
318   std::string::const_reverse_iterator iter = filename.rbegin();
319   enum { SUFFIX, HASH, HASH_DOT, DONE } state = SUFFIX;
320   while(iter < filename.rend())
321   {
322     switch(state)
323     {
324       case SUFFIX:
325       {
326         if(*iter == '.')
327         {
328           state = HASH;
329         }
330         else if(!isalnum(*iter))
331         {
332           state = DONE;
333         }
334       }
335       break;
336       case HASH:
337       {
338         if( *iter == '#' || *iter == '9' )
339         {
340           state = HASH_DOT;
341         }
342         else
343         {
344           state = DONE;
345         }
346       }
347       break;
348       case HASH_DOT:
349       {
350         if(*iter == '.')
351         {
352           match = true;
353         }
354         state = DONE; // Stop testing characters
355       }
356       break;
357       case DONE:
358       {
359       }
360       break;
361     }
362
363     // Satisfy prevent
364     if( state == DONE )
365     {
366       break;
367     }
368
369     iter++;
370   }
371   return match;
372 }
373
374
375 } // namespace Internal
376
377 } // namespace Dali