[dali_1.0.17] Merge branch 'tizen'
[platform/core/uifw/dali-demo.git] / examples / builder / examples.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 //------------------------------------------------------------------------------
19 //
20 //
21 //------------------------------------------------------------------------------
22
23 #include "dali.h"
24 #include <dali-toolkit/dali-toolkit.h>
25 #include <dali-toolkit/public-api/builder/builder.h>
26 #include <dali-toolkit/public-api/builder/tree-node.h>
27 #include <dali-toolkit/public-api/builder/json-parser.h>
28 #include <map>
29 #include <string>
30 #include <fstream>
31 #include <streambuf>
32 #include <sstream>
33 #include <boost/scoped_ptr.hpp>
34 #include <dirent.h>
35 #include <stdio.h>
36
37 //#include <boost/regex.hpp>
38 #include "sys/stat.h"
39 #include <ctime>
40
41 #include <dali/integration-api/debug.h>
42 #include "../shared/view.h"
43
44 #define TOKEN_STRING(x) #x
45
46 using namespace Dali;
47 using namespace Dali::Toolkit;
48
49 namespace
50 {
51
52 const char* BACKGROUND_IMAGE( "" );
53 const char* TOOLBAR_IMAGE( DALI_IMAGE_DIR "top-bar.png" );
54 const char* EDIT_IMAGE( DALI_IMAGE_DIR "icon-change.png" );
55
56 std::string USER_DIRECTORY;
57
58 std::string JSON_BROKEN("                                      \
59 {                                                              \
60   'stage':                                                     \
61   [                                                            \
62     {                                                          \
63       'type':'TextView',                                       \
64       'size': [50,50,1],                                       \
65       'parent-origin': 'CENTER',                               \
66       'text':'COULD NOT LOAD JSON FILE'                        \
67     }                                                          \
68   ]                                                            \
69 }                                                              \
70 ");
71
72 std::string ReplaceQuotes(const std::string &single_quoted)
73 {
74   std::string s(single_quoted);
75
76   // wrong as no embedded quote but had regex link problems
77   std::replace(s.begin(), s.end(), '\'', '"');
78
79   return s;
80 }
81
82 std::string GetFileContents(const std::string &fn)
83 {
84   std::ifstream t(fn.c_str());
85   return std::string((std::istreambuf_iterator<char>(t)),
86                      std::istreambuf_iterator<char>());
87 };
88
89 typedef std::vector<std::string> FileList;
90
91 void DirectoryFileList(const std::string& directory, FileList& files)
92 {
93   DIR           *d;
94   struct dirent *dir;
95   d = opendir(directory.c_str());
96   if (d)
97   {
98     while ((dir = readdir(d)) != NULL)
99     {
100       if (dir->d_type == DT_REG)
101       {
102         files.push_back( directory + std::string(dir->d_name) );
103       }
104     }
105
106     closedir(d);
107   }
108 }
109
110 void DirectoryFilesByType(const std::string& dir, const std::string& fileType /* ie "json" */, FileList& files)
111 {
112   typedef FileList Collection;
113   typedef FileList::iterator Iter;
114
115   Collection allFiles;
116   DirectoryFileList(dir, allFiles);
117
118   for(Iter iter = allFiles.begin(); iter != allFiles.end(); ++iter)
119   {
120     size_t pos = (*iter).rfind( '.' );
121     if( pos != std::string::npos )
122     {
123       if( (*iter).substr( pos+1 ) == fileType )
124       {
125         files.push_back( (*iter) );
126       }
127     }
128   }
129 }
130
131 const std::string ShortName( const std::string& name )
132 {
133   size_t pos = name.rfind( '/' );
134
135   if( pos != std::string::npos )
136   {
137     return name.substr( pos );
138   }
139   else
140   {
141     return name;
142   }
143 }
144
145 static Vector3 SetItemSize(unsigned int numberOfColumns, float layoutWidth, float sideMargin, float columnSpacing)
146 {
147   return Vector3(layoutWidth, 50, 1);
148 }
149
150 //------------------------------------------------------------------------------
151 //
152 //
153 //
154 //------------------------------------------------------------------------------
155 class FileWatcher
156 {
157 public:
158   FileWatcher(void);
159   ~FileWatcher(void);
160   explicit FileWatcher(const std::string &fn) { SetFilename(fn) ; };
161
162   void SetFilename(const std::string &fn);
163   std::string GetFilename() const;
164
165   bool FileHasChanged(void);
166   std::string GetFileContents(void) const { return ::GetFileContents(mstringPath) ; };
167
168 private:
169   // compiler does
170   // FileWatcher(const FileWatcher&);
171   // FileWatcher &operator=(const FileWatcher &);
172
173   std::time_t mLastTime;
174   std::string mstringPath;
175
176 };
177
178 FileWatcher::FileWatcher(void) : mLastTime(0)
179 {
180 }
181
182 bool FileWatcher::FileHasChanged(void)
183 {
184   struct stat buf;
185
186   if(0 != stat(mstringPath.c_str(), &buf))
187   {
188     return false;
189   }
190   else
191   {
192     if(buf.st_mtime > mLastTime)
193     {
194       mLastTime = buf.st_mtime;
195       return true;
196     }
197     else
198     {
199       mLastTime = buf.st_mtime;
200       return false;
201     }
202   }
203
204   return false;
205 }
206
207 FileWatcher::~FileWatcher()
208 {
209 }
210
211 void FileWatcher::SetFilename(const std::string &fn)
212 {
213   mstringPath = fn;
214   FileHasChanged(); // update last time
215 }
216
217 std::string FileWatcher::GetFilename(void) const
218 {
219   return mstringPath;
220 }
221
222
223 } // anon namespace
224
225
226 //------------------------------------------------------------------------------
227 //
228 //
229 //
230 //------------------------------------------------------------------------------
231 class ExampleApp : public ConnectionTracker, public Toolkit::ItemFactory
232 {
233 public:
234   ExampleApp(Application &app) : mApp(app)
235   {
236     app.InitSignal().Connect(this, &ExampleApp::Create);
237   }
238
239   ~ExampleApp() {}
240
241 public:
242
243   void SetTitle(const std::string& title)
244   {
245     if(!mTitleActor)
246     {
247       mTitleActor = TextView::New();
248       // Add title to the tool bar.
249       mToolBar.AddControl( mTitleActor, DemoHelper::DEFAULT_VIEW_STYLE.mToolBarTitlePercentage, Alignment::HorizontalCenter );
250     }
251
252     Font font = Font::New();
253     mTitleActor.SetText( title );
254     mTitleActor.SetSize( font.MeasureText( title ) );
255     mTitleActor.SetStyleToCurrentText(DemoHelper::GetDefaultTextStyle());
256   }
257
258   bool OnToolSelectLayout( Toolkit::Button button )
259   {
260     bool on = mItemView.IsVisible();
261
262     if( on )
263     {
264       LeaveSelection();
265     }
266     else
267     {
268       EnterSelection();
269     }
270
271     return true;
272   }
273
274   void LeaveSelection()
275   {
276
277   }
278
279   void EnterSelection()
280   {
281     Stage stage = Stage::GetCurrent();
282
283     if( mItemView )
284     {
285       stage.Remove( mItemView );
286     }
287
288     mFiles.clear();
289
290     mItemView = ItemView::New(*this);
291     stage.Add( mItemView );
292     mItemView.SetParentOrigin(ParentOrigin::CENTER);
293     mItemView.SetAnchorPoint(AnchorPoint::CENTER);
294     mGridLayout = GridLayout::New();
295     mGridLayout->SetNumberOfColumns(1);
296
297     mGridLayout->SetItemSizeFunction(SetItemSize);
298
299     mGridLayout->SetTopMargin(DemoHelper::DEFAULT_VIEW_STYLE.mToolBarHeight);
300
301     mItemView.AddLayout(*mGridLayout);
302
303     Vector3 size(stage.GetSize());
304     mItemView.ActivateLayout(0, size, 0.0f/*immediate*/);
305     mItemView.SetKeyboardFocusable( true );
306
307     mFiles.clear();
308     FileList files;
309
310     if( USER_DIRECTORY.size() )
311     {
312       DirectoryFilesByType( USER_DIRECTORY, "json", files );
313     }
314     else
315     {
316       DirectoryFilesByType( DALI_SCRIPT_DIR, "json", files );
317     }
318
319     std::sort(files.begin(), files.end());
320
321     ItemId itemId = 0;
322     for(FileList::iterator iter = files.begin(); iter != files.end(); ++iter)
323     {
324       JsonParser parser = JsonParser::New();
325
326       std::string data( GetFileContents( *iter ) );
327
328       parser.Parse( data );
329
330       if( parser.ParseError() )
331       {
332         std::cout << "Parser Error:" << *iter << std::endl;
333         std::cout << parser.GetErrorLineNumber() << "(" << parser.GetErrorColumn() << "):" << parser.GetErrorDescription() << std::endl;
334         exit(1);
335       }
336
337       if( parser.GetRoot() )
338       {
339         if( const TreeNode* node = parser.GetRoot()->Find("stage") )
340         {
341           // only those with a stage section
342           if( node->Size() )
343           {
344             mFiles.push_back( *iter );
345
346             mItemView.InsertItem( Item(itemId,
347                                        MenuItem( ShortName( *iter ) ) ),
348                                   0.5f );
349
350             itemId++;
351           }
352           else
353           {
354             std::cout << "Ignored file (stage has no nodes?):" << *iter << std::endl;
355           }
356         }
357         else
358         {
359           std::cout << "Ignored file (no stage section):" << *iter << std::endl;
360         }
361       }
362     }
363
364     mTapDetector = TapGestureDetector::New();
365
366     for( unsigned int i = 0u; i < mItemView.GetChildCount(); ++i )
367     {
368       mTapDetector.Attach( mItemView.GetChildAt(i) );
369     }
370
371     mTapDetector.DetectedSignal().Connect( this, &ExampleApp::OnTap );
372
373     // Display item view on the stage
374     stage.Add( mItemView );
375
376     mItemView.SetVisible( true );
377     mBuilderLayer.SetVisible( false );
378
379     SetTitle("Select");
380
381     // Itemview renderes the previous items unless its scrolled. Not sure why at the moment so we force a scroll
382     mItemView.ScrollToItem(0, 0);
383
384   }
385
386   void ExitSelection()
387   {
388     mTapDetector.Reset();
389
390     mItemView.SetVisible( false );
391     mBuilderLayer.SetVisible( true );
392
393     SetTitle("View");
394   }
395
396   void OnTap( Actor actor, const TapGesture& tap )
397   {
398     ItemId id = mItemView.GetItemId( actor );
399
400     LoadFromFileList( id );
401   }
402
403   Actor MenuItem(const std::string& text)
404   {
405     TextView t = TextView::New();
406     t.SetMarkupProcessingEnabled(true);
407
408     int size = static_cast<int>(DemoHelper::ScalePointSize(6));
409
410     std::ostringstream fontString;
411     fontString << "<font size="<< size <<">"<<  ShortName( text ) << "</font>";
412
413     t.SetText( fontString.str() );
414
415     t.SetTextAlignment( Alignment::HorizontalLeft );
416     return t;
417   }
418
419   bool OnTimer()
420   {
421     if( mFileWatcher.FileHasChanged() )
422     {
423       LoadFromFile( mFileWatcher.GetFilename() );
424     }
425
426     return true;
427   }
428
429   void ReloadJsonFile(const std::string& filename, Builder& builder, Layer& layer)
430   {
431     Stage stage = Stage::GetCurrent();
432
433     builder = Builder::New();
434     builder.QuitSignal().Connect( this, &ExampleApp::OnBuilderQuit );
435
436     Property::Map defaultDirs;
437     defaultDirs[ TOKEN_STRING(DALI_IMAGE_DIR) ]  = DALI_IMAGE_DIR;
438     defaultDirs[ TOKEN_STRING(DALI_MODEL_DIR) ]  = DALI_MODEL_DIR;
439     defaultDirs[ TOKEN_STRING(DALI_SCRIPT_DIR) ] = DALI_SCRIPT_DIR;
440
441     builder.AddConstants( defaultDirs );
442
443     // render tasks may have been setup last load so remove them
444     RenderTaskList taskList = stage.GetRenderTaskList();
445     if( taskList.GetTaskCount() > 1 )
446     {
447       typedef std::vector<RenderTask> Collection;
448       typedef Collection::iterator ColIter;
449       Collection tasks;
450
451       for(unsigned int i = 1; i < taskList.GetTaskCount(); ++i)
452       {
453         tasks.push_back( taskList.GetTask(i) );
454       }
455
456       for(ColIter iter = tasks.begin(); iter != tasks.end(); ++iter)
457       {
458         taskList.RemoveTask(*iter);
459       }
460
461       RenderTask defaultTask = taskList.GetTask(0);
462       defaultTask.SetSourceActor( stage.GetRootLayer() );
463       defaultTask.SetTargetFrameBuffer( FrameBufferImage() );
464     }
465
466     unsigned int numChildren = layer.GetChildCount();
467
468     for(unsigned int i=0; i<numChildren; ++i)
469     {
470       layer.Remove( layer.GetChildAt(0) );
471     }
472
473     std::string data(GetFileContents(filename));
474
475     try
476     {
477       builder.LoadFromString(data);
478     }
479     catch(...)
480     {
481       builder.LoadFromString(ReplaceQuotes(JSON_BROKEN));
482     }
483
484     builder.AddActors( layer );
485
486   }
487
488
489   void LoadFromFileList( size_t index )
490   {
491     if( index < mFiles.size())
492     {
493       const std::string& name = mFiles[index];
494       mFileWatcher.SetFilename( name );
495       LoadFromFile( name );
496     }
497   }
498
499   void LoadFromFile( const std::string& name )
500   {
501     ReloadJsonFile( name, mBuilder, mBuilderLayer );
502
503     // do this here as GetCurrentSize()
504     mBuilderLayer.SetParentOrigin(ParentOrigin::CENTER);
505     mBuilderLayer.SetAnchorPoint(AnchorPoint::CENTER);
506     Dali::Vector3 size = Stage::GetCurrent().GetRootLayer().GetCurrentSize();
507     size.y -= DemoHelper::DEFAULT_VIEW_STYLE.mToolBarHeight;
508     mBuilderLayer.SetSize( size );
509
510     mBuilderLayer.LowerToBottom();
511     Stage::GetCurrent().GetRootLayer().RaiseToTop();
512
513     ExitSelection();
514   }
515
516   void Create(Application& app)
517   {
518     Stage stage = Stage::GetCurrent();
519
520     Stage::GetCurrent().KeyEventSignal().Connect(this, &ExampleApp::OnKeyEvent);
521
522     Layer contents = DemoHelper::CreateView( app,
523                                              mView,
524                                              mToolBar,
525                                              BACKGROUND_IMAGE,
526                                              TOOLBAR_IMAGE,
527                                              "" );
528
529     SetTitle("Builder");
530
531     mBuilderLayer = Layer::New();
532     stage.GetRootLayer().Add(mBuilderLayer);
533
534
535     // Create an edit mode button. (left of toolbar)
536     Toolkit::PushButton editButton = Toolkit::PushButton::New();
537     editButton.SetBackgroundImage( Image::New( EDIT_IMAGE ) );
538     editButton.ClickedSignal().Connect( this, &ExampleApp::OnToolSelectLayout);
539     editButton.SetLeaveRequired( true );
540     mToolBar.AddControl( editButton, DemoHelper::DEFAULT_VIEW_STYLE.mToolBarButtonPercentage, Toolkit::Alignment::HorizontalLeft, DemoHelper::DEFAULT_MODE_SWITCH_PADDING  );
541
542     EnterSelection();
543
544     mTimer = Timer::New( 500 ); // ms
545     mTimer.TickSignal().Connect( this, &ExampleApp::OnTimer);
546     mTimer.Start();
547
548   } // Create(app)
549
550   virtual unsigned int GetNumberOfItems()
551   {
552     return mFiles.size();
553   }
554
555   virtual Actor NewItem(unsigned int itemId)
556   {
557     DALI_ASSERT_DEBUG( itemId < mFiles.size() );
558     return MenuItem( ShortName( mFiles[itemId] ) );
559   }
560
561   /**
562    * Main key event handler
563    */
564   void OnKeyEvent(const KeyEvent& event)
565   {
566     if(event.state == KeyEvent::Down)
567     {
568       if( IsKey( event, Dali::DALI_KEY_ESCAPE) || IsKey( event, Dali::DALI_KEY_BACK) )
569       {
570         if ( mItemView.IsVisible() )
571         {
572           mApp.Quit();
573         }
574         else
575         {
576           EnterSelection();
577         }
578       }
579     }
580   }
581
582   /**
583    * Event handler when Builder wants to quit (we only want to close the shown json unless we're at the top-level)
584    */
585   void OnBuilderQuit()
586   {
587     if ( mItemView.IsVisible() )
588     {
589       mApp.Quit();
590     }
591     else
592     {
593       EnterSelection();
594     }
595   }
596
597 private:
598   Application& mApp;
599
600   GridLayoutPtr mGridLayout;
601   ItemView mItemView;
602
603   Toolkit::View mView;
604   unsigned int mOrientation;
605
606   Toolkit::ToolBar mToolBar;
607   TextView mTitleActor;             ///< The Toolbar's Title.
608
609   Layer mBuilderLayer;
610
611   Toolkit::Popup mMenu;
612
613   TapGestureDetector mTapDetector;
614
615   // builder
616   Builder mBuilder;
617
618   FileList mFiles;
619
620   FileWatcher mFileWatcher;
621   Timer mTimer;
622
623
624 };
625
626 //------------------------------------------------------------------------------
627 //
628 //
629 //
630 //------------------------------------------------------------------------------
631 int main(int argc, char **argv)
632 {
633   if(argc > 2)
634   {
635     if(strcmp(argv[1], "-f") == 0)
636     {
637       USER_DIRECTORY = argv[2];
638     }
639   }
640
641   Application app = Application::New(&argc, &argv);
642
643   ExampleApp dali_app(app);
644
645   app.MainLoop();
646
647   return 0;
648 }