|
1 # |
|
2 # Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies). |
|
3 # All rights reserved. |
|
4 # This component and the accompanying materials are made available |
|
5 # under the terms of "Eclipse Public License v1.0" |
|
6 # which accompanies this distribution, and is available |
|
7 # at the URL "http://www.eclipse.org/legal/epl-v10.html". |
|
8 # |
|
9 # Initial Contributors: |
|
10 # Nokia Corporation - initial contribution. |
|
11 # |
|
12 # Contributors: |
|
13 # |
|
14 # Description: |
|
15 # |
|
16 |
|
17 use strict; |
|
18 use warnings; |
|
19 use File::Spec::Functions; |
|
20 use Getopt::Long; |
|
21 use Pod::Usage; |
|
22 |
|
23 # Version of the script - just use the date |
|
24 $main::VERSION = '12-July-2010'; |
|
25 |
|
26 # New getopt::long provides this - but old version doesn't? |
|
27 sub version |
|
28 { |
|
29 print sprintf("$0: $main::VERSION\n$^X: %vd\nos: $^O", $^V); |
|
30 exit; |
|
31 } |
|
32 |
|
33 # Get arguments |
|
34 my ( $help, $version, $verbose, $debug, $epocroot, $l10n ); |
|
35 GetOptions( "help" => \$help, "version|ver" => \$version, |
|
36 "verbose|v" => \$verbose, "debug" => \$debug, |
|
37 "epocroot|e=s" => \$epocroot, "localization|l=s" => \$l10n ) |
|
38 or pod2usage('Invalid parameters'); |
|
39 |
|
40 # Handle help and version |
|
41 pod2usage({ verbose => 1, exitval => 0}) if $help; |
|
42 version() if $version; |
|
43 |
|
44 # Interpret arguments |
|
45 $verbose = 1 if $debug; |
|
46 $verbose = $verbose ? '-v' : ''; |
|
47 $epocroot = $ENV{EPOCROOT} if !$epocroot; |
|
48 |
|
49 # Interpret localization l10n argument |
|
50 |
|
51 # There are two modes. In one mode you specify the location of a |
|
52 # lproj.xml file and the code computes l10n files for all languages so |
|
53 # that runtime locale selection should work. In the other mode you |
|
54 # specify a language code (example, fr for French) and the code |
|
55 # computes l10n just for that language. |
|
56 $l10n = '' if !$l10n; |
|
57 my $lproj = ''; |
|
58 if ($l10n =~ /lproj\.xml$/i) |
|
59 { |
|
60 $lproj = $l10n; |
|
61 $l10n = ''; |
|
62 } |
|
63 |
|
64 my ($configini, @unnecessary) = @ARGV; |
|
65 |
|
66 if ($configini) |
|
67 { |
|
68 # There shouldn't be any need for any other arguments |
|
69 pod2usage('Unnecessary arguments supplied.') if @unnecessary; |
|
70 |
|
71 # Now do the installation |
|
72 my $installer = WidgetInstaller->new('verbose' => $verbose, 'debug' => $debug, 'epocroot' => $epocroot, 'lproj' => $lproj, 'l10n' => $l10n); |
|
73 $installer->installConfig($configini); |
|
74 } |
|
75 else |
|
76 { |
|
77 pod2usage('Missing config.ini argument.'); |
|
78 } |
|
79 |
|
80 # *** |
|
81 # Package for installing Widgets |
|
82 # |
|
83 |
|
84 package WidgetInstaller; |
|
85 |
|
86 use File::Spec::Functions; |
|
87 use File::Path; |
|
88 use File::Basename; |
|
89 use Unicode::String qw(utf8 utf16); |
|
90 use File::Temp qw/tempdir/; |
|
91 use File::Copy; |
|
92 use XML::Parser; |
|
93 use Data::Dumper; |
|
94 use XML::Simple; |
|
95 |
|
96 # CONSTANTS |
|
97 use constant ICON_SIZES => "88,32,24"; # Size of icons to generate in the MBM file |
|
98 #use constant INTERNAL_UID_LOWER => 0x2000DAD2; # Lower UID bound for midlets installed to internal drives and rom |
|
99 #use constant EXTERNAL_UID_LOWER => 0x2000DCC6; # Lower UID bound for midlets installed to removable cards |
|
100 use constant INTERNAL_UID_LOWER_WRT => 0x2000DAD2; # Lower UID bound for WRT widgets installed to internal drives and rom |
|
101 use constant INTERNAL_UID_LOWER_CWRT => 0x2000DBCC; # Lower UID bound for CWRT widgets installed to internal drives and rom |
|
102 use constant EXTERNAL_UID_LOWER_WRT => 0x2000DCC6; # Lower UID bound for WRT widgets installed to removable cards |
|
103 use constant EXTERNAL_UID_LOWER_CWRT => 0x2000DDC0; # Lower UID bound for CWRT widgets installed to removable cards |
|
104 use constant WIDGET_UI_UID => 0x10282822; # UID of the widget app |
|
105 use constant CWRT_WIDGET_UI_UID => 0x200267C0; # UID of the cwrt widget app |
|
106 use constant SWIDGET_UI_UID => 0x102829A0; # UID of the securewidget app |
|
107 use constant CWRT_WIDGET_UI_NON_NOKIA_UID => 0x200267D6; # UID of the cwrt widget app |
|
108 |
|
109 # Folder paths |
|
110 use constant DESTINATION => 'private/10003a3f/import/apps/NonNative/Resource'; |
|
111 use constant ROM_DEST => 'epoc32/release/winscw/udeb/Z/'; |
|
112 use constant DRIVE_DEST => 'epoc32/winscw/%s'; |
|
113 use constant WIDGET_REGISTRY => 'private/10282f06/WidgetEntryStore.xml'; |
|
114 use constant CWRT_WIDGET_REGISTRY => 'private/10282f06/CWRTWidgetEntryStore.xml'; |
|
115 use constant CWRT_WEBAPP_REGISTRY => 'private/200267D9/webapp_registry.db'; |
|
116 use constant CWRT_ACCESS_POLICY_PATH => 'browser_access_policy.xml'; |
|
117 use constant WIDGET_NAME => 'en-gb/widget/name'; |
|
118 use constant WIDGET_PROCESSUID => 'processUid'; |
|
119 use constant WIDGET_ID => 'widget/id'; |
|
120 use constant WIDGET_VERSION => 'widget/version'; |
|
121 use constant WIDGET_CONTENT => 'widget/content'; |
|
122 use constant WIDGET_FEATURE => 'widget/feature/'; |
|
123 use constant WIDGET_FEATURE_PARAM => '/param/'; |
|
124 use constant WIDGET_ACCESS => 'widget/access/'; |
|
125 use constant WIDGET_HEIGHT => 'widget/height'; |
|
126 use constant WIDGET_WIDTH => 'widget/width'; |
|
127 use constant WIDGET_NOKIA_SHAREDLIB => 'widget/NOKIA:sharedlibrary'; |
|
128 use constant WIDGET_NOKIA_SHAREDLIB_FOLDER => 'widget/NOKIA:sharedlibrary/NOKIA:folder'; |
|
129 use constant WIDGET_NOKIA_SHAREDLIB_WIDGET => 'widget/NOKIA:sharedlibrary/NOKIA:widget'; |
|
130 use constant WIDGET_NOKIA_SHAREDLIB_ICON => 'widget/NOKIA:sharedlibrary/NOKIA:icon'; |
|
131 use constant KEY_VALUE_SEPERATOR => ','; |
|
132 use constant KEY_VALUE_PAIR_SEPERATOR => ';'; |
|
133 use constant KEY_ATTR_SEPERATOR => ':'; |
|
134 |
|
135 our $w3c_widget = 0 ; |
|
136 our $hashval; |
|
137 our $non_nokia_widget = 0; |
|
138 our ($isSharedLibrary, $sharedFolderName ,$isSharedWidget, $isSharedIcon)= (0,'',0,0); |
|
139 our (@staticUidWrtIntList,@staticUidCwrtIntList,@staticUidWrtExtList,@staticUidCwrtExtList) = ({},{},{},{}); |
|
140 our $isSecureWidget = 0; |
|
141 |
|
142 # Create a new object |
|
143 sub new |
|
144 { |
|
145 my $invocant = shift; |
|
146 my $self = bless({}, ref $invocant || $invocant); |
|
147 |
|
148 my %args = @_; |
|
149 $self->{'args'} = \%args; |
|
150 $self->{'args'}->{'verbose'} = $self->{'args'}->{'verbose'} ? '-v' : ''; |
|
151 |
|
152 $self->{'installedFiles'} = (); |
|
153 $self->{'freeuidwrtint'} = INTERNAL_UID_LOWER_WRT; |
|
154 $self->{'freeuidwrtext'} = EXTERNAL_UID_LOWER_WRT; |
|
155 $self->{'freeuidcwrtint'} = INTERNAL_UID_LOWER_CWRT; |
|
156 $self->{'freeuidcwrtext'} = EXTERNAL_UID_LOWER_CWRT; |
|
157 $self->{'langmapping'} = $self->getLangMapping($self->{'args'}->{'lproj'}) unless $self->{'args'}->{'l10n'}; |
|
158 |
|
159 #Check if localisation argument is land number ID; If true, get the lang Name using langmapping file - Widget_lproj.xml |
|
160 if ($l10n =~ /\d+$/) |
|
161 { |
|
162 print "\nLANG ID given as localisation argument :: $l10n \n"; |
|
163 $self->{'langmapping'} = $self->getLangMapping(); |
|
164 |
|
165 if($self->{'langmapping'}) |
|
166 { |
|
167 # Iterate through all the languages we know about |
|
168 for(my $i = 0; $i < scalar(@{ $self->{'langmapping'}->{LangID} }); $i++) |
|
169 { |
|
170 my ( $langid, $langname ) = ( $self->{'langmapping'}->{LangID}->[$i], $self->{'langmapping'}->{LangDir}->[$i] ); |
|
171 if($l10n == $langid) |
|
172 { |
|
173 print "Setting argument to LangName - $langname \n"; |
|
174 $self->{'args'}->{'l10n'} = lc($langname); |
|
175 last; |
|
176 } |
|
177 } |
|
178 } |
|
179 } |
|
180 |
|
181 return $self; |
|
182 } |
|
183 |
|
184 # Gets the language mapping from the Widget_lproj.xml file |
|
185 sub getLangMapping |
|
186 { |
|
187 my ( $self, $lproj ) = @_; |
|
188 |
|
189 # Get the LPROJ file which specifies lang id mappings |
|
190 if (!$lproj) |
|
191 { |
|
192 (my $EPOCROOT = $ENV{EPOCROOT}) =~ s/[\/\\]+$//; |
|
193 $lproj = WidgetInstaller::fixFilename("${epocroot}epoc32/winscw/c/private/10282f06/Widget_lproj.xml"); |
|
194 $lproj = WidgetInstaller::fixFilename("$EPOCROOT/epoc32/winscw/c/private/10282f06/Widget_lproj.xml") if !-e $lproj; |
|
195 $lproj = WidgetInstaller::fixFilename("${epocroot}epoc32/data/Z/private/10282f06/Widget_lproj.xml") if !-e $lproj; |
|
196 $lproj = WidgetInstaller::fixFilename("$EPOCROOT/epoc32/data/Z/private/10282f06/Widget_lproj.xml") if !-e $lproj; |
|
197 $lproj = WidgetInstaller::fixFilename("${epocroot}S60/mw/web/WebEngine/WidgetRegistry/Data/Widget_lproj.xml") if !-e $lproj; |
|
198 $lproj = WidgetInstaller::fixFilename("$EPOCROOT/S60/mw/web/WebEngine/WidgetRegistry/Data/Widget_lproj.xml") if !-e $lproj; |
|
199 $lproj = WidgetInstaller::findCmd('Widget_lproj.xml') if !-e $lproj; |
|
200 $lproj = WidgetInstaller::findCmd('internal/Widget_lproj.xml') if !-e $lproj; |
|
201 undef $lproj if !-e $lproj; |
|
202 |
|
203 # Display a warning if localisation can't be performed |
|
204 warn "WARNING: Can't find Widget_lproj.xml - localisation is disabled.\n" if !$lproj; |
|
205 } |
|
206 |
|
207 # Load the mapping file |
|
208 if ($lproj) |
|
209 { |
|
210 die "Can't find $lproj file." if !-e $lproj; |
|
211 print "Localisation support enabled using config: $lproj\n" if $verbose; |
|
212 my $mapping = XMLin($lproj); |
|
213 print "Found ", scalar(@{ $mapping->{'LangID'} }), " language mappings.\n" if $verbose; |
|
214 return $mapping; |
|
215 } |
|
216 } |
|
217 |
|
218 # Install Widgets listed in config file |
|
219 # Format is as follows where "drive-z" specifies widgets for drive z: etc... |
|
220 # Comments are okay - they start with a # character |
|
221 # If the file doesn't exist as given then EPOCROOT is prepended to the filename to try and find it |
|
222 # |
|
223 # You can specify whether a widget is intended for the homescreen by adding the text [HomeScreen] after the filename |
|
224 # This will set the BlanketPermissionGranted attribute in the registry. |
|
225 # Widgets intended for the homescreen must have the MiniViewEnabled attribute set in its PLIST file otherwise an error is generated. |
|
226 # |
|
227 # You can specify commands to run before exit after a [run-commands] or [homescreen-processor] |
|
228 # |
|
229 # [drive-z] |
|
230 # \path\widget1.wgz |
|
231 # widget2.wdgt.zip [HomeScreen] |
|
232 # |
|
233 # [drive-e] |
|
234 # widget3.wgz |
|
235 # |
|
236 # [run-commands] |
|
237 # dostuff.pl |
|
238 # |
|
239 |
|
240 sub installConfig |
|
241 { |
|
242 my ( $self, $file ) = @_; |
|
243 $self->{'installedFiles'} = (); |
|
244 |
|
245 my ( %installData, $sectionName ); |
|
246 open CONFIG, $file or die "Failed to open config file $file: $!"; |
|
247 |
|
248 # Loop for gathering STATIC UIDs allocated for widgets in config INI file |
|
249 while(my $line = <CONFIG>) |
|
250 { |
|
251 # Ignore comments |
|
252 $line =~ s/\#.*$//; |
|
253 |
|
254 if ($line =~ /^\[([^\]]+)\]/) |
|
255 { |
|
256 $sectionName = lc $1; |
|
257 next; |
|
258 } |
|
259 |
|
260 # Process sections after this point |
|
261 next if !$sectionName; |
|
262 chomp $line; |
|
263 |
|
264 # Have we found a list of widgets? |
|
265 if ($sectionName =~ /^drive-([a-z])/) |
|
266 { |
|
267 my $drive = uc($1); |
|
268 #Assuming Static UID given in ini are within int/ext/wgt/wgz specified ranges |
|
269 # Add Static UID to the respective list |
|
270 if ($line =~ /^(.+?\.(?:wdgt\.zip|wgz|wgt))\s*(\[HomeScreen\])?\s*(0x[a-fA-F0-9]{8})?\s*(\[Operator\])?\s*$/i) |
|
271 { |
|
272 if ($3) |
|
273 { |
|
274 my $uid = hex($3); |
|
275 |
|
276 if ($drive =~ /^[zcZC]/) |
|
277 { |
|
278 if(($uid >= INTERNAL_UID_LOWER_WRT) && ($uid < INTERNAL_UID_LOWER_CWRT)) |
|
279 { |
|
280 push(@staticUidWrtIntList,$uid) ; |
|
281 } |
|
282 else |
|
283 { |
|
284 push(@staticUidCwrtIntList,$uid); |
|
285 } |
|
286 } |
|
287 else |
|
288 { |
|
289 if(($uid >= EXTERNAL_UID_LOWER_WRT) && ($uid < EXTERNAL_UID_LOWER_CWRT)) |
|
290 { |
|
291 push(@staticUidWrtExtList,$uid) ; |
|
292 } |
|
293 else |
|
294 { |
|
295 push(@staticUidCwrtExtList,$uid) ; |
|
296 } |
|
297 } |
|
298 } |
|
299 } |
|
300 } |
|
301 } |
|
302 close CONFIG; |
|
303 @staticUidWrtIntList = sort(@staticUidWrtIntList); |
|
304 print "SORTED WRT INTERNAL STATIC UID @staticUidWrtIntList \n"; |
|
305 @staticUidCwrtIntList = sort(@staticUidCwrtIntList); |
|
306 print "SORTED CWRT INTERNAL STATIC UID @staticUidCwrtIntList \n"; |
|
307 @staticUidWrtExtList = sort(@staticUidWrtExtList); |
|
308 print "SORTED WRT EXTERNAL STATIC UID @staticUidWrtExtList \n"; |
|
309 @staticUidCwrtExtList = sort(@staticUidCwrtExtList); |
|
310 print "SORTED CWRT EXTERNAL STATIC UID @staticUidCwrtExtList \n"; |
|
311 |
|
312 open CONFIG, $file or die "Failed to open config file $file: $!"; |
|
313 # Main loop which reads the config INI line by line and installs widgets[wgz/wgt] in respective drives |
|
314 while(my $line = <CONFIG>) |
|
315 { |
|
316 # Ignore comments |
|
317 $line =~ s/\#.*$//; |
|
318 |
|
319 if ($line =~ /^\[([^\]]+)\]/) |
|
320 { |
|
321 $sectionName = lc $1; |
|
322 |
|
323 # Remember destination if any specified |
|
324 if ($sectionName =~ /^drive-([a-z])=(.+)$/) |
|
325 { |
|
326 $self->{'destination'}->{uc($1)} = $2; |
|
327 } |
|
328 next; |
|
329 } |
|
330 |
|
331 # Process sections after this point |
|
332 next if !$sectionName; |
|
333 chomp $line; |
|
334 |
|
335 # Have we found a list of widgets? |
|
336 if ($sectionName =~ /^drive-([a-z])/) |
|
337 { |
|
338 my $drive = uc($1); |
|
339 |
|
340 # Add to the list of Widget files |
|
341 if ($line =~ /^(.+?\.(?:wdgt\.zip|wgz|wgt))\s*(\[HomeScreen\])?\s*(0x[a-fA-F0-9]{8})?\s*(\[Operator\])?\s*$/i) |
|
342 { |
|
343 my $widget = $1; |
|
344 $widget = fixFilename(catfile($self->{'args'}->{'epocroot'}, $1)) if !-e $widget; |
|
345 die "Can't find widget $widget" if !-e $widget; |
|
346 $self->{'homescreen'}{lc $widget} = 1 if $2; # Intended for the homescreen? |
|
347 $self->{'staticUID'}{lc $widget} = hex($3) if $3; |
|
348 $self->{'secure'}{lc $widget} = 0; |
|
349 $self->{'operator'}{lc $widget} = 1 if $4; |
|
350 print "Widget for drive $drive: $widget\n" if $self->{'args'}->{'verbose'}; |
|
351 push @{ $installData{"$drive:"} }, $widget; |
|
352 } |
|
353 elsif ($line =~ /^(.+?\.(?:wdgt\.zip|wgz))\s*(\[Secure\])?\s*(0x[a-fA-F0-9]{8})?\s*$/i) |
|
354 { |
|
355 my $widget = $1; |
|
356 $widget = fixFilename(catfile($self->{'args'}->{'epocroot'}, $1)) if !-e $widget; |
|
357 die "Can't find widget $widget" if !-e $widget; |
|
358 $self->{'secure'}{lc $widget} = 1 if $2; # Intended for a secure widget handling? |
|
359 $self->{'staticUID'}{lc $widget} = hex($3) if $3; |
|
360 print "Widget for drive $drive: $widget\n" if $self->{'args'}->{'verbose'}; |
|
361 push @{ $installData{"$drive:"} }, $widget; |
|
362 } |
|
363 else |
|
364 { |
|
365 my $widget = $1; |
|
366 $self->{'secure'}{lc $widget} = 0; |
|
367 } |
|
368 } |
|
369 # Retrieve the command to execute before exit |
|
370 elsif ($sectionName =~ /^(run-commands|homescreen-processor)$/i && $line) |
|
371 { |
|
372 push @{ $self->{'exitcmds'} }, $line; |
|
373 } |
|
374 } |
|
375 close CONFIG; |
|
376 |
|
377 # Now intall the widgets for each drive specified in the config file |
|
378 foreach my $drive ( keys %installData ) |
|
379 { |
|
380 $self->installFiles($drive, $installData{$drive} ); |
|
381 $self->makeIBY($drive); |
|
382 $self->makeIBY($drive, 'localised') unless $self->{'args'}->{'l10n'}; |
|
383 } |
|
384 |
|
385 # Exit commands at the end of the process |
|
386 if ($self->{'exitcmds'}) |
|
387 { |
|
388 foreach my $cmd ( @{ $self->{'exitcmds'} } ) |
|
389 { |
|
390 print "Executing: $cmd\n" if $self->{'args'}->{'verbose'}; |
|
391 warn "WARNING: error running $cmd" if system($cmd) != 0; |
|
392 } |
|
393 } |
|
394 } |
|
395 |
|
396 # *** |
|
397 # Installs files to a drive |
|
398 # |
|
399 sub installFiles |
|
400 { |
|
401 my ( $self, $drive, $fileList ) = @_; |
|
402 my $startsrc ; |
|
403 my $iconsrc ; |
|
404 |
|
405 $isSharedLibrary = 0; |
|
406 $isSharedWidget = 0; |
|
407 $isSharedIcon = 0; |
|
408 $sharedFolderName =''; |
|
409 $isSecureWidget = 0; |
|
410 |
|
411 print "Installing files for drive $drive\n" if $self->{'args'}->{'verbose'}; |
|
412 |
|
413 # Unregister any existing widgets as otherwise when the registry is rewritten their icons will appear in the emulator but they won't work |
|
414 $self->unregisterWidgets($drive); |
|
415 |
|
416 print "\n INSTALLING FILES FOR DRIVE $drive \n"; |
|
417 # Process each widget in turn |
|
418 my ( @installedProps ); |
|
419 my ( @installedCWRTProps ); |
|
420 foreach my $filename ( @$fileList ) |
|
421 { |
|
422 # Check the file exists |
|
423 die "Can't find $filename" if !-e $filename; |
|
424 |
|
425 # Create a temporary folder |
|
426 print "\nInstalling $filename\n"; |
|
427 my $tempdir = tempdir ( DIR => '.', CLEANUP => !$self->{args}->{'debug'}); |
|
428 |
|
429 # Prefer to use 7zip rather than unzip because it's better behaved wrt Unicode |
|
430 if (findCmd('7z.exe')) |
|
431 { |
|
432 $self->un7zipWidget($filename, $tempdir); |
|
433 } |
|
434 else |
|
435 { |
|
436 $self->unzipWidget($filename, $tempdir); |
|
437 } |
|
438 my $widgetdata ; |
|
439 |
|
440 $isSharedLibrary = 0; |
|
441 $isSharedWidget = 0; |
|
442 $isSharedIcon = 0; |
|
443 $sharedFolderName =''; |
|
444 |
|
445 my ( $root, $extracted, $size ) = $self->getFiles($tempdir); |
|
446 die "No files extracted from $filename" if !@$extracted; |
|
447 |
|
448 if ($self->{'operator'}{lc $filename}) |
|
449 { |
|
450 print "\n NON NOKIA WGT \n"; |
|
451 $non_nokia_widget = 1; |
|
452 } |
|
453 else |
|
454 { |
|
455 print "\n NOKIA WGT \n"; |
|
456 $non_nokia_widget = 0; |
|
457 } |
|
458 |
|
459 |
|
460 if (($filename =~ /.wgz/ )&& (my $plist = catfile($tempdir, $root, 'Info.plist'))) |
|
461 { |
|
462 print ("S60 widget \n"); |
|
463 |
|
464 $w3c_widget = 0; |
|
465 # Parse the XML file into a hash |
|
466 $widgetdata = parsePList($plist); |
|
467 } |
|
468 elsif (($filename =~ /.wgt/ ) && ( my $confxml = catfile($tempdir, $root, 'config.xml'))) |
|
469 { |
|
470 print (" W3C widget \n"); |
|
471 |
|
472 # Parse the XML file into a hash |
|
473 $widgetdata = parseConfXml($confxml,$self->{'args'}->{'l10n'}); |
|
474 $w3c_widget = 1; |
|
475 } |
|
476 else |
|
477 { |
|
478 die "Can't find $root/Info.plist , $root/config.xml file"; |
|
479 } |
|
480 |
|
481 if ($self->{'secure'}{lc $filename}){ |
|
482 $isSecureWidget = 1; |
|
483 } else { |
|
484 $isSecureWidget = 0; |
|
485 } |
|
486 |
|
487 if( $w3c_widget ) |
|
488 { |
|
489 if( $widgetdata->{'MainHTML'} ) |
|
490 { |
|
491 $startsrc = $widgetdata->{'MainHTML'}; |
|
492 } |
|
493 else |
|
494 { |
|
495 $startsrc = 'index.html'; |
|
496 } |
|
497 if( $widgetdata->{'IconPath'} ) |
|
498 { |
|
499 $iconsrc = $widgetdata->{'IconPath'}; |
|
500 } |
|
501 else |
|
502 { |
|
503 $iconsrc = 'icon.png'; |
|
504 } |
|
505 |
|
506 if($non_nokia_widget) |
|
507 { |
|
508 $widgetdata->{'ProcessUid'} = CWRT_WIDGET_UI_NON_NOKIA_UID; |
|
509 } |
|
510 else |
|
511 { |
|
512 $widgetdata->{'ProcessUid'} = CWRT_WIDGET_UI_UID; |
|
513 } |
|
514 |
|
515 $widgetdata->{'MimeType'} = "application/widget"; |
|
516 } |
|
517 |
|
518 print "Identifier: $widgetdata->{'BundleIdentifier'}\n" if $self->{args}->{'verbose'}; |
|
519 |
|
520 # Set widget package properties |
|
521 $widgetdata->{'FileName'} = $filename; |
|
522 $widgetdata->{'FileSize'} = $size; |
|
523 |
|
524 # Load the language translations for BundleDisplayName |
|
525 if ($self->{'args'}->{'l10n'}) |
|
526 { |
|
527 my $localised = $self->getLocalisedStringByLangCode(catfile($tempdir, $root), 'DisplayName', $self->{'args'}->{'l10n'}); |
|
528 $widgetdata->{'BundleDisplayName'} = $localised if $localised; |
|
529 } |
|
530 else |
|
531 { |
|
532 $widgetdata->{'LocBundleDisplayName'} = $self->getLocalisedStrings(catfile($tempdir, $root), 'DisplayName'); |
|
533 } |
|
534 |
|
535 # Fix up some of the fields |
|
536 $widgetdata->{'BundleName'} = $widgetdata->{'BundleDisplayName'} if !$widgetdata->{'BundleName'}; |
|
537 $widgetdata->{'AllowNetworkAccess'} = $widgetdata->{'AllowFullAccess'} if !$widgetdata->{'AllowNetworkAccess'}; |
|
538 $widgetdata->{'DriveName'} = $drive; |
|
539 |
|
540 if( $w3c_widget ) |
|
541 { |
|
542 $widgetdata->{'PropertyListVersion'} = 4; |
|
543 my $widget_uid; |
|
544 |
|
545 if($non_nokia_widget) |
|
546 { |
|
547 $widget_uid = CWRT_WIDGET_UI_NON_NOKIA_UID; |
|
548 } |
|
549 else |
|
550 { |
|
551 $widget_uid = CWRT_WIDGET_UI_UID; |
|
552 } |
|
553 |
|
554 if($isSharedLibrary) |
|
555 { |
|
556 #In case of NOKIA:sharedlibrary, the extracted widget contents and data source are stored under private\<uid>\..\lib\<sharedfoldername>\ |
|
557 $widgetdata->{'BasePath'} = sprintf('%s\\private\\%08X\\%s\\%s\\%s\\', $widgetdata->{'DriveName'}, $widget_uid, "widgets_21D_4C7","lib",$sharedFolderName); |
|
558 $widgetdata->{'IconPath'} = sprintf('%s\\private\\%08X\\%s\\%s\\%s\\',$widgetdata->{'DriveName'}, $widget_uid, "data" ,"lib" ,$sharedFolderName); |
|
559 } |
|
560 else |
|
561 { |
|
562 $widgetdata->{'BasePath'} = sprintf('%s\\private\\%08X\\%s\\%s\\', $widgetdata->{'DriveName'}, $widget_uid, "widgets_21D_4C7", hashpjw($widgetdata->{'BundleIdentifier'})); |
|
563 $widgetdata->{'IconPath'} = sprintf('%s\\private\\%08X\\%s\\%s\\',$widgetdata->{'DriveName'}, $widget_uid, "data" , hashpjw($widgetdata->{'BundleIdentifier'})); |
|
564 } |
|
565 $widgetdata->{'MainHTML'} = "$widgetdata->{'BasePath'}$startsrc"; |
|
566 $widgetdata->{'AllowNetworkAccess'} = 1; |
|
567 |
|
568 } |
|
569 else |
|
570 { |
|
571 $widgetdata->{'PropertyListVersion'} = 1; |
|
572 $widgetdata->{'BasePath'} = sprintf('%s\\private\\%08x\\%s\\', $widgetdata->{'DriveName'}, WIDGET_UI_UID, $widgetdata->{'BundleIdentifier'}); |
|
573 $widgetdata->{'MainHTML'} = "$widgetdata->{'BasePath'}$root\\$widgetdata->{'MainHTML'}"; |
|
574 $widgetdata->{'IconPath'} = "$widgetdata->{'BasePath'}$root\\"; |
|
575 } |
|
576 |
|
577 $widgetdata->{'BlanketPermissionGranted'} = 0 if( ! $widgetdata->{'BlanketPermissionGranted'} ); |
|
578 $widgetdata->{'MiniViewEnabled'} = 0 if( ! $widgetdata->{'MiniViewEnabled'} ); |
|
579 |
|
580 # Indicate that the widget was pre-installed |
|
581 $widgetdata->{'PreInstalled'} = 1; |
|
582 |
|
583 # Set BlanketPermissionGranted flag if Widget is listed as a homescreen widget in the INI file |
|
584 # Error if MiniViewEnabled isn't set |
|
585 if ($self->{'homescreen'}{lc $filename}) |
|
586 { |
|
587 $widgetdata->{'BlanketPermissionGranted'} = 1; |
|
588 die "ERROR: $filename - MiniViewEnabled not set for homescreen widget" if !$widgetdata->{'MiniViewEnabled'}; |
|
589 } |
|
590 |
|
591 # Defining UID |
|
592 if($self->{'staticUID'}{lc $filename}) |
|
593 { |
|
594 print "Using Static UID from INI file\n"; |
|
595 $widgetdata->{'Uid'} = $self->{'staticUID'}{lc $filename}; |
|
596 } |
|
597 else |
|
598 { |
|
599 # Find the next free UID to use |
|
600 $widgetdata->{'Uid'} = $self->findfreeUid($widgetdata); |
|
601 } |
|
602 print sprintf("Using UID for midlet: 0x%08X\n", $widgetdata->{'Uid'}) if $self->{args}->{'verbose'}; |
|
603 |
|
604 # Make sure the destination exists |
|
605 my $dest = $self->regFileName($drive); |
|
606 mkpath $dest; |
|
607 |
|
608 my $icon; |
|
609 if($w3c_widget) |
|
610 { |
|
611 my $locate = $self->{'args'}->{'l10n'}; |
|
612 my $seperator ='-'; |
|
613 my $position = -1; |
|
614 my $found = 0 ; |
|
615 if ($self->{'args'}->{'l10n'}) |
|
616 { |
|
617 do |
|
618 { |
|
619 my $searchdir = '.\\'.$tempdir.'\\locales\\'.$locate; |
|
620 if( opendir(DIR, $searchdir)) |
|
621 { |
|
622 my @files = grep(/$iconsrc$/,readdir(DIR)); |
|
623 closedir(DIR); |
|
624 if(@files) |
|
625 { |
|
626 $iconsrc = 'locales\\'.$locate.'\\'.$iconsrc ; |
|
627 $found = 1; |
|
628 } |
|
629 } |
|
630 if(($position = rindex($locate,$seperator)) > 0 ) |
|
631 { |
|
632 $locate = substr($locate,0,$position); |
|
633 } |
|
634 } while (!$found && $position > 0); |
|
635 } |
|
636 if ($iconsrc) |
|
637 { |
|
638 $icon = catfile($tempdir, $root, $iconsrc); |
|
639 } |
|
640 |
|
641 if(-e $icon) |
|
642 { |
|
643 $widgetdata->{'DBIconPath'}= "$widgetdata->{'BasePath'}$iconsrc"; |
|
644 print "Icon found -- $widgetdata->{'DBIconPath'} \n"; |
|
645 } |
|
646 else |
|
647 { |
|
648 $icon = findCmd('default_widget_icon.png'); |
|
649 print "default_widget_icon.png used \n"; |
|
650 $widgetdata->{'DBIconPath'}= ":/resource/default_widget_icon.png"; |
|
651 } |
|
652 } |
|
653 else |
|
654 { |
|
655 $icon = catfile($tempdir, $root, 'Icon.png'); |
|
656 die "ERROR: Widget bundle must include an Icon.png file in $root directory.\n" unless -e $icon; |
|
657 } |
|
658 |
|
659 # Create the MBM file icon - in case of sharedlibrary, create only if NOKIA:widget is true |
|
660 my $mbm; |
|
661 |
|
662 print("\nIs shared library - $isSharedLibrary \nIs sharedlib widget - $isSharedWidget \nIs Secure WGZ - $self->{'secure'}{lc $filename}\n"); |
|
663 |
|
664 if( ((!$isSharedLibrary) || ($isSharedLibrary && $isSharedWidget)) && (!$self->{'secure'}{lc $filename})) |
|
665 { |
|
666 print "Creating mbm !!!\n"; |
|
667 die "ERROR: Can't find PNG2MBM command in PATH." if !(my $cmd = findCmd('png2mbm.exe')); |
|
668 $mbm = $self->regFileName($drive, sprintf("[%08x].mbm", $widgetdata->{'Uid'})); |
|
669 print "Generating: $mbm\n"; |
|
670 die "Failed to create MBM $mbm " if system("$cmd $self->{args}->{'verbose'} -in \"$icon\" -out $mbm -sizes ".ICON_SIZES) != 0; |
|
671 |
|
672 # Add the mbm to the list of files |
|
673 $self->addToRomList($drive, $mbm); |
|
674 |
|
675 # Create the INI file which defines the registry info |
|
676 my $ini = catfile($tempdir, 'reg.ini'); |
|
677 $self->makeIni($widgetdata, $ini); |
|
678 unlink 'debug.ini'; copy($ini, 'debug.ini') if $debug; |
|
679 print "Generated INI file: $ini\n" if $self->{args}->{'verbose'}; |
|
680 |
|
681 # Generate the registry files |
|
682 die "ERROR: Can't find WidgetRegFiles.exe command in PATH." if !($cmd = findCmd('WidgetRegFiles.exe')); |
|
683 die "Failed to generate registry files" if system("$cmd $ini") != 0; |
|
684 my ( $reg, $loc ) = ( catfile($dest, sprintf("%08x_reg.rsc", $widgetdata->{'Uid'})), catfile($dest, sprintf("%08x_loc.rsc", $widgetdata->{'Uid'})) ); |
|
685 die "Failed to generate REG file: $!" if !-e $reg; |
|
686 $self->addToRomList($drive, $reg); |
|
687 die "Failed to generate LOC file: $!" if !-e $loc; |
|
688 if ($self->{'args'}->{'l10n'}) |
|
689 { |
|
690 $self->addToRomList($drive, $loc); |
|
691 } |
|
692 else |
|
693 { |
|
694 $self->addToRomList($drive, $loc, 'localised'); |
|
695 } |
|
696 } |
|
697 |
|
698 # Create install folder |
|
699 my $dir = $self->installDir($drive, $widgetdata->{'BundleIdentifier'}); |
|
700 mkpath $dir; |
|
701 |
|
702 # Now copy the widget files to the right place |
|
703 print "Install Directory: $dir\n"; |
|
704 foreach my $widgetfile ( @$extracted ) |
|
705 { |
|
706 my ( $sourceFile, $destFile ) = ( catfile($tempdir, $widgetfile), catfile($dir, $widgetfile) ); |
|
707 print "Copying $sourceFile to $destFile\n" if $self->{args}->{'debug'}; |
|
708 |
|
709 # Create the directory if it doesn't exist already |
|
710 my $dir = dirname($destFile); |
|
711 if (!-d $dir) |
|
712 { |
|
713 mkpath ($dir) or die "Failed to create $dir $!"; |
|
714 } |
|
715 unlink $destFile; |
|
716 if (!copy($sourceFile, $destFile)) |
|
717 { |
|
718 warn "WARNING: Failed to copy $sourceFile to $destFile: $!"; |
|
719 } |
|
720 #else |
|
721 { |
|
722 $self->addToRomList($drive, $destFile); |
|
723 } |
|
724 } |
|
725 |
|
726 # Copy the MBM file into the widget install directory |
|
727 # because native installation does and uses it after |
|
728 # installation to manage UID consistency. |
|
729 if( (!$isSharedLibrary) || ($isSharedLibrary && $isSharedWidget)) |
|
730 { |
|
731 my $mbmName = sprintf("[%08x].mbm", $widgetdata->{'Uid'}); |
|
732 my $destFile; |
|
733 if($w3c_widget) |
|
734 { |
|
735 my $dataDir; |
|
736 |
|
737 my $widget_uid; |
|
738 |
|
739 if($non_nokia_widget) |
|
740 { |
|
741 $widget_uid = CWRT_WIDGET_UI_NON_NOKIA_UID; |
|
742 } |
|
743 else |
|
744 { |
|
745 $widget_uid = CWRT_WIDGET_UI_UID; |
|
746 } |
|
747 |
|
748 # NOKIA:sharedlibrary check |
|
749 if($isSharedWidget) |
|
750 { |
|
751 $dataDir = fixFilename(catfile($self->destLocation($drive), 'private', sprintf("%08X", $widget_uid),"data","lib",$sharedFolderName)); |
|
752 } |
|
753 else |
|
754 { |
|
755 #in case of WGT copy .mbm to \epoc32\winscw\C\private\200267c0\data\hash<BundleId>\ |
|
756 $dataDir = fixFilename(catfile($self->destLocation($drive), 'private', sprintf("%08X", $widget_uid),"data", $hashval)); |
|
757 } |
|
758 mkpath $dataDir; |
|
759 $destFile = catfile($dataDir,$mbmName); |
|
760 } |
|
761 else |
|
762 { |
|
763 $destFile = catfile($dir, $root, $mbmName); |
|
764 } |
|
765 # Exclude securewidget from making a mbm file |
|
766 if (!$self->{'secure'}{lc $filename}){ |
|
767 print "Copying $mbm to $destFile\n" if $self->{args}->{'debug'}; |
|
768 copy $mbm, $destFile or die "Failed to copy $mbm to $destFile: $!"; |
|
769 $self->addToRomList($drive, $destFile); |
|
770 } |
|
771 } |
|
772 |
|
773 # Remember the data for the registry |
|
774 if( $w3c_widget ) |
|
775 { |
|
776 push @installedCWRTProps, $widgetdata; |
|
777 } |
|
778 else |
|
779 { |
|
780 if (!$self->{'secure'}{lc $filename}){ |
|
781 push @installedProps, $widgetdata; |
|
782 } |
|
783 } |
|
784 |
|
785 if (!$debug) |
|
786 { |
|
787 # Perl v5.6 can't delete files with unicode in their name on Windows |
|
788 if ($] < 5.008 && $^O =~ /MSWin32/) |
|
789 { |
|
790 system("rmdir /s /q $tempdir"); |
|
791 } |
|
792 else |
|
793 { |
|
794 rmtree $tempdir; |
|
795 } |
|
796 } |
|
797 #Generate the secsession for .wgt widgets |
|
798 if( $w3c_widget ) |
|
799 { |
|
800 $self->addToRomList($drive, $self->makeSecSession($drive)); |
|
801 } |
|
802 } |
|
803 |
|
804 my $cwrtdbPath = catfile($self->destLocation($drive), CWRT_WEBAPP_REGISTRY); |
|
805 $self->addToRomList($drive, $cwrtdbPath); |
|
806 |
|
807 # Generate the registry and IBY file |
|
808 $self->addToRomList($drive, $self->makeRegistry($drive, \@installedProps, 0)); |
|
809 $self->addToRomList($drive, $self->makeRegistry($drive, \@installedCWRTProps, 1)); |
|
810 |
|
811 |
|
812 print "\n\n %%%%% WIDGET PRE-INSTALLATION completed for Drive $drive %%%%% \n\n"; |
|
813 } |
|
814 |
|
815 sub un7zipWidget |
|
816 { |
|
817 my ( $self, $filename, $tempdir ) = @_; |
|
818 |
|
819 # Unzip the file |
|
820 die "Can't find 7z.exe tool on PATH." if !findCmd('7z.exe'); |
|
821 die "Failed to extract widget contents using 7zip: $!" if (system("7z x -aoa -o$tempdir \"$filename\" >nul 2>&1") != 0); |
|
822 } |
|
823 |
|
824 sub unzipWidget |
|
825 { |
|
826 my ( $self, $filename, $tempdir ) = @_; |
|
827 |
|
828 # Unzip the file |
|
829 die "Can't find unzip.exe tool on PATH." if !findCmd('unzip.exe'); |
|
830 die "Failed to extract widget contents using zip: $!" if (system("unzip.exe \"$filename\" -d $tempdir >nul 2>&1") != 0); |
|
831 } |
|
832 |
|
833 # Recursively get list of files in a folder |
|
834 sub getFiles |
|
835 { |
|
836 my ( $self, $tempdir ) = @_; |
|
837 |
|
838 my $root = '.'; |
|
839 my @extracted; |
|
840 my $size =0; |
|
841 |
|
842 my @dirs = '.'; |
|
843 foreach my $dir ( @dirs ) |
|
844 { |
|
845 opendir DIR, catfile($tempdir, $dir) or die "Failed to opendir $dir: $!"; |
|
846 foreach ( grep !/^\.{1,2}$/, readdir DIR ) |
|
847 { |
|
848 my $name = catfile($dir, $_); |
|
849 my $fullname = catfile($tempdir, $name); |
|
850 if (-d $fullname) |
|
851 { |
|
852 push @dirs, $name; |
|
853 } |
|
854 else |
|
855 { |
|
856 # Remember total size for later |
|
857 if (-e $fullname) |
|
858 { |
|
859 push @extracted, $name; |
|
860 print "Extracted: $name\n" if $self->{args}->{'debug'}; |
|
861 $size += -s $fullname; |
|
862 |
|
863 # Get root |
|
864 if ($name =~ /info.plist$/i && $name =~ /^([^\/\\]+)[\/\\]/) |
|
865 { |
|
866 $root = $1; |
|
867 } |
|
868 } |
|
869 else |
|
870 { |
|
871 warn "WARNING: Failed to find extracted file $fullname"; |
|
872 } |
|
873 } |
|
874 } |
|
875 closedir DIR; |
|
876 } |
|
877 |
|
878 return ( $root, \@extracted, $size ); |
|
879 } |
|
880 |
|
881 ## see http://www.loc.gov/standards/iso639-2/php/code_list.php |
|
882 ## see http://www.loc.gov/standards/iso639-2/faq.html#2 |
|
883 sub getLocalisedStringByLangCode |
|
884 { |
|
885 my ( $self, $dir, $strName, $langCode ) = @_; |
|
886 my $localised; |
|
887 |
|
888 # Generate the name of the file containing localised strings for this language |
|
889 my $locfile = catfile($dir, "$langCode.lproj", "InfoPlist.strings"); |
|
890 if (-e $locfile) |
|
891 { |
|
892 # Open the file |
|
893 print "Found $langCode language translations in $locfile\n" if $verbose; |
|
894 open LOC, $locfile or die "Failed to open $locfile: $!"; |
|
895 |
|
896 # Get the byte order mark from the start of the file |
|
897 my $bom; |
|
898 $bom = unpack("S", $bom) if (read(LOC, $bom, 2) == 2); |
|
899 if ($bom) |
|
900 { |
|
901 if ($bom == 0xEFBB) |
|
902 { |
|
903 # Skip utf8 bom |
|
904 seek LOC, 3, 0; |
|
905 } |
|
906 elsif ($bom != 0xFFFE && $bom != 0xFEFF) |
|
907 { |
|
908 seek LOC, 0, 0; |
|
909 undef $bom; |
|
910 } |
|
911 } |
|
912 else |
|
913 { |
|
914 # go back to start of file if no bom |
|
915 seek LOC, 0, 0; |
|
916 undef $bom; |
|
917 } |
|
918 |
|
919 while(my $line = <LOC>) |
|
920 { |
|
921 # Do unicode conversion |
|
922 my $ustr; |
|
923 if ($bom) |
|
924 { |
|
925 $ustr = utf16($line); |
|
926 $ustr->byteswap if $bom != 0xFFFE; |
|
927 } |
|
928 else |
|
929 { |
|
930 $ustr = utf8($line); |
|
931 } |
|
932 |
|
933 # Find the string we're looking for |
|
934 if ($ustr->utf8 =~ /(?:^|\s)$strName\s*=\s*\"([^\"]*)\"/) |
|
935 { |
|
936 print "\t...$strName => $1\n" if $debug; |
|
937 $localised = utf8($1); |
|
938 } |
|
939 } |
|
940 close LOC; |
|
941 } |
|
942 return $localised; |
|
943 } |
|
944 |
|
945 # Get localised version of strings if they exist |
|
946 sub getLocalisedStrings |
|
947 { |
|
948 my ( $self, $dir, $strName ) = @_; |
|
949 return if (!$self->{'langmapping'}); |
|
950 |
|
951 # Iterate through all the languages we know about |
|
952 my %result; |
|
953 for(my $i = 0; $i < scalar(@{ $self->{'langmapping'}->{LangID} }); $i++) |
|
954 { |
|
955 my ( $langid, $langname ) = ( $self->{'langmapping'}->{LangID}->[$i], $self->{'langmapping'}->{LangDir}->[$i] ); |
|
956 |
|
957 my $localised = $self->getLocalisedStringByLangCode($dir, $strName, $langname); |
|
958 $result{$langid} = $localised if $localised; |
|
959 } |
|
960 |
|
961 return \%result if keys %result; |
|
962 } |
|
963 |
|
964 # Find and execute a command |
|
965 sub findCmd |
|
966 { |
|
967 my $cmd = shift; |
|
968 return fixFilename("./$cmd") if -e $cmd; |
|
969 |
|
970 # PATH separator differs on Windows and Linux |
|
971 my $sep = $^O =~ /MSWin32/ ? ';' : ':'; |
|
972 |
|
973 # Search each entry in the PATH |
|
974 my @paths = split /$sep/, $ENV{PATH}; |
|
975 push @paths, dirname($0); |
|
976 foreach my $path ( @paths ) |
|
977 { |
|
978 my $fullcmd = fixFilename(catfile($path, $cmd)); |
|
979 return $fullcmd if -e $fullcmd; |
|
980 } |
|
981 } |
|
982 |
|
983 # Make INI file describing widget - this is passed to widgetregfiles.exe |
|
984 sub makeIni |
|
985 { |
|
986 my ( $self, $data, $file ) = @_; |
|
987 open INI, ">$file" or die "Failed to open $file for writing: $!"; |
|
988 binmode INI, ":utf8" if $] >= 5.008; |
|
989 |
|
990 # Get directory where mbm should go |
|
991 my $dir = $self->regFileName($data->{'DriveName'}); |
|
992 |
|
993 print INI "[app_registration_info]\n"; |
|
994 print INI sprintf("uid=%08x\n", $data->{'Uid'}); |
|
995 print INI "app_file=$data->{'MainHTML'}\n"; |
|
996 print INI "caption=$data->{'BundleDisplayName'}\n"; |
|
997 print INI "drive_name=$data->{'DriveName'}\n"; |
|
998 print INI "results_dir=$dir\n"; |
|
999 |
|
1000 if( $w3c_widget ) |
|
1001 { |
|
1002 print INI "app_type=200267DC\n"; |
|
1003 print " WGT Widget:: app_type added to ini of widgetregfiles.exe \n" if $debug; |
|
1004 } |
|
1005 |
|
1006 |
|
1007 # Add language stuff if we have the mapping |
|
1008 if ($data->{'LocBundleDisplayName'}) |
|
1009 { |
|
1010 my @langList; |
|
1011 foreach my $langid ( sort { $a <=> $b } keys %{ $data->{'LocBundleDisplayName'} } ) |
|
1012 { |
|
1013 my $symid = sprintf("%02d", $langid); |
|
1014 push @langList, $symid; |
|
1015 print INI "caption$symid=", $data->{'LocBundleDisplayName'}->{$langid}->utf8, "\n"; |
|
1016 } |
|
1017 print INI "languages=", join(' ', @langList), "\n"; |
|
1018 } |
|
1019 |
|
1020 close INI; |
|
1021 convert2Unicode($file); |
|
1022 } |
|
1023 |
|
1024 sub findfreeUid |
|
1025 { |
|
1026 my ( $self, $data ) = @_; |
|
1027 my $appfile = lc $data->{'MainHTML'}; |
|
1028 my $uid; |
|
1029 my $size; |
|
1030 if($w3c_widget) #CWRT Widget |
|
1031 { |
|
1032 # pick the next free CWRT Internal UID |
|
1033 if(isInternal($data->{'DriveName'})) |
|
1034 { |
|
1035 print "\nInternal CWRT UID"; |
|
1036 $size = scalar @staticUidCwrtIntList; |
|
1037 print " Static UID :: @staticUidCwrtIntList Length -- $size \n" if($size); |
|
1038 for(my $i=0; $i<$size; $i++) |
|
1039 { |
|
1040 #skip looping if freeUid found |
|
1041 if( $self->{'freeuidcwrtint'} lt $staticUidCwrtIntList[$i] ) {last}; |
|
1042 |
|
1043 if($self->{'freeuidcwrtint'} == $staticUidCwrtIntList[$i]) |
|
1044 { |
|
1045 $self->{'freeuidcwrtint'}++; |
|
1046 } |
|
1047 } |
|
1048 $uid = $self->{'freeuidcwrtint'}++; # Assign and then set next free UID |
|
1049 } |
|
1050 else |
|
1051 { |
|
1052 # pick the next free CWRT External UID |
|
1053 print "\nExternal CWRT UID "; |
|
1054 $size = scalar @staticUidCwrtExtList; |
|
1055 print "Static UID :: @staticUidCwrtExtList Length -- $size \n"; |
|
1056 |
|
1057 for(my $i=0; $i<$size; $i++) |
|
1058 { |
|
1059 if( $self->{'freeuidcwrtext'} lt $staticUidCwrtExtList[$i] ) {last}; |
|
1060 |
|
1061 if($self->{'freeuidcwrtext'} == $staticUidCwrtExtList[$i]) |
|
1062 { |
|
1063 $self->{'freeuidcwrtext'}++; |
|
1064 } |
|
1065 } |
|
1066 $uid = $self->{'freeuidcwrtext'}++; # Assign and then set next free UID |
|
1067 } |
|
1068 } |
|
1069 else #WRT widget |
|
1070 { |
|
1071 # pick the next free WRT Internal UID |
|
1072 if(isInternal($data->{'DriveName'})) |
|
1073 { |
|
1074 # pick the next free CWRT Internal UID |
|
1075 if(isInternal($data->{'DriveName'})) |
|
1076 { |
|
1077 print "\nInternal WRT UID "; |
|
1078 $size = scalar @staticUidWrtIntList; |
|
1079 print " STATIC UID :: @staticUidWrtIntList Length -- $size\n"; |
|
1080 |
|
1081 for(my $i=0; $i<$size; $i++) |
|
1082 { |
|
1083 if( $self->{'freeuidwrtint'} lt $staticUidWrtIntList[$i] ) {last}; |
|
1084 |
|
1085 if($self->{'freeuidwrtint'} == $staticUidWrtIntList[$i]) |
|
1086 { |
|
1087 $self->{'freeuidwrtint'}++; |
|
1088 } |
|
1089 } |
|
1090 $uid = $self->{'freeuidwrtint'}++; # Assign and then set next free UID |
|
1091 } |
|
1092 } |
|
1093 else |
|
1094 { |
|
1095 # pick the next free WRT External UID |
|
1096 print "\nExternal WRT UID "; |
|
1097 $size = scalar @staticUidWrtExtList; |
|
1098 print "STATIC UID :: @staticUidWrtExtList Length -- $size\n"; |
|
1099 |
|
1100 for(my $i=0; $i<$size; $i++) |
|
1101 { |
|
1102 if( $self->{'freeuidwrtext'} lt $staticUidWrtExtList[$i] ) {last}; |
|
1103 |
|
1104 if($self->{'freeuidwrtext'} == $staticUidWrtExtList[$i]) |
|
1105 { |
|
1106 $self->{'freeuidwrtext'}++; |
|
1107 } |
|
1108 } |
|
1109 $uid = $self->{'freeuidwrtext'}++; # Assign and then set next free UID |
|
1110 } |
|
1111 } |
|
1112 print sprintf("Generated UID: hex->0x%08X dec->$uid\n \n",$uid); |
|
1113 return $uid; |
|
1114 } |
|
1115 |
|
1116 # Fix slash problems in a filename |
|
1117 sub fixFilename |
|
1118 { |
|
1119 my $filename = shift; |
|
1120 $filename =~ s/([\\\/])[\\\/]/$1/g; |
|
1121 return catfile(split(/[\\\/]/, $filename)); |
|
1122 } |
|
1123 |
|
1124 # Get install location |
|
1125 sub destLocation |
|
1126 { |
|
1127 my ( $self, $drive ) = @_; |
|
1128 my $letter = uc(substr($drive, 0, 1)); |
|
1129 |
|
1130 # Was any destination location specified in the config file? |
|
1131 if ($self->{destination} && $self->{destination}->{$letter}) |
|
1132 { |
|
1133 return catfile($self->{'args'}->{'epocroot'}, $self->{destination}->{$letter}); |
|
1134 } |
|
1135 |
|
1136 # No destination specified - use emulator location |
|
1137 return catfile($self->{'args'}->{'epocroot'}, $letter eq 'Z' ? ROM_DEST : sprintf(DRIVE_DEST, $letter)); |
|
1138 } |
|
1139 |
|
1140 # Get the destination for REG/LOC/MBM file |
|
1141 sub regFileName |
|
1142 { |
|
1143 my ( $self, $drive, $filename ) = @_; |
|
1144 my $result = catfile($self->destLocation($drive), DESTINATION, $filename); |
|
1145 return fixFilename($result); |
|
1146 } |
|
1147 |
|
1148 # Widget install directory |
|
1149 sub installDir |
|
1150 { |
|
1151 my ( $self, $drive, $id ) = @_; |
|
1152 my $result ; |
|
1153 $hashval = hashpjw($id); |
|
1154 if( $w3c_widget ) |
|
1155 { |
|
1156 my $widget_uid; |
|
1157 if($non_nokia_widget) |
|
1158 { |
|
1159 $widget_uid = CWRT_WIDGET_UI_NON_NOKIA_UID; |
|
1160 } |
|
1161 else |
|
1162 { |
|
1163 $widget_uid = CWRT_WIDGET_UI_UID; |
|
1164 } |
|
1165 if($isSharedLibrary) |
|
1166 { |
|
1167 $result = catfile($self->destLocation($drive), 'private', sprintf("%08X", $widget_uid),"widgets_21D_4C7", "lib",$sharedFolderName); |
|
1168 } |
|
1169 else |
|
1170 { |
|
1171 $result = catfile($self->destLocation($drive), 'private', sprintf("%08X", $widget_uid),"widgets_21D_4C7", $hashval); |
|
1172 } |
|
1173 } |
|
1174 else |
|
1175 { |
|
1176 if ($isSecureWidget){ |
|
1177 print "Installing secure WGZ Widget\n"; |
|
1178 $result = catfile($self->destLocation($drive), 'private', sprintf("%08x", SWIDGET_UI_UID), $id); |
|
1179 } else { |
|
1180 print "Installing WGZ Widget\n"; |
|
1181 $result = catfile($self->destLocation($drive), 'private', sprintf("%08x", WIDGET_UI_UID), $id); |
|
1182 } |
|
1183 } |
|
1184 return fixFilename($result); |
|
1185 } |
|
1186 |
|
1187 # Determines whether a drive should be considered internal or not |
|
1188 sub isInternal |
|
1189 { |
|
1190 my $drive = shift; |
|
1191 die "Invalid drive format: $drive" if $drive !~ /^[a-zA-Z]:/; |
|
1192 return 1 if $drive =~ /^[zcZC]/; |
|
1193 } |
|
1194 |
|
1195 # Parse these awful PLIST files |
|
1196 sub parsePList |
|
1197 { |
|
1198 my $file = shift; |
|
1199 |
|
1200 # Create parser object |
|
1201 our ($key, $val, $plisthash ) = ('', '', {}); |
|
1202 my $parser = new XML::Parser; |
|
1203 $parser->setHandlers('Doctype' => \&docT, 'Start' => \&startH, 'End' => \&endH, 'Char' => \&dataH); |
|
1204 |
|
1205 # Parse the file |
|
1206 open XML, $file or die "Couldn't open $file"; |
|
1207 |
|
1208 # Skip the UTF8 BOM - perl 5.6 can't handle it |
|
1209 my $bom; |
|
1210 read XML, $bom, 3; |
|
1211 $bom = join('', map(sprintf('%X', $_), unpack("CCC", $bom))); |
|
1212 print "Testing the following for BOM: $bom\n" if $debug; |
|
1213 seek(XML, 0, 0) if $bom ne 'EFBBBF'; |
|
1214 |
|
1215 $parser->parse(*XML); |
|
1216 close XML; |
|
1217 |
|
1218 # Check required fields exist |
|
1219 die "Widget MainHTML unknown" if !$plisthash->{'MainHTML'}; |
|
1220 die "Widget BundleIdentifier unknown" if !$plisthash->{'BundleIdentifier'}; |
|
1221 die "Widget BundleDisplayName unknown" if !$plisthash->{'BundleDisplayName'}; |
|
1222 |
|
1223 # Return result |
|
1224 return $plisthash; |
|
1225 |
|
1226 # Called on a start tag |
|
1227 sub startH |
|
1228 { |
|
1229 my ($p, $el, %atts) = @_; |
|
1230 undef $key if ($el =~ /^key$/i); |
|
1231 $val = ''; |
|
1232 } |
|
1233 |
|
1234 # Receives document type |
|
1235 sub docT |
|
1236 { |
|
1237 my ($expat, $name, $sysid, $pubid, $internal ) = @_; |
|
1238 die "PLIST format looks wrong!" if lc($name) ne 'plist'; |
|
1239 $plisthash->{'NokiaWidget'} = ( $pubid =~ m[^-//Nokia//DTD PLIST]i ) ? 1 : 0; |
|
1240 } |
|
1241 |
|
1242 # Receives character data |
|
1243 sub dataH |
|
1244 { |
|
1245 my ($p, $s) = @_; |
|
1246 $val .= $s; |
|
1247 } |
|
1248 |
|
1249 # Called on an end tag |
|
1250 sub endH |
|
1251 { |
|
1252 my ($p, $el) = @_; |
|
1253 if ($el =~ /^key$/i) |
|
1254 { |
|
1255 $key = $val; |
|
1256 } |
|
1257 elsif ($key) |
|
1258 { |
|
1259 $val = 1 if $el =~ /^true$/i; |
|
1260 $val = 0 if $el =~ /^false$/i; |
|
1261 |
|
1262 # Fix stuff so it's in the correct format |
|
1263 $key =~ s/^CF//; |
|
1264 $key = 'BundleIdentifier' if $key =~ /^Identifier$/i; |
|
1265 $key = 'BundleDisplayName' if $key =~ /^DisplayName$/i; |
|
1266 $key = 'BundleVersion' if $key =~ /^Version$/i; |
|
1267 |
|
1268 $plisthash->{$key} = $val; |
|
1269 undef $key; |
|
1270 } |
|
1271 $val = '' |
|
1272 } |
|
1273 } |
|
1274 |
|
1275 |
|
1276 sub parseConfXml |
|
1277 { |
|
1278 my ( $file , $language ) = @_; |
|
1279 |
|
1280 our ($key, $val, $plisthash, $localizationValue,$localizationIconValue, $lang,$langIcon) = ('', '', {}, '','', {},{}); |
|
1281 our ($featureCount, $paramCount, $accessCount) = (1,1,1); |
|
1282 our ($attributeMap, $featureMap, $featureParamMap, $accessMap)=('','','',''); |
|
1283 |
|
1284 # Create parser object |
|
1285 my $parser = new XML::Parser; |
|
1286 $parser->setHandlers( 'Start' => \&startC, 'End' => \&endC, 'Char' => \&dataC); |
|
1287 |
|
1288 # Parse the file |
|
1289 open XML,$file or die "Couldn't open $file"; |
|
1290 # Skip the UTF8 BOM - perl 5.6 can't handle it |
|
1291 my $bom; |
|
1292 read XML, $bom, 3; |
|
1293 $bom = join('', map(sprintf('%X', $_), unpack("CCC", $bom))); |
|
1294 print "Testing the following for BOM: $bom\n" if $debug; |
|
1295 seek(XML, 0, 0) if $bom ne 'EFBBBF'; |
|
1296 $plisthash->{'NokiaWidget'} = 2 ; |
|
1297 $plisthash->{'WidgetPackagingFormat'} = "w3c-partial-v1" ; |
|
1298 |
|
1299 $parser->parse(*XML); |
|
1300 close XML; |
|
1301 |
|
1302 if( $language ) |
|
1303 { |
|
1304 if($lang->{$language}) |
|
1305 { |
|
1306 $plisthash->{'BundleDisplayName'} = $lang->{$language}; |
|
1307 } |
|
1308 if($langIcon->{$language}) |
|
1309 { |
|
1310 $plisthash->{'IconPath'} = $langIcon->{$language}; |
|
1311 } |
|
1312 } |
|
1313 die "Widget BundleIdentifier unknown" if !$plisthash->{'BundleIdentifier'}; |
|
1314 die "Widget BundleDisplayName unknown" if !$plisthash->{'BundleDisplayName'}; |
|
1315 |
|
1316 $attributeMap = $attributeMap.WIDGET_NAME.KEY_VALUE_SEPERATOR.$plisthash->{'BundleDisplayName'}.KEY_VALUE_PAIR_SEPERATOR; |
|
1317 |
|
1318 if($non_nokia_widget) |
|
1319 { |
|
1320 $attributeMap = $attributeMap.WIDGET_PROCESSUID.KEY_VALUE_SEPERATOR."200267D6".KEY_VALUE_PAIR_SEPERATOR; |
|
1321 } |
|
1322 else |
|
1323 { |
|
1324 $attributeMap = $attributeMap.WIDGET_PROCESSUID.KEY_VALUE_SEPERATOR."200267C0".KEY_VALUE_PAIR_SEPERATOR; |
|
1325 } |
|
1326 |
|
1327 if($attributeMap) |
|
1328 { |
|
1329 $plisthash->{'AttributeList'} = $attributeMap; |
|
1330 print "\nAttributeList::$plisthash->{'AttributeList'}\n" |
|
1331 } |
|
1332 |
|
1333 # Return result |
|
1334 return $plisthash; |
|
1335 |
|
1336 # Called on a start tag |
|
1337 sub startC |
|
1338 { |
|
1339 my ($p, $el, %atts) = @_; |
|
1340 |
|
1341 if($el eq "widget") |
|
1342 { |
|
1343 if ( $atts{"id"} ) |
|
1344 { |
|
1345 $plisthash->{'BundleIdentifier'} = $atts{"id"}; |
|
1346 $attributeMap = $attributeMap.WIDGET_ID.KEY_VALUE_SEPERATOR.$atts{"id"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1347 |
|
1348 } |
|
1349 if ( $atts{"version"} ) |
|
1350 { |
|
1351 $plisthash->{'BundleVersion'} = $atts{"version"}; |
|
1352 $attributeMap = $attributeMap.WIDGET_VERSION.KEY_VALUE_SEPERATOR.$atts{"version"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1353 } |
|
1354 if ( $atts{"viewmodes"} && $atts{"viewmodes"} eq "all" ) |
|
1355 { |
|
1356 $plisthash->{'MiniViewEnabled'} = 1; |
|
1357 } |
|
1358 $attributeMap = $attributeMap.WIDGET_HEIGHT.KEY_VALUE_SEPERATOR.$atts{"height"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1359 $attributeMap = $attributeMap.WIDGET_WIDTH.KEY_VALUE_SEPERATOR.$atts{"width"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1360 } |
|
1361 elsif ($el eq "icon") |
|
1362 { |
|
1363 if ( $atts{"xml:lang"} ) |
|
1364 { |
|
1365 $localizationIconValue = lc($atts{"xml:lang"}) ; |
|
1366 } |
|
1367 if ( $atts{"src"} ) |
|
1368 { |
|
1369 $plisthash->{'IconPath'} = $atts{"src"}; |
|
1370 } |
|
1371 } |
|
1372 elsif ($el eq "content") |
|
1373 { |
|
1374 $attributeMap = $attributeMap.WIDGET_CONTENT.KEY_VALUE_SEPERATOR.KEY_VALUE_PAIR_SEPERATOR; |
|
1375 if ( $atts{"src"} ) |
|
1376 { |
|
1377 $plisthash->{'MainHTML'} = $atts{"src"}; |
|
1378 $attributeMap = $attributeMap.WIDGET_CONTENT.KEY_ATTR_SEPERATOR."src".KEY_VALUE_SEPERATOR.$atts{"src"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1379 } |
|
1380 } |
|
1381 elsif ($el eq "name") |
|
1382 { |
|
1383 if ( $atts{"xml:lang"} ) |
|
1384 { |
|
1385 $localizationValue = lc($atts{"xml:lang"}) ; |
|
1386 } |
|
1387 } |
|
1388 elsif ($el eq "feature") |
|
1389 { |
|
1390 $featureMap = WIDGET_FEATURE . $featureCount; |
|
1391 $attributeMap = $attributeMap.$featureMap.KEY_VALUE_SEPERATOR.KEY_VALUE_PAIR_SEPERATOR; |
|
1392 |
|
1393 if ( $atts{"name"} ) |
|
1394 { |
|
1395 $attributeMap = $attributeMap.$featureMap.KEY_ATTR_SEPERATOR."name".KEY_VALUE_SEPERATOR.$atts{"name"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1396 my $jilFeature = "http://jil.org/jil"; |
|
1397 if($atts{"name"}=~ m/^$jilFeature/i) |
|
1398 { |
|
1399 $plisthash->{'WidgetPackagingFormat'} = "jil"; |
|
1400 } |
|
1401 } |
|
1402 if ( $atts{"required"} ) |
|
1403 { |
|
1404 $attributeMap = $attributeMap.$featureMap.KEY_ATTR_SEPERATOR."required".KEY_VALUE_SEPERATOR.$atts{"required"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1405 } |
|
1406 } |
|
1407 elsif($el eq "param") |
|
1408 { |
|
1409 $featureParamMap = $featureMap. WIDGET_FEATURE_PARAM . $paramCount; |
|
1410 $attributeMap = $attributeMap.$featureParamMap.KEY_VALUE_SEPERATOR.KEY_VALUE_PAIR_SEPERATOR; |
|
1411 if ($atts{"name"}) |
|
1412 { |
|
1413 $attributeMap = $attributeMap.$featureParamMap.KEY_ATTR_SEPERATOR."name".KEY_VALUE_SEPERATOR.$atts{"name"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1414 } |
|
1415 if ($atts{"value"}) |
|
1416 { |
|
1417 $attributeMap = $attributeMap.$featureParamMap.KEY_ATTR_SEPERATOR."value".KEY_VALUE_SEPERATOR.$atts{"value"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1418 } |
|
1419 } |
|
1420 elsif ($el eq "access" || $el eq "jil:access") |
|
1421 { |
|
1422 $accessMap = WIDGET_ACCESS . $accessCount; |
|
1423 $attributeMap = $attributeMap.$accessMap.KEY_VALUE_SEPERATOR.KEY_VALUE_PAIR_SEPERATOR; |
|
1424 |
|
1425 if ( $atts{"network"} ) |
|
1426 { |
|
1427 $attributeMap = $attributeMap.$accessMap.KEY_ATTR_SEPERATOR."network".KEY_VALUE_SEPERATOR.$atts{"network"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1428 } |
|
1429 if ( $atts{"remotescripts"} ) |
|
1430 { |
|
1431 $attributeMap = $attributeMap.$accessMap.KEY_ATTR_SEPERATOR."remotescripts".KEY_VALUE_SEPERATOR.$atts{"remotescripts"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1432 } |
|
1433 if ( $atts{"localfs"} ) |
|
1434 { |
|
1435 $attributeMap = $attributeMap.$accessMap.KEY_ATTR_SEPERATOR."localfs".KEY_VALUE_SEPERATOR.$atts{"localfs"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1436 } |
|
1437 if ( $atts{"origin"} ) |
|
1438 { |
|
1439 $attributeMap = $attributeMap.$accessMap.KEY_ATTR_SEPERATOR."origin".KEY_VALUE_SEPERATOR.$atts{"origin"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1440 } |
|
1441 if ( $atts{"subdomains"} ) |
|
1442 { |
|
1443 $attributeMap = $attributeMap.$accessMap. KEY_ATTR_SEPERATOR."subdomains".KEY_VALUE_SEPERATOR.$atts{"subdomains"}.KEY_VALUE_PAIR_SEPERATOR; |
|
1444 } |
|
1445 } |
|
1446 elsif ($el eq "NOKIA:sharedlibrary") |
|
1447 { |
|
1448 print " \n\n ^^^^^^^^^^^^^^^^^^^^^^^NOKIA:sharedlibrary ^^^^^^^^^^^^^^^^^^^^ \n\n"; |
|
1449 $isSharedLibrary = 1; |
|
1450 $plisthash->{'WidgetPackagingFormat'} = "shared-library"; |
|
1451 $attributeMap = $attributeMap.WIDGET_NOKIA_SHAREDLIB.KEY_VALUE_SEPERATOR.KEY_VALUE_PAIR_SEPERATOR; |
|
1452 } |
|
1453 $val = ''; |
|
1454 } |
|
1455 |
|
1456 sub docC{} |
|
1457 # Receives character data |
|
1458 sub dataC |
|
1459 { |
|
1460 my ($p, $s) = @_; |
|
1461 $val .= $s; |
|
1462 } |
|
1463 |
|
1464 # Called on an end tag |
|
1465 sub endC |
|
1466 { |
|
1467 my ($p, $el) = @_; |
|
1468 if ($el eq "name") |
|
1469 { |
|
1470 if( $localizationValue ) |
|
1471 { |
|
1472 chomp($val); |
|
1473 $lang->{$localizationValue} = $val; |
|
1474 $localizationValue = ''; |
|
1475 } |
|
1476 else |
|
1477 { |
|
1478 chomp($val); |
|
1479 $plisthash->{'BundleDisplayName'} = $val; |
|
1480 } |
|
1481 } |
|
1482 elsif ($el eq "icon") |
|
1483 { |
|
1484 if( $localizationIconValue ) |
|
1485 { |
|
1486 $langIcon->{$localizationIconValue} = $plisthash->{'IconPath'}; |
|
1487 print "\n %%%%%%%%%%% $localizationIconValue :: $plisthash->{'IconPath'}%%%%%%%%%%%%%\n"; |
|
1488 $localizationIconValue = ''; |
|
1489 } |
|
1490 } |
|
1491 elsif($el eq "feature") |
|
1492 { |
|
1493 $featureCount++; |
|
1494 $paramCount = 0; |
|
1495 } |
|
1496 elsif($el eq "param") |
|
1497 { |
|
1498 $paramCount++; |
|
1499 } |
|
1500 elsif($el eq "access" || $el eq "jil:access") |
|
1501 { |
|
1502 $accessCount++; |
|
1503 } |
|
1504 elsif ($el eq "NOKIA:folder") |
|
1505 { |
|
1506 chomp($val); |
|
1507 $sharedFolderName = $val; |
|
1508 $attributeMap = $attributeMap.WIDGET_NOKIA_SHAREDLIB_FOLDER.KEY_VALUE_SEPERATOR.$sharedFolderName.KEY_VALUE_PAIR_SEPERATOR; |
|
1509 } |
|
1510 elsif ($el eq "NOKIA:widget") |
|
1511 { |
|
1512 chomp($val); |
|
1513 if( lc($val) eq "true" ) |
|
1514 { |
|
1515 $isSharedWidget = 1; |
|
1516 $plisthash->{'WidgetPackagingFormat'} = "w3c-partial-v1"; |
|
1517 } |
|
1518 $attributeMap = $attributeMap.WIDGET_NOKIA_SHAREDLIB_WIDGET.KEY_VALUE_SEPERATOR.$val.KEY_VALUE_PAIR_SEPERATOR; |
|
1519 } |
|
1520 elsif ($el eq "NOKIA:icon") |
|
1521 { |
|
1522 chomp($val); |
|
1523 if( lc($val) eq "true" ) |
|
1524 { |
|
1525 $isSharedIcon = 1; |
|
1526 } |
|
1527 $attributeMap = $attributeMap.WIDGET_NOKIA_SHAREDLIB_ICON.KEY_VALUE_SEPERATOR.$val.KEY_VALUE_PAIR_SEPERATOR; |
|
1528 } |
|
1529 $val = '' |
|
1530 } |
|
1531 |
|
1532 } |
|
1533 |
|
1534 |
|
1535 # Stores the details of files to be added to "rom" |
|
1536 sub addToRomList |
|
1537 { |
|
1538 my ( $self, $drive, $file, $localised ) = @_; |
|
1539 $file = fixFilename($file); |
|
1540 |
|
1541 # All files should be under epoc32 somewhere - need to drop a bit of the path for the rom destination |
|
1542 my $localpath = $self->destLocation($drive); |
|
1543 |
|
1544 my $dest = fixFilename($file); |
|
1545 $dest =~ s/^\Q$localpath\E//i; |
|
1546 |
|
1547 # Add the file to the list for the rom |
|
1548 # It may be localised - in which it'll be put in a different IBY file |
|
1549 $localised = $localised ? '_rsc' : ''; |
|
1550 $self->{"installedFiles${localised}"}->{$drive}->{$file} = $dest; |
|
1551 } |
|
1552 |
|
1553 # Make the IBY file |
|
1554 sub makeIBY |
|
1555 { |
|
1556 my ( $self, $drive, $localised ) = @_; |
|
1557 |
|
1558 # Generate the file name for the IBY file |
|
1559 $localised = $localised ? '_rsc' : ''; |
|
1560 my $name = $drive =~ /^[zZ]/ ? "preinstalledwidgets${localised}.iby" : sprintf("preinstalledwidgets_drive%s${localised}.iby", substr($drive, 0, 1)); |
|
1561 my $iby = fixFilename(catfile($self->{'args'}->{'epocroot'}, 'epoc32', 'rom', 'include', $name)); |
|
1562 print "Generating: $iby\n"; |
|
1563 |
|
1564 mkpath dirname($iby); |
|
1565 open IBY, ">$iby" or die "Failed to open $iby for writing: $!"; |
|
1566 $name =~ s/\./_/g; $name = uc($name); |
|
1567 print IBY "// GENERATED FILE: EDIT WITH CARE\n\#ifndef __${name}__\n\#define __${name}__\n\n"; |
|
1568 foreach my $file ( sort keys %{ $self->{"installedFiles${localised}"}->{$drive} } ) |
|
1569 { |
|
1570 my $dest = $self->{"installedFiles${localised}"}->{$drive}->{$file}; |
|
1571 |
|
1572 # Quote filenames as they may contain spaces! |
|
1573 print IBY "data=\"$file\"\t\"$dest\"\n"; |
|
1574 } |
|
1575 print IBY "\#endif\n"; |
|
1576 close IBY; |
|
1577 } |
|
1578 |
|
1579 # Unregister (with Apparc) existing Widgets |
|
1580 sub unregisterWidgets |
|
1581 { |
|
1582 my ( $self, $drive ) = @_; |
|
1583 |
|
1584 my $registry = fixFilename(catfile($self->destLocation($drive), WIDGET_REGISTRY)); |
|
1585 # If the WRT registry already exists, remove apparc registry info for those widgets |
|
1586 # This should avoid problems with unregistered widget icons in the emulator? |
|
1587 if (-e $registry) |
|
1588 { |
|
1589 print("\n UNREGISTERING WGZ WIDGETS \n"); |
|
1590 |
|
1591 my $ref = XMLin($registry, 'forcearray' => [ 'entry' ], 'keyattr' => { 'prop' => 'content' } ); |
|
1592 foreach my $entry ( @{ $ref->{entry} } ) |
|
1593 { |
|
1594 my $uid = $entry->{prop}->{Uid}->{val}->{content}; |
|
1595 |
|
1596 print "Unregistering existing Widget: $entry->{prop}->{BundleIdentifier}->{val}->{content}\n" if $verbose; |
|
1597 my $dest = $self->regFileName($drive); |
|
1598 my $mbm = catfile($dest, sprintf("[%08x].mbm", $uid)); |
|
1599 my ( $reg, $loc ) = ( catfile($dest, sprintf("%08x_reg.rsc", $uid)), catfile($dest, sprintf("%08x_loc.rsc", $uid)) ); |
|
1600 unlink $mbm, $reg, $loc; |
|
1601 |
|
1602 # We also have to delete the widget directory otherwise it'll be re-registered |
|
1603 my $id = $entry->{prop}->{BundleIdentifier}->{val}->{content}; |
|
1604 $w3c_widget = 0; |
|
1605 my $dir = $self->installDir($drive, $id); |
|
1606 rmtree $dir; |
|
1607 } |
|
1608 } |
|
1609 |
|
1610 $registry = fixFilename(catfile($self->destLocation($drive), CWRT_WIDGET_REGISTRY)); |
|
1611 |
|
1612 # If the CWRT registry already exists, remove apparc registry info for those widgets |
|
1613 # This should avoid problems with unregistered widget icons in the emulator? |
|
1614 if (-e $registry) |
|
1615 { |
|
1616 print("\n UNREGISTERING WGT WIDGETS \n"); |
|
1617 |
|
1618 my $ref = XMLin($registry, 'forcearray' => [ 'entry' ], 'keyattr' => { 'prop' => 'content' } ); |
|
1619 foreach my $entry ( @{ $ref->{entry} } ) |
|
1620 { |
|
1621 my $uid = $entry->{prop}->{Uid}->{val}->{content}; |
|
1622 |
|
1623 print "Unregistering existing Widget: $entry->{prop}->{BundleIdentifier}->{val}->{content}\n" if $verbose; |
|
1624 my $dest = $self->regFileName($drive); |
|
1625 my $mbm = catfile($dest, sprintf("[%08x].mbm", $uid)); |
|
1626 my ( $reg, $loc ) = ( catfile($dest, sprintf("%08x_reg.rsc", $uid)), catfile($dest, sprintf("%08x_loc.rsc", $uid)) ); |
|
1627 unlink $mbm, $reg, $loc; |
|
1628 |
|
1629 # We also have to delete the widget directory otherwise it'll be re-registered |
|
1630 my $id = $entry->{prop}->{BundleIdentifier}->{val}->{content}; |
|
1631 print " Unregistering $id "; |
|
1632 my $basepath = $entry->{prop}->{BasePath}->{val}->{content}; |
|
1633 $w3c_widget = 1; |
|
1634 my $sharedLib = "lib"; #BasePath will have lib only if the widget is shared Library |
|
1635 |
|
1636 if($basepath =~ m/$sharedLib/) |
|
1637 { |
|
1638 $isSharedLibrary = 1; |
|
1639 } |
|
1640 |
|
1641 if($basepath =~ m/\\200267d6\\/i) |
|
1642 { |
|
1643 $non_nokia_widget = 1; |
|
1644 } |
|
1645 #sharedFolderName TBD |
|
1646 my $dir = $self->installDir($drive, $id); |
|
1647 rmtree $dir; |
|
1648 |
|
1649 print("BasePath:$basepath \nIs Non-Nokia? $non_nokia_widget Is SharedLibrary? $isSharedLibrary \nDirectory:$dir \n"); |
|
1650 |
|
1651 $dir =~ s/widgets_21D_4C7/data/; |
|
1652 rmtree $dir; |
|
1653 } |
|
1654 } |
|
1655 #delete CWRT webapp DB if exists |
|
1656 my $cwrtdbPath = catfile($self->destLocation($drive), CWRT_WEBAPP_REGISTRY); |
|
1657 if($cwrtdbPath) |
|
1658 { |
|
1659 print "Deleting CWRT_WEBAPP_REGISTRY \n"; |
|
1660 rmtree $cwrtdbPath; |
|
1661 } |
|
1662 } |
|
1663 |
|
1664 # Make the registry |
|
1665 sub makeRegistry |
|
1666 { |
|
1667 my ($self, $drive, $installed, $w3c) = @_; |
|
1668 |
|
1669 my ($registry); |
|
1670 if($w3c) |
|
1671 { |
|
1672 $registry = fixFilename(catfile($self->destLocation($drive), CWRT_WIDGET_REGISTRY)); |
|
1673 } |
|
1674 else |
|
1675 { |
|
1676 $registry = fixFilename(catfile($self->destLocation($drive), WIDGET_REGISTRY)); |
|
1677 } |
|
1678 |
|
1679 print "\nGenerating: $registry\n"; |
|
1680 |
|
1681 # Write the file |
|
1682 mkpath dirname($registry); |
|
1683 open OUT, ">$registry" or die "Failed to open WidgetEntryStore.xml: $!"; |
|
1684 |
|
1685 print OUT "<?xml version=\"1.0\" encoding=\"UTF-16\" standalone=\"yes\" ?>\n<widgetregistry>\n"; |
|
1686 binmode OUT, ":utf8" if $] >= 5.008; |
|
1687 foreach my $pList ( @$installed ) |
|
1688 { |
|
1689 $self->dumpPList(\*OUT, $pList, $w3c); |
|
1690 } |
|
1691 print OUT "</widgetregistry>\n"; |
|
1692 close OUT; |
|
1693 |
|
1694 # Convert the file to unicode |
|
1695 convert2Unicode($registry); |
|
1696 |
|
1697 # Return the generated file |
|
1698 return $registry; |
|
1699 } |
|
1700 |
|
1701 # Converts a file to Unicode |
|
1702 sub convert2Unicode |
|
1703 { |
|
1704 my $file = shift; |
|
1705 |
|
1706 my @lines; |
|
1707 open IN, $file or die "Failed to open $file: $!"; |
|
1708 binmode IN; |
|
1709 while(<IN>) |
|
1710 { |
|
1711 my $u = utf8($_); |
|
1712 $u->byteswap; |
|
1713 push @lines, $u->utf16; |
|
1714 } |
|
1715 close IN; |
|
1716 |
|
1717 open OUT, ">$file" or die "Failed to open $file for writing: $!"; |
|
1718 binmode OUT; |
|
1719 print OUT pack("CC", 0xff, 0xfe); |
|
1720 print OUT @lines; |
|
1721 close OUT; |
|
1722 } |
|
1723 |
|
1724 sub hashpjw { |
|
1725 my $s = shift; |
|
1726 my $hashval = 0; |
|
1727 my $g; |
|
1728 |
|
1729 for ( split //, $s ) { |
|
1730 $hashval = ( $hashval << 4 ) + ord $_; |
|
1731 |
|
1732 if ($g = $hashval & 0xf0000000) { |
|
1733 $hashval ^= $g >> 23; |
|
1734 } |
|
1735 $hashval &= ~$g; |
|
1736 } |
|
1737 #print "$hashval \n"; |
|
1738 return $hashval; |
|
1739 } |
|
1740 |
|
1741 # Dumps a single PList hash object |
|
1742 sub dumpPList |
|
1743 { |
|
1744 my ($self, $fh, $data, $w3c) = @_; |
|
1745 my @regProperties = ( |
|
1746 [ 'PropertyListVersion', 'int' ], |
|
1747 [ 'BundleIdentifier', 'string' ], |
|
1748 [ 'BundleName', 'string' ], |
|
1749 [ 'BundleDisplayName', 'string' ], |
|
1750 [ 'MainHTML', 'string' ], |
|
1751 [ 'BundleVersion', 'string' ], |
|
1752 # [ 'Height', 'int' ], |
|
1753 # [ 'Width', 'int' ], |
|
1754 [ 'AllowNetworkAccess', 'int' ], |
|
1755 [ 'DriveName', 'string' ], |
|
1756 [ 'BasePath', 'string' ], |
|
1757 [ 'IconPath', 'string' ], |
|
1758 [ 'FileSize', 'int' ], |
|
1759 [ 'Uid', 'int' ], |
|
1760 [ 'NokiaWidget', 'int' ], |
|
1761 [ 'MiniViewEnabled', 'int' ], |
|
1762 [ 'ProcessUid', 'int' ], |
|
1763 [ 'MimeType', 'string'], |
|
1764 [ 'WidgetPackagingFormat', 'string'], |
|
1765 [ 'DBIconPath', 'string'], |
|
1766 [ 'AttributeList', 'string'], |
|
1767 [ 'BlanketPermissionGranted', 'int' ], |
|
1768 [ 'PreInstalled', 'int' ], |
|
1769 ); |
|
1770 |
|
1771 print $fh "<entry>\n"; |
|
1772 my $encodeddata = {}; |
|
1773 foreach my $prop ( @regProperties ) |
|
1774 { |
|
1775 my ( $key, $type ) = @$prop; |
|
1776 if($key ne "DBIconPath" && $key ne "WidgetPackagingFormat" && $key ne "AttributeList" && $key ne "BlanketPermissionGranted" && $key ne "PreInstalled") |
|
1777 { |
|
1778 print $fh "<prop>$key<val>$data->{$key}<type>$type</type></val></prop>\n" if defined $data->{$key}; |
|
1779 } |
|
1780 |
|
1781 if( $w3c ) |
|
1782 { |
|
1783 $encodeddata->{$key} = encodeChar($data->{$key}) if defined $data->{$key}; |
|
1784 } |
|
1785 else |
|
1786 { |
|
1787 if($key eq "BlanketPermissionGranted" || $key eq "PreInstalled") |
|
1788 { |
|
1789 print $fh "<prop>$key<val>$data->{$key}<type>$type</type></val></prop>\n" if defined $data->{$key}; |
|
1790 } |
|
1791 } |
|
1792 } |
|
1793 print $fh "</entry>\n"; |
|
1794 print "\n Is a w3c widget? -- $w3c\n"; |
|
1795 if( $w3c ) |
|
1796 { |
|
1797 my $dbPath = fixFilename(catfile($self->destLocation('C'), CWRT_WEBAPP_REGISTRY)); |
|
1798 mkpath dirname($dbPath); |
|
1799 die "ERROR: Can't find CWRTWebAppRegistry.exe in PATH" if !(my $cmd = findCmd('CWRTWebAppRegistry.exe')); |
|
1800 my $regCmd; |
|
1801 |
|
1802 if($encodeddata->{'AttributeList'}) |
|
1803 { |
|
1804 print "\n AttributeList argument sent to DB\n"; |
|
1805 $regCmd = "$cmd $dbPath $encodeddata->{'BundleIdentifier'} $encodeddata->{'Uid'} $encodeddata->{'BundleDisplayName'} $encodeddata->{'BasePath'} $encodeddata->{'DBIconPath'} $encodeddata->{'WidgetPackagingFormat'} $encodeddata->{'MainHTML'} $encodeddata->{'AttributeList'}"; |
|
1806 } |
|
1807 else |
|
1808 { |
|
1809 print "\n AttributeList argument not sent to DB\n"; |
|
1810 $regCmd = "$cmd $dbPath $encodeddata->{'BundleIdentifier'} $encodeddata->{'Uid'} $encodeddata->{'BundleDisplayName'} $encodeddata->{'BasePath'} $encodeddata->{'DBIconPath'} $encodeddata->{'WidgetPackagingFormat'} $encodeddata->{'MainHTML'}"; |
|
1811 } |
|
1812 print "\n regCmd : $regCmd \n\n"; |
|
1813 system($regCmd); |
|
1814 } |
|
1815 } |
|
1816 |
|
1817 #encode the space character |
|
1818 sub encodeChar |
|
1819 { |
|
1820 my ($encoded) = @_; |
|
1821 |
|
1822 $encoded =~ s/%/%10/g; |
|
1823 $encoded =~ s/\s/%40/g; |
|
1824 $encoded =~ s/\'/%apos/g; |
|
1825 $encoded =~ s/&/%amp/g; |
|
1826 $encoded =~ s/</%lt/g; |
|
1827 $encoded =~ s/>/%gt/g; |
|
1828 $encoded =~ s/\"/%quot/g; |
|
1829 return $encoded; |
|
1830 } |
|
1831 |
|
1832 |
|
1833 # Make secssion |
|
1834 sub makeSecSession |
|
1835 { |
|
1836 my ($self, $drive) = @_; |
|
1837 my $secSession; |
|
1838 my $widget_uid; |
|
1839 if($non_nokia_widget) |
|
1840 { |
|
1841 $widget_uid = CWRT_WIDGET_UI_NON_NOKIA_UID; |
|
1842 } |
|
1843 else |
|
1844 { |
|
1845 $widget_uid = CWRT_WIDGET_UI_UID; |
|
1846 } |
|
1847 |
|
1848 if($isSharedLibrary) |
|
1849 { |
|
1850 $secSession = fixFilename(catfile($self->destLocation($drive), 'private', sprintf("%08X", $widget_uid),"data","lib",$sharedFolderName,"secsession")); |
|
1851 } |
|
1852 else |
|
1853 { |
|
1854 $secSession = fixFilename(catfile($self->destLocation($drive), 'private', sprintf("%08X", $widget_uid),"data", $hashval,"secsession")); |
|
1855 } |
|
1856 print "\nGenerating: $secSession\n\n"; |
|
1857 |
|
1858 # Write the file |
|
1859 mkpath dirname($secSession); |
|
1860 open OUT, ">$secSession" or die "Failed to open SecSession.xml: $!"; |
|
1861 |
|
1862 print OUT "<?xml version=\"1.0\" encoding=\"ISO-8859-1\" ?>\n<accesspolicy>\n"; |
|
1863 if($non_nokia_widget) |
|
1864 { |
|
1865 print OUT "<domain name=\"Operator\">\n"; |
|
1866 } |
|
1867 else |
|
1868 { |
|
1869 print OUT "<domain name=\"TrustedWidgets\">\n"; |
|
1870 } |
|
1871 |
|
1872 if(CWRT_ACCESS_POLICY_PATH) { |
|
1873 my $accesspolicypath = findCmd(CWRT_ACCESS_POLICY_PATH); |
|
1874 # Copy Secsession capabilities from browser_access_policy.xml if it exists |
|
1875 if (-e $accesspolicypath) { |
|
1876 print "Browser Access Policy File exists :: ",$accesspolicypath,"\n" ; |
|
1877 my $accesspolicy = XMLin($accesspolicypath, keyattr => {domain => 'name'}, forcearray =>['domain'] ); |
|
1878 my $domainName; |
|
1879 |
|
1880 if($non_nokia_widget) |
|
1881 { |
|
1882 $domainName = $accesspolicy->{domain}->{Operator}; |
|
1883 } |
|
1884 else |
|
1885 { |
|
1886 $domainName = $accesspolicy->{domain}->{TrustedWidgets}; |
|
1887 } |
|
1888 |
|
1889 my $xml = XMLout($domainName, keeproot => 1); |
|
1890 print OUT $xml; |
|
1891 } |
|
1892 #browser_access_policy.xml does not exist in the path |
|
1893 else { |
|
1894 print "\n Browser Access policy file '$accesspolicypath' does not exist in the given path\n"; |
|
1895 if(!$non_nokia_widget) |
|
1896 { |
|
1897 print OUT " <capability name=\"Download\"/>\n <capability name=\"ApplicationManagement\"/>\n <capability name=\"WebAppUpdate\"/>\n <capability name=\"DeviceConfiguration\"/>\n <capability name=\"NokiaAccount\"/>\n <capability name=\"ConnectionManagement\"/>\n <capability name=\"ApplicationManagement.Launch\"/>\n <capability name=\"SecureStorage\"/>\n <capability name=\"EventsAndMessaging\"/>\n <capability name=\"Vibra\"/>\n <capability name=\"RuntimeInfo\"/>\n <capability name=\"OviMessagingBus\"/>\n <capability name=\"NativeUpdate\"/>\n <capability name=\"LocalConnectivity\"/>\n <capability name=\"Network\"/>\n <capability name=\"Cryptographic\"/>\n <capability name=\"pim.*\"/>\n <capability name=\"devicestatus.*\"/>\n <capability name=\"message.*\"/>\n <capability name=\"sensor.*\"/>\n <capability name=\"landmark.*\"/>\n <capability name=\"camera.*\"/>\n <capability name=\"commlog.*\"/>\n <capability name=\"media.*\"/>\n <capability name=\"io.file.*\"/>\n <capability name=\"player.*\"/>\n <capability name=\"location.position\"/>\n <capability name=\"telephony.voicecall\"/>\n <capability name=\"NokiaAdEnabler\"/>\n"; |
|
1898 } |
|
1899 } |
|
1900 } |
|
1901 #Access policy path not defined |
|
1902 else { |
|
1903 print "\n Browser Access policy path not defined \n"; |
|
1904 if(!$non_nokia_widget) |
|
1905 { |
|
1906 print OUT " <capability name=\"Download\"/>\n <capability name=\"ApplicationManagement\"/>\n <capability name=\"WebAppUpdate\"/>\n <capability name=\"DeviceConfiguration\"/>\n <capability name=\"NokiaAccount\"/>\n <capability name=\"ConnectionManagement\"/>\n <capability name=\"ApplicationManagement.Launch\"/>\n <capability name=\"SecureStorage\"/>\n <capability name=\"EventsAndMessaging\"/>\n <capability name=\"Vibra\"/>\n <capability name=\"RuntimeInfo\"/>\n <capability name=\"OviMessagingBus\"/>\n <capability name=\"NativeUpdate\"/>\n <capability name=\"LocalConnectivity\"/>\n <capability name=\"Network\"/>\n <capability name=\"Cryptographic\"/>\n <capability name=\"pim.*\"/>\n <capability name=\"devicestatus.*\"/>\n <capability name=\"message.*\"/>\n <capability name=\"sensor.*\"/>\n <capability name=\"landmark.*\"/>\n <capability name=\"camera.*\"/>\n <capability name=\"commlog.*\"/>\n <capability name=\"media.*\"/>\n <capability name=\"io.file.*\"/>\n <capability name=\"player.*\"/>\n <capability name=\"location.position\"/>\n <capability name=\"telephony.voicecall\"/>\n <capability name=\"NokiaAdEnabler\"/>\n"; |
|
1907 } |
|
1908 } |
|
1909 |
|
1910 print OUT "</domain>\n"; |
|
1911 print OUT "</accesspolicy>\n"; |
|
1912 close OUT; |
|
1913 |
|
1914 # Return the generated file |
|
1915 return $secSession; |
|
1916 } |
|
1917 __END__ |
|
1918 |
|
1919 =head1 NAME |
|
1920 |
|
1921 installwidgets.pl - A script for generating all the files needed to install Widgets |
|
1922 |
|
1923 =head1 SYNOPSIS |
|
1924 |
|
1925 installwidgets.pl [-h] [-ver] [-v] [-debug] [-e <dir>] [-l <lang_code|lproj.xml>] config.ini |
|
1926 |
|
1927 Options: |
|
1928 -help|h Show this help |
|
1929 -version|ver Show version number |
|
1930 -verbose|v Show verbose output |
|
1931 -debug Show debug output |
|
1932 -epocroot|e Override value of EPOCROOT |
|
1933 -localization|l lproj_dir |
|
1934 |
|
1935 A script for generating all the files needed to preinstall Widgets. |
|
1936 |
|
1937 Example: |
|
1938 perl installwidgets.pl -l fr config.ini Install widgets listed in config.ini using French localization |
|
1939 |
|
1940 Author: |
|
1941 peter.harper@sosco.com |
|
1942 |
|
1943 =head1 DESCRIPTION |
|
1944 |
|
1945 This tool can be used to pre-generate all the files needed to install Widgets. The tool and its dependencies can be placed anywhere on your PATH. |
|
1946 It generates the results in the epoc32 folder - in the appropriate locations for the emulator. |
|
1947 It finds the epoc32 folder using the EPOCROOT environment variable which can be overridden via the -e command line option. |
|
1948 |
|
1949 =head2 CONFIG FILE |
|
1950 |
|
1951 The preferred way to run the tool is via a configuration INI file. |
|
1952 You can list the widgets to install on each drive. You can specify the exact location of the Widget, otherwise it will try and find the Widget via EPOCROOT. |
|
1953 |
|
1954 You can specify whether a widget is intended for the homescreen by adding the text [HomeScreen] after the filename |
|
1955 This will set the BlanketPermissionGranted attribute in the registry. |
|
1956 Widgets intended for the homescreen must have the MiniViewEnabled attribute set in its PLIST file otherwise an error is generated. |
|
1957 |
|
1958 # Widgets to be pre-installed for the ROM |
|
1959 [drive-z] |
|
1960 \somepath\foo.wgz |
|
1961 \somepath\bar.wgz |
|
1962 |
|
1963 # Widgets for the internal disk |
|
1964 [drive-c] |
|
1965 \somepath\widget1.wdgt.zip [HomeScreen] |
|
1966 |
|
1967 # Widgets for the removable disk |
|
1968 [drive-e] |
|
1969 \somepath\widget2.wdgt.zip |
|
1970 |
|
1971 # Commands to run at the end |
|
1972 [run-commands] |
|
1973 dostuff.pl |
|
1974 domorestuff.exe |
|
1975 |
|
1976 =head2 DEPENDENCIES |
|
1977 |
|
1978 The tool has some dependencies which must exist for it to work. |
|
1979 |
|
1980 =over |
|
1981 |
|
1982 =item 1 |
|
1983 |
|
1984 png2mbm.pl - A script to generate an MBM file from a PNG |
|
1985 |
|
1986 =item 2 |
|
1987 |
|
1988 WidgetRegFiles.exe - an EXE which can generate Symbian REG and LOC files for registering non-native Widget apps. |
|
1989 This tool is built with "SymPort" a native tools port of basic Symbian OS services. |
|
1990 |
|
1991 =item 3 |
|
1992 |
|
1993 7z/unzip - For extracting files from the Widget archive. |
|
1994 7Zip will be used in preference to unzip if it's found on your path because it handles Unicode a better. |
|
1995 |
|
1996 =item 4 |
|
1997 |
|
1998 GD.pm - Perl support for the GD graphics library for PNG support, see http://www.libgd.org . |
|
1999 |
|
2000 =back |
|
2001 |
|
2002 =head3 INSTALLING GD |
|
2003 |
|
2004 You can install GD automatically with a simple command - however the command you need to use differs depending on the version of Perl you have installed. |
|
2005 At the time of writing Symbian requires Perl version 5.6 - although in my experience Perl 5.8 works okay. To find out which version of Perl you have type "perl -v" on the command line. |
|
2006 |
|
2007 To install the GD library: |
|
2008 |
|
2009 =over |
|
2010 |
|
2011 =item * |
|
2012 |
|
2013 For Perl v5.6: "ppm install http://theoryx5.uwinnipeg.ca/ppmpackages/GD.ppd " |
|
2014 |
|
2015 =item * |
|
2016 |
|
2017 For Perl v5.8: "ppm install http://theoryx5.uwinnipeg.ca/ppms/GD.ppd " |
|
2018 |
|
2019 =back |
|
2020 |
|
2021 =head2 WIDGET INSTALL PROCESS |
|
2022 |
|
2023 Here's a detailed breakdown of what the script does. |
|
2024 |
|
2025 =over |
|
2026 |
|
2027 =item 1 |
|
2028 |
|
2029 It gets the lists of Widgets from the config.ini file passed on the command line. |
|
2030 This process is repeated for all Widgets and all drives listed in the config file. |
|
2031 |
|
2032 =item 2 |
|
2033 |
|
2034 Any existing Widgets listed in "private\10282f06\WidgetEntryStore.xml" are deleted from the epoc32 tree. |
|
2035 This ensures that there are no problems when testing Widgets in the emulator. |
|
2036 |
|
2037 =item 3 |
|
2038 |
|
2039 All the compressed files in the Widget are extracted to a temporary folder. |
|
2040 |
|
2041 =item 4 |
|
2042 |
|
2043 The details for the Widget are loaded from its "Info.plist" file. |
|
2044 |
|
2045 =item 5 |
|
2046 |
|
2047 A UID is chosen for the widget. This differs depending on whether installation is for an internal drive (z: or c:) or an external drive (e: etc). |
|
2048 |
|
2049 =item 5 |
|
2050 |
|
2051 A Symbian MBM file is generated from the "Icon.png" file supplied by the Widgets. |
|
2052 Three different sized icons are generated "88x88", "32x32" and "24x24". |
|
2053 The MBM file is placed in "private/10003a3f/import/apps/NonNative/Resource/[<UID>].mbm". |
|
2054 |
|
2055 =item 6 |
|
2056 |
|
2057 "WidgetRegFiles.exe" is executed to generate REG and LOC resource files used to register the Widget as an app in Symbian OS. |
|
2058 These files are placed in "private/10003a3f/import/apps/NonNative/Resource". |
|
2059 |
|
2060 =item 7 |
|
2061 |
|
2062 All the widgets files are copied to a folder under "private\10282822". |
|
2063 The Widget's bundle identifier is used to create a unique folder under here for the Widget. |
|
2064 |
|
2065 =item 8 |
|
2066 |
|
2067 The Widget registry is generated in "private\10282f06\WidgetEntryStore.xml" |
|
2068 |
|
2069 =item 9 |
|
2070 |
|
2071 If Widgets are being preinstalled for ROM an IBY file is created in "epoc32\rom\include\preinstalledwidgets.iby". |
|
2072 A separate IBY file is generated for the localised parts of a Widget "epoc32\rom\include\preinstalledwidgets_rsc.iby". |
|
2073 Separate IBY files (per drive) are generated for Widgets preinstalled to UDA, e.g. preinstalledwidgets_driveC.iby and preinstalledwidgets_driveC_rsc.iby. |
|
2074 These IBY files can be used to add all the Widgets to ROM, ROFS or UDA. |
|
2075 |
|
2076 =back |
|
2077 |
|
2078 =head3 INSTALLING ON HARDWARE USING iMaker |
|
2079 |
|
2080 =over |
|
2081 |
|
2082 =item 1 |
|
2083 |
|
2084 Create the following folder structure at the root level. |
|
2085 |
|
2086 X:\variants\content |
|
2087 |
|
2088 =item 2 |
|
2089 |
|
2090 Copy the files specified in the generated ibys (preinstalledwidgets_driveC.iby and preinstalledwidgets_driveC_rsc.iby) to X:\variants\content. Preserve the dir structure. (Note, this step will be automated in the future) |
|
2091 |
|
2092 For example if you want the following file on UDA (User Disk Area, C drive on phone) at the following location C:\private\10282f06\WidgetEntryStore.xml |
|
2093 |
|
2094 Drop the file under X:\variants\content\private\10282f06\WidgetEntryStore.xml |
|
2095 |
|
2096 =item 3 |
|
2097 |
|
2098 Run the foll command to generate UDA |
|
2099 |
|
2100 B<Gadget:> |
|
2101 X:\epoc32\tools>imaker -f /epoc32/rom/s60_makefiles/image_conf_sp_rnd_gadget.mk VARIANT_DIR=/variants variantuda |
|
2102 |
|
2103 B<Tube:> |
|
2104 Y:\epoc32\tools>imaker -f /epoc32/rom/config/ncp52/tube/image_conf_tube_ui.mk VARIANT_DIR=/variants variantuda |
|
2105 |
|
2106 =item 4 |
|
2107 |
|
2108 Flash the fpsx file generated under X:\epoc32\rombuild\gadget\uda for Gadget and Y:\epoc32\rombuild\tube\uda for Tube to your device. |
|
2109 |
|
2110 Note: More info on iMaker tool at: L<http://configurationtools.nmp.nokia.com/imaker/wiki/iMakerUserGuide> |
|
2111 |
|
2112 =back |
|
2113 |
|
2114 =head3 LOCALISATION |
|
2115 |
|
2116 Widget handles localization by providing localized resources in various language project directories(lproj_dir), one level deep than the root directory. In order to specify a language variant for pre-installing widget, you need to provide the language project directory name, e.g. 'en' for english, 'fr' for French. |
|
2117 |
|
2118 A list of Nokia supported languages can be found in Widget_lproj.xml or at L<http://wiki.forum.nokia.com/index.php/Web_Runtime_localisation_support>. If the widget does not provide the localized resource for the variant you specified, the default resources in widget's home directory will be used instead. |
|
2119 |
|
2120 =head3 NOTES |
|
2121 |
|
2122 =over |
|
2123 |
|
2124 =item 1 |
|
2125 |
|
2126 The location of the private folder is in the appropriate place for the files to appear in the emulator. |
|
2127 This is different depending on the intended destination drive (see -dd command line option) for the Widget. |
|
2128 e.g. "epoc32/release/winscw/udeb/z/", "epoc32/winscw/c" or "epoc32/winscw/e" |
|
2129 |
|
2130 =item 2 |
|
2131 |
|
2132 Files are extracted to epoc32 on the current drive relative to the EPOCROOT environment variable or the value given for -epocroot (-e) on the command line. |
|
2133 |
|
2134 =item 3 |
|
2135 |
|
2136 A different IBY file is generated for each drive. |
|
2137 |
|
2138 =over |
|
2139 |
|
2140 =item * |
|
2141 |
|
2142 Z: - \epoc32\rom\include\preinstalledwidgets.iby |
|
2143 |
|
2144 =item * |
|
2145 |
|
2146 C: - \epoc32\rom\include\preinstalledwidgets_driveC.iby |
|
2147 |
|
2148 =item * |
|
2149 |
|
2150 E: - \epoc32\rom\include\preinstalledwidgets_driveE.iby |
|
2151 |
|
2152 =back |
|
2153 |
|
2154 There are separate resource files for localised resources e.g. preinstalledwidgets_rsc.iby. |
|
2155 |
|
2156 =cut |