Rev 1867 Rev 1868
1 <?php 1 <?php
2   2  
3   3  
4 define('WEBSVNSCRIPT', '/WebSVN/listing.php', true); // Cesta k WebSVN skriptu 4 define('WEBSVNSCRIPT', '/WebSVN/listing.php', true); // Cesta k WebSVN skriptu
5 define('WIKISERVER', 'wiki', true); // Jméno Wiki serveru (co se strčí místo jména stávajícího serveru) 5 define('WIKISERVER', 'wiki', true); // Jméno Wiki serveru (co se strčí místo jména stávajícího serveru)
6   6  
7   7  
8 function Error($str) 8 function Error($str)
9 // Ukončení běhu při chybhě 9 // Ukončení běhu při chybhě
10 { 10 {
11 echo '<b>Ups...<b>'.$str; 11 echo '<b>Ups...<b>'.$str;
12 exit; 12 exit;
13 } 13 }
14   14  
15   15  
16 function WrapString($String, $Wrap=0) 16 function WrapString($String, $Wrap=0)
17 // Provede odstranění nadbytečných bílých znaků 17 // Provede odstranění nadbytečných bílých znaků
18 // a volitleně (pokud je $Wrap nenulové) provede zálámání na zadaný počet znaků 18 // a volitleně (pokud je $Wrap nenulové) provede zálámání na zadaný počet znaků
19 { 19 {
20 // Odstaň bílé znaky 20 // Odstaň bílé znaky
21 $String = preg_replace('/\s+/su', ' ', $String); 21 $String = preg_replace('/\s+/su', ' ', $String);
22 22
23 // Volitelné nalámání 23 // Volitelné nalámání
24 if ($Wrap!=0) 24 if ($Wrap!=0)
25 $String = wordwrap($String, $Wrap); 25 $String = wordwrap($String, $Wrap);
26   26  
27 return $String; 27 return $String;
28 } 28 }
29   29  
30   30  
31 function CheckEnvironment() 31 function CheckEnvironment()
32 { 32 {
33 // POZOR DOPSAT 33 // POZOR DOPSAT
34 } 34 }
35   35  
36 function CheckWebPath($Path) 36 function CheckWebPath($Path)
37 // Zadaná cesta se chápe jako cesta uvnitř webu (vzhledem k document root). 37 // Zadaná cesta se chápe jako cesta uvnitř webu (vzhledem k document root).
38 // Případné relativní části (/../..) převede na absolutní a zkontroluje 38 // Případné relativní části (/../..) převede na absolutní a zkontroluje
39 // výsledek, zda nevybočuje mimo web. 39 // výsledek, zda nevybočuje mimo web.
40 // Na začátku nechá lomítko, na konci ho odstraní. 40 // Na začátku nechá lomítko, na konci ho odstraní.
41 // Pokud je výsledná cesta neplatná, vrátí prázdný řetězec. Kontroluje, zda 41 // Pokud je výsledná cesta neplatná, vrátí prázdný řetězec. Kontroluje, zda
42 // zadaný adresář existuje. 42 // zadaný adresář existuje.
43 { 43 {
44 // Převeď cestu na opravdovou, odstraň relativní odkazy 44 // Převeď cestu na opravdovou, odstraň relativní odkazy
45 $Path = realpath($_SERVER['DOCUMENT_ROOT'].$Path); 45 $Path = realpath($_SERVER['DOCUMENT_ROOT'].$Path);
46   46  
47 // Zkontroluj, zda jsme nevybočili z webu 47 // Zkontroluj, zda jsme nevybočili z webu
48 if ( 0 == strncmp($Path, $_SERVER['DOCUMENT_ROOT'], strlen($_SERVER['DOCUMENT_ROOT']) ) ) 48 if ( 0 == strncmp($Path, $_SERVER['DOCUMENT_ROOT'], strlen($_SERVER['DOCUMENT_ROOT']) ) )
49 { 49 {
50 // Shoda - odstraň DOCUMENT_ROOT část cesty 50 // Shoda - odstraň DOCUMENT_ROOT část cesty
51 return substr( $Path, strlen($_SERVER['DOCUMENT_ROOT']) ); 51 return substr( $Path, strlen($_SERVER['DOCUMENT_ROOT']) );
52 } 52 }
53 else 53 else
54 { 54 {
55 // Vybočili jsme z webu 55 // Vybočili jsme z webu
56 return ''; 56 return '';
57 } 57 }
58 } 58 }
59   59  
60   60  
61 function GetSection($Path) 61 function GetSection($Path)
62 // Vrátí první část cesty (to je to, co nazýváme sekce) 62 // Vrátí první část cesty (to je to, co nazýváme sekce)
63 { 63 {
64 // Ignoruj lomítko na začátku 64 // Ignoruj lomítko na začátku
65 $Path = ltrim($Path, '/'); 65 $Path = ltrim($Path, '/');
66   66  
67 // Vykousni první slovo až po lomítko 67 // Vykousni první slovo až po lomítko
68 $Section = substr($Path, 0, strpos($Path.'/', '/')); 68 $Section = substr($Path, 0, strpos($Path.'/', '/'));
69   69  
70 // Ošetři root 70 // Ošetři root
71 if ($Section=='') 71 if ($Section=='')
72 $Section = "(root)"; 72 $Section = "(root)";
73   73  
74 // Vrať výsledek 74 // Vrať výsledek
75 return $Section; 75 return $Section;
76 } 76 }
77   77  
78   78  
79 function LangFile($FileName, $Lang) 79 function LangFile($FileName, $Lang)
80 // Dostává cestu k souboru a jazyk. Zkusí ověřit existenci souboru s doplněným 80 // Dostává cestu k souboru a jazyk. Zkusí ověřit existenci souboru s doplněným
81 // zadaným jazykem, dále zkusí angličtinu a nakonec zkusí nemodifikovaný název. 81 // zadaným jazykem, dále zkusí angličtinu a nakonec zkusí nemodifikovaný název.
82 // Vrací jméno souboru s tou variantou, kterou našel. 82 // Vrací jméno souboru s tou variantou, kterou našel.
83 { 83 {
84 // Najdi příponu (poslední tečka) 84 // Najdi příponu (poslední tečka)
85 $Pos = strrpos($FileName, '.'); 85 $Pos = strrpos($FileName, '.');
86 if ($Pos === false) 86 if ($Pos === false)
87 return ''; 87 return '';
88 if ($Pos < strrchr($FileName, '/')) 88 if ($Pos < strrchr($FileName, '/'))
89 return ''; 89 return '';
90   90  
91 // Zkus doplnit zadaný jazyk 91 // Zkus doplnit zadaný jazyk
92 $File = substr($FileName, 0, $Pos+1).$Lang.substr($FileName, $Pos); 92 $File = substr($FileName, 0, $Pos+1).$Lang.substr($FileName, $Pos);
93   93  
94 if (is_file($File)) 94 if (is_file($File))
95 return $File; 95 return $File;
96   96  
97 // Zkus doplnit en 97 // Zkus doplnit en
98 $File = substr($FileName, 0, $Pos+1).'en'.substr($FileName, $Pos); 98 $File = substr($FileName, 0, $Pos+1).'en'.substr($FileName, $Pos);
99 if (is_file($File)) 99 if (is_file($File))
100 return $File; 100 return $File;
101   101  
102 // Zkus soubor tak jak je 102 // Zkus soubor tak jak je
103 if (is_file($FileName)) 103 if (is_file($FileName))
104 return $FileName; 104 return $FileName;
105   105  
106 // Nic nevyšlo 106 // Nic nevyšlo
107 return ''; 107 return '';
108 } 108 }
109   109  
110 function CreateLink($InternetWebPath, $Title='') -  
111 // Vstupem je cesta do jiného webu. -  
112 // a volitelně title (zobrazí se při najetí myši na odkaz). -  
113 // Vrací řetězec s odkazem -  
114 { -  
115 $Link = '<a href="'.$InternetWebPath.'"'; -  
116 if ($Title != '') -  
117 $Link .= "\n".' title="'.$Title.'"'; -  
118 $Link .= '>'; -  
119 $Link .= $Title; -  
120 $Link .= '</a>'; -  
121   -  
122 return $Link; -  
123 } -  
124   110  
125 function CreateFileLink($FileName, $Title='') 111 function CreateFileLink($FileName, $Title='')
126 // Vstupem je cesta k souboru v rámci webu (od DOCUMENT_ROOT) 112 // Vstupem je cesta k souboru v rámci webu (od DOCUMENT_ROOT)
127 // a volitelně title (zobrazí se při najetí myši na odkaz). 113 // a volitelně title (zobrazí se při najetí myši na odkaz).
128 // Vrací řetězec s odkazem 114 // Vrací řetězec s odkazem
129 { 115 {
130 $FileName = htmlspecialchars($FileName); 116 $FileName = htmlspecialchars($FileName);
131   117  
132 $Link = '<a href="http://'.$_SERVER['SERVER_NAME'].$FileName.'"'; 118 $Link = '<a href="http://'.$_SERVER['SERVER_NAME'].$FileName.'"';
133 if ($Title != '') 119 if ($Title != '')
134 $Link .= "\n".' title="'.$Title.'"'; 120 $Link .= "\n".' title="'.$Title.'"';
135 $Link .= '>'; 121 $Link .= '>';
136 if (basename($FileName)=='') 122 if (basename($FileName)=='')
137 $Link .= '(root)'; 123 $Link .= '(root)';
138 else 124 else
139 $Link .= basename($FileName); 125 $Link .= basename($FileName);
140 $Link .= '</a>'; 126 $Link .= '</a>';
141   127  
142 return $Link; 128 return $Link;
143 } 129 }
144   130  
145 function CreateFilesLink($Path, $Glue, $Title='') 131 function CreateFilesLink($Path, $Glue, $Title='')
146 // Sestaví odkazy na všechny soubory zadaného jména slepené lepidlem 132 // Sestaví odkazy na všechny soubory zadaného jména slepené lepidlem
147 // Volitelně lze zadat title (zobrazí se při najetí myši na odkaz). 133 // Volitelně lze zadat title (zobrazí se při najetí myši na odkaz).
148 // Cesta se chápe v rámci webu (od DOCUMENT_ROOT). 134 // Cesta se chápe v rámci webu (od DOCUMENT_ROOT).
149 // Vrací řetězec s odkazy 135 // Vrací řetězec s odkazy
150 { 136 {
151 // Při lepení odřádkujeme aby byl generovaný kód strukturovaný 137 // Při lepení odřádkujeme aby byl generovaný kód strukturovaný
152 $Glue = ' '.$Glue."\n"; 138 $Glue = ' '.$Glue."\n";
153 139
154 // Pro každou položku 140 // Pro každou položku
155 $Line = ''; 141 $Line = '';
156 $Ofset = strlen($_SERVER['DOCUMENT_ROOT']); 142 $Ofset = strlen($_SERVER['DOCUMENT_ROOT']);
157 foreach (glob($_SERVER['DOCUMENT_ROOT'].$Path, 0) as $DirName) 143 foreach (glob($_SERVER['DOCUMENT_ROOT'].$Path, 0) as $DirName)
158 { 144 {
159 // Sestav odkaz 145 // Sestav odkaz
160 if (is_file($DirName)) 146 if (is_file($DirName))
161 $Line .= $Glue.CreateFileLink(substr($DirName, $Ofset), $Title); 147 $Line .= $Glue.CreateFileLink(substr($DirName, $Ofset), $Title);
162 } 148 }
163   149  
164 // Vrať výsledek ale bez lepidla na začátku 150 // Vrať výsledek ale bez lepidla na začátku
165 return substr($Line, strlen($Glue)); 151 return substr($Line, strlen($Glue));
166 } 152 }
167   153  
168   154  
169 function CreateScriptLink($Params, $DirName='', $Title='') 155 function CreateScriptLink($Params, $DirName='', $Title='')
170 // Vytvoř odkaz na skript s parametry skriptu. Parametry jsou předány jako 156 // Vytvoř odkaz na skript s parametry skriptu. Parametry jsou předány jako
171 // pole s jmény položek a hodnotami. Volitelná položka $DirName se přidá k cestě 157 // pole s jmény položek a hodnotami. Volitelná položka $DirName se přidá k cestě
172 // v položce $Params. Pokud je zadaná cesta prázdná nebo / zobrazí se slovo root 158 // v položce $Params. Pokud je zadaná cesta prázdná nebo / zobrazí se slovo root
173 { 159 {
174 // Připoj volitelné jméno adresáře 160 // Připoj volitelné jméno adresáře
175 if ($DirName!='') 161 if ($DirName!='')
176 $Params['path'] = $Params['path'].'/'.$DirName; 162 $Params['path'] = $Params['path'].'/'.$DirName;
177   163  
178 // Odvoď jméno z názvu adresáře 164 // Odvoď jméno z názvu adresáře
179 $DirName = basename($Params['path']); 165 $DirName = basename($Params['path']);
180 if ($DirName == '') 166 if ($DirName == '')
181 { 167 {
182 $DirName = '(root)'; 168 $DirName = '(root)';
183 $Params['path'] = '/'; 169 $Params['path'] = '/';
184 } 170 }
185   171  
186 // Zpracuj pole parametrů 172 // Zpracuj pole parametrů
187 $ParamsLine = ''; 173 $ParamsLine = '';
188 $LocalGlue = ''; 174 $LocalGlue = '';
189 foreach($Params as $Key => $Value) 175 foreach($Params as $Key => $Value)
190 { 176 {
191 $ParamsLine .= $LocalGlue.$Key.'='.urlencode($Value); 177 $ParamsLine .= $LocalGlue.$Key.'='.urlencode($Value);
192 $LocalGlue = '&amp;'; 178 $LocalGlue = '&amp;';
193 } 179 }
194 unset($LocalGlue); 180 unset($LocalGlue);
195   181  
196 // Sestav odkaz 182 // Sestav odkaz
197 $Link = '<a href="'.$_SERVER['SCRIPT_NAME'].'?'.$ParamsLine.'"'; 183 $Link = '<a href="'.$_SERVER['SCRIPT_NAME'].'?'.$ParamsLine.'"';
198 if ($Title!='') 184 if ($Title!='')
199 $Link .= "\n".' title="'.htmlspecialchars($Title).'"'; 185 $Link .= "\n".' title="'.htmlspecialchars($Title).'"';
200 $Link .= ">$DirName</a>"; 186 $Link .= ">$DirName</a>";
201   187  
202 return $Link; 188 return $Link;
203 } 189 }
204   190  
205   191  
206 function CreateScriptLinkPath($Params, $Title) 192 function CreateScriptLinkPath($Params, $Title)
207 // Vstupem je cesta (začínající /) od DOCUMENT_ROOT 193 // Vstupem je cesta (začínající /) od DOCUMENT_ROOT
208 // Výsledkem je řádka odkazů na jednotlivé adresáře cesty 194 // Výsledkem je řádka odkazů na jednotlivé adresáře cesty
209 // Odkazy jsou odkazy na skript s parametrem cesty a jazyka 195 // Odkazy jsou odkazy na skript s parametrem cesty a jazyka
210 // Vstupem je pole parametrů ze kterých použijeme parametr 'path', 196 // Vstupem je pole parametrů ze kterých použijeme parametr 'path',
211 // ostatní jen umístíme do odkazů. 197 // ostatní jen umístíme do odkazů.
212 { 198 {
213 // Rozděl cestu na kousky 199 // Rozděl cestu na kousky
214 $Parts = explode('/', $Params['path']); 200 $Parts = explode('/', $Params['path']);
215 if ($Parts[0]=='') 201 if ($Parts[0]=='')
216 array_shift($Parts); // vyhoď případnou prázdnou položku 202 array_shift($Parts); // vyhoď případnou prázdnou položku
217   203  
218 // Začínáme od rootu 204 // Začínáme od rootu
219 $TempPath = ''; 205 $TempPath = '';
220   206  
221 // Odkaz na root (je vždy) 207 // Odkaz na root (je vždy)
222 $Params['path'] = $TempPath; 208 $Params['path'] = $TempPath;
223 $Link = CreateScriptLink($Params, '', $Title); 209 $Link = CreateScriptLink($Params, '', $Title);
224   210  
225 // Je něco víc než root? 211 // Je něco víc než root?
226 if (isset($Parts[0]) && $Parts[0]!='') 212 if (isset($Parts[0]) && $Parts[0]!='')
227 { 213 {
228 // Odkaz na všechny další adresáře cesty 214 // Odkaz na všechny další adresáře cesty
229 foreach($Parts as $Key => $Value) 215 foreach($Parts as $Key => $Value)
230 { 216 {
231 // Přidej adresář 217 // Přidej adresář
232 $TempPath .= '/'.$Value; 218 $TempPath .= '/'.$Value;
233   219  
234 // Vyrob odkaz 220 // Vyrob odkaz
235 $Link .= " /\n"; 221 $Link .= " /\n";
236 $Params['path'] = $TempPath; 222 $Params['path'] = $TempPath;
237 $Link .= CreateScriptLink($Params, '', $Title); 223 $Link .= CreateScriptLink($Params, '', $Title);
238 } 224 }
239 } 225 }
240   226  
241 // Vrať výsledek 227 // Vrať výsledek
242 return $Link; 228 return $Link;
243 } 229 }
244   230  
245   231  
246 function CreateSVNLink($Path, $Lang, $Title) 232 function CreateSVNLink($Path, $Lang, $Title)
247 // Vytvoří odkaz do WebSVN podle cesty a jazyka 233 // Vytvoří odkaz do WebSVN podle cesty a jazyka
248 { 234 {
249 $Path = htmlspecialchars($Path); 235 $Path = htmlspecialchars($Path);
250   236  
251 $Link = '<a href="http://'; 237 $Link = '<a href="http://';
252 $Link .= $_SERVER['SERVER_NAME']; 238 $Link .= $_SERVER['SERVER_NAME'];
253 $Link .= WEBSVNSCRIPT; 239 $Link .= WEBSVNSCRIPT;
254 $Link .= '?repname=MLAB&amp;path='; 240 $Link .= '?repname=MLAB&amp;path=';
255 $Link .= $Path; 241 $Link .= $Path;
256 $Link .= '/'; 242 $Link .= '/';
257 if ($Lang=='cs') 243 if ($Lang=='cs')
258 $Link .= '&amp;langchoice=Česky'; 244 $Link .= '&amp;langchoice=Česky';
259 else 245 else
260 $Link .= '&amp;langchoice=English'; 246 $Link .= '&amp;langchoice=English';
261 $Link .= '"'; 247 $Link .= '"';
262 if ($Title!='') 248 if ($Title!='')
263 $Link .= "\n".' title="'.htmlspecialchars($Title).'"'; 249 $Link .= "\n".' title="'.htmlspecialchars($Title).'"';
264 $Link .= '>WebSVN</a>'; 250 $Link .= '>WebSVN</a>';
265   251  
266 return $Link; 252 return $Link;
267 } 253 }
268   254  
269   255  
270 function CreateWikiLink($Name, $Lang, $Title) 256 function CreateWikiLink($Name, $Lang, $Title)
271 // Vytvoří odkaz do Wiki podle jména modulu a jazyka 257 // Vytvoří odkaz do Wiki podle jména modulu a jazyka
272 { 258 {
273 // Ukousni verzi a revizi modulu (například 01A) 259 // Ukousni verzi a revizi modulu (například 01A)
274 $Name = preg_replace('/[0-9][0-9][a-z]$/i', '', $Name); 260 $Name = preg_replace('/[0-9][0-9][a-z]$/i', '', $Name);
275 $Name = htmlspecialchars($Name); 261 $Name = htmlspecialchars($Name);
276   262  
277 // Sestav odkaz 263 // Sestav odkaz
278 $Link = '<a href="http://'; 264 $Link = '<a href="http://';
279 $Link .= preg_replace('/^.+?\./', WIKISERVER.'.', $_SERVER['SERVER_NAME']); 265 $Link .= preg_replace('/^.+?\./', WIKISERVER.'.', $_SERVER['SERVER_NAME']);
280 $Link .= '/doku.php?id='.$Lang.':'.$Name; 266 $Link .= '/doku.php?id='.$Lang.':'.$Name;
281 $Link .= '"'; 267 $Link .= '"';
282 268
283 if ($Title!='') 269 if ($Title!='')
284 $Link .= "\n".' title="'.htmlspecialchars($Title).'"'; 270 $Link .= "\n".' title="'.htmlspecialchars($Title).'"';
285 $Link .= '>Wiki</a>'; 271 $Link .= '>Wiki</a>';
286   272  
287 return $Link; 273 return $Link;
288 } 274 }
289   275  
290   276  
291 function CreatePicturesLink($Path, $Glue, $Alt='', $NoPicture) 277 function CreatePicturesLink($Path, $Glue, $Alt='', $NoPicture)
292 // Najde na uvedené cestě obrázky, které mají v názvu Small 278 // Najde na uvedené cestě obrázky, které mají v názvu Small
293 // a sestaví na ně odkaz. Pokud není žádný obrázek, vytvoří odkaz na obrázek 279 // a sestaví na ně odkaz. Pokud není žádný obrázek, vytvoří odkaz na obrázek
294 // v parametru $NoPicture. 280 // v parametru $NoPicture.
295 { 281 {
296 // Najdi všechny soubory 282 // Najdi všechny soubory
297 $Files = array(); 283 $Files = array();
298 foreach (glob($_SERVER['DOCUMENT_ROOT'].$Path.'/*') as $File) 284 foreach (glob($_SERVER['DOCUMENT_ROOT'].$Path.'/*') as $File)
299 { 285 {
300 if (is_file($File)) 286 if (is_file($File))
301 { 287 {
302 // Jen jméno souboru 288 // Jen jméno souboru
303 $File = substr($File, 1+strlen($_SERVER['DOCUMENT_ROOT'].$Path)); 289 $File = substr($File, 1+strlen($_SERVER['DOCUMENT_ROOT'].$Path));
304 290
305 // TOHLE ASI RADĚJI PŘEZ REGEX 291 // TOHLE ASI RADĚJI PŘEZ REGEX
306   292  
307 // Odděl příponu 293 // Odděl příponu
308 $DotPos = strrpos($File, '.'); 294 $DotPos = strrpos($File, '.');
309 $Extension = substr($File, $DotPos+1); 295 $Extension = substr($File, $DotPos+1);
310 $BaseName = substr($File, 0, $DotPos); 296 $BaseName = substr($File, 0, $DotPos);
311 297
312 // Vezmi obrázky, které mají ve jméně Small 298 // Vezmi obrázky, které mají ve jméně Small
313 if( (strtolower($Extension)=='jpg' or strtolower($Extension)=='png') 299 if( (strtolower($Extension)=='jpg' or strtolower($Extension)=='png')
314 and strpos($BaseName, 'Small')>0 ) 300 and strpos($BaseName, 'Small')>0 )
315 { 301 {
316 $Files[] = $Path.'/'.$File; 302 $Files[] = $Path.'/'.$File;
317 } 303 }
318 } 304 }
319 } 305 }
320   306  
321 // V adresáři nejsou žádné vhodné obrázky 307 // V adresáři nejsou žádné vhodné obrázky
322 if (count($Files) == 0) 308 if (count($Files) == 0)
323 { 309 {
324 // Není na co udělat odkaz 310 // Není na co udělat odkaz
325 if ($NoPicture=='') 311 if ($NoPicture=='')
326 return ''; 312 return '';
327 // Přidej odkaz nikam 313 // Přidej odkaz nikam
328 $Files[] = $NoPicture; 314 $Files[] = $NoPicture;
329 } 315 }
330   316  
331 // Udělej odkazy na pole souborů 317 // Udělej odkazy na pole souborů
332 $Alt = htmlspecialchars($Alt); 318 $Alt = htmlspecialchars($Alt);
333 $Line = ''; 319 $Line = '';
334 $FirstGlue = TRUE; 320 $FirstGlue = TRUE;
335 foreach($Files as $File) 321 foreach($Files as $File)
336 { 322 {
337 $File = htmlspecialchars($File); 323 $File = htmlspecialchars($File);
338   324  
339 if(!$FirstGlue) 325 if(!$FirstGlue)
340 $Line .= ' '.$Glue."\n"; 326 $Line .= ' '.$Glue."\n";
341 $FirstGlue = FALSE; 327 $FirstGlue = FALSE;
342   328  
343 $Line .= '<a href="'.$File.'">'."\n"; 329 $Line .= '<a href="'.$File.'">'."\n";
344 $Line .= '<img src="'.$File.'"'."\n"; 330 $Line .= '<img src="'.$File.'"'."\n";
345 $Line .= ' width="150"'."\n"; 331 $Line .= ' width="150"'."\n";
346 $Line .= ' alt="'.$Alt.'">'."\n"; 332 $Line .= ' alt="'.$Alt.'">'."\n";
347 $Line .= '</a>'."\n"; 333 $Line .= '</a>'."\n";
348 } 334 }
349   335  
350 return $Line; 336 return $Line;
351 } 337 }
352   338  
353   339  
354 ?> 340 ?>