4 Plugin URI: http://f-droid.org/repository
5 Description: An FDroid repository browser
6 Author: Ciaran Gultnieks
8 Author URI: http://ciarang.com
11 0.01 - 2010-12-04: Initial development version
15 include('android-permissions.php');
20 // Our text domain, for internationalisation
21 private $textdom='wp-fdroid';
27 // Add filters etc here!
28 add_shortcode('fdroidrepo',array($this, 'do_shortcode'));
29 add_filter('query_vars',array($this, 'queryvars'));
31 $this->site_path=getenv('DOCUMENT_ROOT');
32 wp_register_sidebar_widget('fdroid_latest', 'FDroid Latest', 'widget_fdroidlatest');
36 // Register additional query variables. (Handler for the 'query_vars' filter)
37 function queryvars($qvars) {
39 $qvars[]='fdcategory';
47 // Lazy initialise. All non-trivial members should call this before doing anything else.
50 load_plugin_textdomain($this->textdom, PLUGINDIR.'/'.dirname(plugin_basename(__FILE__)), dirname(plugin_basename(__FILE__)));
56 // Gets a required query parameter by name.
57 function getrequiredparam($name) {
59 if(!isset($wp_query->query_vars[$name]))
60 wp_die("Missing parameter ".$name,"Error");
61 return $wp_query->query_vars[$name];
64 // Handler for the 'fdroidrepo' shortcode.
65 // $attribs - shortcode attributes
66 // $content - optional content enclosed between the starting and
68 // Returns the generated content.
69 function do_shortcode($attribs,$content=null) {
70 global $wp_query,$wp_rewrite;
73 // Init local query vars
74 foreach($this->queryvars(array()) as $qv) {
75 if(array_key_exists($qv,$wp_query->query_vars)) {
76 $query_vars[$qv] = $wp_query->query_vars[$qv];
78 $query_vars[$qv] = null;
82 // Sanity check and standardise all query variables...
83 if(!isset($query_vars['fdpage']) || !is_numeric($query_vars['fdpage']) || $query_vars['fdpage'] <= 0) {
84 $query_vars['fdpage'] = 1;
86 $query_vars['fdpage'] = strval(intval($query_vars['fdpage']));
88 if(isset($query_vars['fdstyle']) && ($query_vars['fdstyle'] != 'list' && $query_vars['fdstyle'] != 'grid')) {
89 $query_vars['fdstyle'] = 'list';
91 if(isset($query_vars['fdcategory'])) {
92 if($query_vars['fdcategory'] == 'All categories') {
93 unset($query_vars['fdcategory']);
95 $query_vars['fdcategory'] = sanitize_text_field($query_vars['fdcategory']);
98 if(isset($query_vars['fdfilter'])) {
99 $query_vars['fdfilter'] = sanitize_text_field($query_vars['fdfilter']);
101 if(isset($attribs['search'])) {
102 $query_vars['fdfilter'] = '';
105 if(isset($query_vars['fdid'])) {
106 $query_vars['fdid'] = sanitize_text_field($query_vars['fdid']);
111 if($query_vars['fdid']!==null) {
112 $out.=$this->get_app($query_vars);
114 $out.='<form name="searchform" action="" method="get">';
115 $out.='<p><input name="fdfilter" type="text" value="'.$query_vars['fdfilter'].'" size="30"> ';
116 $out.='<input type="hidden" name="fdpage" value="1">';
117 $out.='<input type="submit" value="Search"></p>';
118 $out.=$this->makeformdata($query_vars);
119 $out.='</form>'."\n";
121 $out.=$this->get_apps($query_vars);
128 // Get a URL for a full description of a license, as given by one of our
129 // pre-defined license abbreviations. This is a temporary function, as this
130 // needs to be data-driven so the same information can be used by the client,
131 // the web site and the documentation.
132 function getlicenseurl($license) {
135 return 'http://www.gnu.org/licenses/license-list.html#X11License';
137 return 'http://www.gnu.org/licenses/license-list.html#ModifiedBSD';
139 return 'http://www.gnu.org/licenses/license-list.html#OriginalBSD';
142 return 'http://www.gnu.org/licenses/license-list.html#GNUGPL';
145 return 'http://www.gnu.org/licenses/license-list.html#GPLv2';
147 return 'http://www.gnu.org/licenses/license-list.html#LGPL';
149 return 'http://www.gnu.org/licenses/license-list.html#apache2';
155 function get_app($query_vars) {
156 global $permissions_data;
157 $permissions_object = new AndroidPermissions($this->site_path.'/wp-content/plugins/wp-fdroid/AndroidManifest.xml',
158 $this->site_path.'/wp-content/plugins/wp-fdroid/strings.xml',
159 sys_get_temp_dir().'/android-permissions.cache');
160 $permissions_data = $permissions_object->get_permissions_array();
163 $xml = simplexml_load_file($this->site_path.'/repo/index.xml');
164 foreach($xml->children() as $app) {
166 $attrs=$app->attributes();
167 if($attrs['id']==$query_vars['fdid']) {
169 foreach($app->children() as $el) {
170 switch($el->getName()) {
209 foreach($el->children() as $pel) {
210 switch($pel->getName()) {
212 $thisapk['version']=$pel;
215 $thisapk['vercode']=$pel;
218 $thisapk['apkname']=$pel;
221 $thisapk['srcname']=$pel;
224 $thisapk['hash']=$pel;
227 $thisapk['size']=$pel;
230 $thisapk['sdkver']=$pel;
233 $thisapk['permissions']=$pel;
242 // Generate app diff data
243 foreach(array_reverse($apks, true) as $key=>$apk) {
244 if(isset($previous)) {
246 $apks[$key]['diff']['size'] = $apk['size']-$previous['size'];
250 $permissions = explode(',',$apk['permissions']);
251 $permissionsPrevious = isset($previous['permissions'])?explode(',',$previous['permissions']):array();
252 $apks[$key]['diff']['permissions']['added'] = array_diff($permissions, $permissionsPrevious);
253 $apks[$key]['diff']['permissions']['removed'] = array_diff($permissionsPrevious, $permissions);
258 // Output app information
259 $out='<div id="appheader">';
260 $out.='<div style="float:left;padding-right:10px;"><img src="' . site_url() . '/repo/icons/'.$icon.'" width=48></div>';
261 $out.='<p><span style="font-size:20px">'.$name."</span>";
262 $out.="<br>".$summary."</p>";
265 $out.=str_replace('href="fdroid.app:', 'href="/repository/browse/?fdid=', $desc);
267 if(isset($antifeatures)) {
268 $antifeaturesArray = explode(',',$antifeatures);
269 foreach($antifeaturesArray as $antifeature) {
270 $antifeatureDescription = $this->get_antifeature_description($antifeature);
271 $out.='<p style="border:3px solid #CC0000;background-color:#FFDDDD;padding:5px;"><strong>'.$antifeatureDescription['name'].'</strong><br />';
272 $out.=$antifeatureDescription['description'].' <a href="/wiki/page/Antifeature:'.$antifeature.'">more...</a></p>';
277 $licenseurl=$this->getlicenseurl($license);
278 $out.="<b>License:</b> ";
280 $out.='<a href="'.$licenseurl.'" target="_blank">';
285 if(isset($requirements)) {
286 $out.='<br /><b>Additional requirements:</b> '.$requirements;
292 $out.='<b>Website:</b> <a href="'.$web.'">'.$web.'</a><br />';
293 if(strlen($issues)>0)
294 $out.='<b>Issue Tracker:</b> <a href="'.$issues.'">'.$issues.'</a><br />';
295 if(strlen($source)>0)
296 $out.='<b>Source Code:</b> <a href="'.$source.'">'.$source.'</a><br />';
297 if($donate && strlen($donate)>0)
298 $out.='<b>Donate:</b> <a href="'.$donate.'">'.$donate.'</a><br />';
301 $out.="<p>For full details and additional technical information, see ";
302 $out.="<a href=\"/wiki/page/".$query_vars['fdid']."\">this application's page</a> on the F-Droid wiki.</p>";
304 $out.='<script type="text/javascript">';
305 $out.='function showHidePermissions(id) {';
306 $out.=' if(document.getElementById(id).style.display==\'none\')';
307 $out.=' document.getElementById(id).style.display=\'block\';';
309 $out.=' document.getElementById(id).style.display=\'none\';';
310 $out.=' return false;';
314 $out.="<h3>Packages</h3>";
316 $out.='<div style="float:right; margin-left:10px;"><a id="downloadbutton" href="https://f-droid.org/FDroid.apk"><span>Download F-Droid</span></a></div>';
317 $out.="<p>Although APK downloads are available below to give ";
318 $out.="you the choice, you should be aware that by installing that way you ";
319 $out.="will not receive update notifications, and it's a less secure way ";
320 $out.="to download. ";
321 $out.="We recommend that you install the F-Droid client and use that.</p>";
324 foreach($apks as $apk) {
325 $first = $i+1==count($apks);
326 $out.="<p><b>Version ".$apk['version']."</b><br />";
327 $out.="Added on ".$apk['added']."<br />";
329 // Is this source or binary?
330 $srcbuild = isset($apk['srcname']) && file_exists($this->site_path.'/repo/'.$apk['srcname']);
332 $out.="<p>This version is built and signed by ";
334 $out.="F-Droid, and guaranteed to correspond to the source tarball below.</p>";
336 $out.="the original developer.</p>";
338 $out.='<a href="https://f-droid.org/repo/'.$apk['apkname'].'">download apk</a> ';
339 $out.=$this->human_readable_size($apk['size']);
340 $diffSize = $apk['diff']['size'];
341 if(abs($diffSize) > 500) {
342 $out.=' <span style="color:#AAAAAA;">(';
343 $out.=$diffSize>0?'+':'';
344 $out.=$this->human_readable_size($diffSize, 1).')</span>';
347 $out.='<br /><a href="https://f-droid.org/repo/'.$apk['srcname'].'">source tarball</a> ';
348 $out.=$this->human_readable_size(filesize($this->site_path.'/repo/'.$apk['srcname']));
351 if(isset($apk['permissions'])) {
352 // Permissions diff link
353 if($first == false) {
354 $permissionsAddedCount = count($apk['diff']['permissions']['added']);
355 $permissionsRemovedCount = count($apk['diff']['permissions']['removed']);
356 $divIdDiff='permissionsDiff'.$i;
357 if($permissionsAddedCount || $permissionsRemovedCount) {
358 $out.='<br /><a href="javascript:void(0);" onClick="showHidePermissions(\''.$divIdDiff.'\');">permissions diff</a>';
359 $out.=' <span style="color:#AAAAAA;">(';
360 if($permissionsAddedCount)
361 $out.='+'.$permissionsAddedCount;
362 if($permissionsAddedCount && $permissionsRemovedCount)
364 if($permissionsRemovedCount)
365 $out.='-'.$permissionsRemovedCount;
370 $out.='<br /><span style="color:#999999;">no permission changes</span>';
374 // Permissions list link
375 $permissionsListString = $this->get_permission_list_string(explode(',',$apk['permissions']), $permissions_data, $summary);
377 $divStyleDisplay='block';
379 $divStyleDisplay='none';
380 $divId='permissions'.$i;
381 $out.='<br /><a href="javascript:void(0);" onClick="showHidePermissions(\''.$divId.'\');">view permissions</a>';
382 $out.=' <span style="color:#AAAAAA;">['.$summary.']</span>';
386 $out.='<div style="display:'.$divStyleDisplay.';" id="'.$divId.'">';
387 $out.=$permissionsListString;
392 $out.='<div style="display:'.$divStyleDisplay.';" id="'.$divIdDiff.'">';
393 $permissionsRemoved = $apk['diff']['permissions']['removed'];
394 usort($permissionsRemoved, "permissions_cmp");
397 if($permissionsAddedCount) {
398 $out.='<h5>ADDED</h5><br />';
399 $out.=$this->get_permission_list_string($apk['diff']['permissions']['added'], $permissions_data, $summary);
402 // Removed permissions
403 if($permissionsRemovedCount) {
404 $out.='<h5>REMOVED</h5><br />';
405 $out.=$this->get_permission_list_string($apk['diff']['permissions']['removed'], $permissions_data, $summary);
412 $out.='<br /><span style="color:#999999;">no extra permissions needed</span><br />';
419 $out.='<hr><p><a href="'.makelink($query_vars,array('fdid'=>null)).'">Index</a></p>';
424 return "<p>Application not found</p>";
427 private function get_permission_list_string($permissions, $permissions_data, &$summary) {
429 usort($permissions, "permissions_cmp");
430 $permission_group_last = '';
431 foreach($permissions as $permission) {
432 $permission_group = $permissions_data['permission'][$permission]['permissionGroup'];
433 if($permission_group != $permission_group_last) {
434 $permission_group_label = $permissions_data['permission-group'][$permission_group]['label'];
435 if($permission_group_label=='') $permission_group_label = 'Extra/Custom';
436 $out.='<strong>'.strtoupper($permission_group_label).'</strong><br/>';
437 $permission_group_last = $permission_group;
440 $out.=$this->get_permission_protection_level_icon($permissions_data['permission'][$permission]['protectionLevel']).' ';
441 $out.='<strong>'.$permissions_data['permission'][$permission]['label'].'</strong> [<code>'.$permission.'</code>]<br/>';
442 if($permissions_data['permission'][$permission]['description']) $out.=$permissions_data['permission'][$permission]['description'].'<br/>';
443 //$out.=$permissions_data['permission'][$permission]['comment'].'<br/>';
446 if(!isset($summaryCount[$permissions_data['permission'][$permission]['protectionLevel']]))
447 $summaryCount[$permissions_data['permission'][$permission]['protectionLevel']] = 0;
448 $summaryCount[$permissions_data['permission'][$permission]['protectionLevel']]++;
452 if(isset($summaryCount)) {
453 foreach($summaryCount as $protectionLevel => $count) {
454 $summary .= $this->get_permission_protection_level_icon($protectionLevel, 'regular').' '.$count;
458 $summary = substr($summary,0,-2);
463 private function get_permission_protection_level_icon($protection_level, $size='adjusted') {
465 if($protection_level=='dangerous') {
466 $iconString .= '<span style="color:#DD9900;';
467 if($size=='adjusted')
468 $iconString .= 'font-size:150%;';
469 $iconString .= '">⚠</span>'; // WARNING SIGN
471 elseif($protection_level=='normal') {
472 $iconString .= '<span style="color:#6666FF;';
473 if($size=='adjusted')
474 $iconString .= 'font-size:110%;';
475 $iconString .= '">ⓘ</span>'; // CIRCLED LATIN SMALL LETTER I
477 elseif($protection_level=='signature') {
478 $iconString .= '<span style="color:#33AAAA;';
479 if($size=='adjusted')
480 $iconString .= 'font-size:140%;';
481 $iconString .= '">✽</span>'; // HEAVY TEARDROP-SPOKED ASTERISK
483 elseif($protection_level=='signatureOrSystem') {
484 $iconString .= '<span style="color:#DD66DD;';
485 if($size=='adjusted')
486 $iconString .= 'font-size:140%;';
487 $iconString .= '">⚛</span>'; // ATOM SYMBOL
490 $iconString .= '<span style="color:#33AA33';
491 if($size=='adjusted')
492 $iconString .= ';font-size:130%;';
493 $iconString .= '">⚙</span>'; // GEAR
499 private function human_readable_size($size, $minDiv=0) {
500 $si_prefix = array('bytes','kB','MB');
503 for($i=0;(abs($size) > $div && $i < count($si_prefix)) || $i<$minDiv;$i++) {
507 return round($size,max(0,$i-1)).' '.$si_prefix[$i];
510 private function get_antifeature_description($antifeature) {
511 // Anti feature names and descriptions
512 $antifeatureDescription['Ads']['name'] = 'Advertising';
513 $antifeatureDescription['Ads']['description'] = 'This application contains advertising.';
514 $antifeatureDescription['Tracking']['name'] = 'Tracks You';
515 $antifeatureDescription['Tracking']['description'] = 'This application tracks and reports your activity to somewhere.';
516 $antifeatureDescription['NonFreeNet']['name'] = 'Non-Free Network Services';
517 $antifeatureDescription['NonFreeNet']['description'] = 'This application promotes a non-Free network service.';
518 $antifeatureDescription['NonFreeAdd']['name'] = 'Non-Free Addons';
519 $antifeatureDescription['NonFreeAdd']['description'] = 'This application promotes non-Free add-ons.';
520 $antifeatureDescription['NonFreeDep']['name'] = 'Non-Free Dependencies';
521 $antifeatureDescription['NonFreeDep']['description'] = 'This application depends on another non-Free application.';
522 $antifeatureDescription['UpstreamNonFree']['name'] = 'Upstream Non-Free';
523 $antifeatureDescription['UpstreamNonFree']['description'] = 'The upstream source code is non-free.';
525 if(isset($antifeatureDescription[$antifeature])) {
526 return $antifeatureDescription[$antifeature];
528 return array('name'=>$antifeature);
532 function get_apps($query_vars) {
534 $xml = simplexml_load_file($this->site_path."/repo/index.xml");
535 $matches = $this->show_apps($xml,$query_vars,$numpages);
539 if(($query_vars['fdfilter']===null || $query_vars['fdfilter']!='') && $numpages>0)
541 $out.='<div style="float:left;">';
542 if($query_vars['fdfilter']===null) {
544 $categories = array('All categories');
545 $handle = fopen(getenv('DOCUMENT_ROOT').'/repo/categories.txt', 'r');
547 while (($buffer = fgets($handle, 4096)) !== false) {
548 $categories[] = rtrim($buffer);
553 $out.='<form name="categoryform" action="" method="get">';
554 $out.=$this->makeformdata($query_vars);
556 $out.='<select name="fdcategory" style="color:#333333;" onChange="document.categoryform.submit();">';
557 foreach($categories as $category) {
559 if(isset($query_vars['fdcategory']) && $category==$query_vars['fdcategory'])
561 $out.='>'.$category.'</option>';
565 $out.='</form>'."\n";
568 $out.='Applications matching "'.$query_vars['fdfilter'].'"';
572 $out.='<div style="float:right;">';
573 $out.='<a href="'.makelink($query_vars, array('fdstyle'=>'list')).'">List</a> | ';
574 $out.='<a href="'.makelink($query_vars, array('fdstyle'=>'grid')).'">Grid</a>';
577 $out.='<br break="all"/>';
585 $out.='<div style="width:20%; float:left; text-align:left;">';
586 $out.=' Page '.$query_vars['fdpage'].' of '.$numpages.' ';
589 $out.='<div style="width:60%; float:left; text-align:center;">';
591 for($i=1;$i<=$numpages;$i++) {
592 if($i == $query_vars['fdpage']) {
593 $out.='<b>'.$i.'</b>';
595 $out.='<a href="'.makelink($query_vars, array('fdpage'=>$i)).'">';
605 $out.='<div style="width:20%; float:left; text-align:right;">';
606 if($query_vars['fdpage']!=$numpages) {
607 $out.='<a href="'.makelink($query_vars, array('fdpage'=>($query_vars['fdpage']+1))).'">next></a> ';
612 } else if($query_vars['fdfilter']!='') {
613 $out.='<p>No matches</p>';
620 function makeformdata($query_vars) {
624 $out.='<input type="hidden" name="page_id" value="'.(int)get_query_var('page_id').'">';
625 foreach($query_vars as $name => $value) {
626 if($value !== null && $name != 'fdfilter' && !($name == 'fdpage' && (int)$value ==1))
627 $out.='<input type="hidden" name="'.$name.'" value="'.sanitize_text_field($value).'">';
634 function show_apps($xml,$query_vars,&$numpages) {
640 if($query_vars['fdstyle']=='grid') {
641 $outputter = new FDOutGrid();
643 $outputter = new FDOutList();
648 $out.=$outputter->outputStart();
650 foreach($xml->children() as $app) {
652 if($app->getName() == 'repo') continue;
653 $appinfo['attrs']=$app->attributes();
654 $appinfo['id']=$appinfo['attrs']['id'];
655 foreach($app->children() as $el) {
656 switch($el->getName()) {
658 $appinfo['name']=$el;
661 $appinfo['icon']=$el;
664 $appinfo['summary']=$el;
667 $appinfo['description']=$el;
670 $appinfo['license']=$el;
673 $appinfo['category']=$el;
678 if(($query_vars['fdfilter']===null || $query_vars['fdfilter']!='' && (stristr($appinfo['name'],$query_vars['fdfilter']) || stristr($appinfo['summary'],$query_vars['fdfilter']) || stristr($appinfo['description'],$query_vars['fdfilter']))) && (!isset($query_vars['fdcategory']) || $query_vars['fdcategory'] && $query_vars['fdcategory']==$appinfo['category'])) {
679 if($skipped<($query_vars['fdpage']-1)*$outputter->perpage) {
681 } else if($got<$outputter->perpage) {
682 $out.=$outputter->outputEntry($query_vars, $appinfo);
690 $out.=$outputter->outputEnd();
692 $numpages = ceil((float)$total/$outputter->perpage);
698 // Class to output app entries in a detailed list format
703 function FDOutList() {
706 function outputStart() {
710 function outputEntry($query_vars, $appinfo) {
712 $out.='<hr style="clear:both;" />'."\n";
713 $out.='<a href="'.makelink($query_vars, array('fdid'=>$appinfo['id'])).'">';
714 $out.='<div id="appheader">';
716 $out.='<div style="float:left;padding-right:10px;"><img src="' . site_url() . '/repo/icons/'.$appinfo['icon'].'" style="width:48px;border:none;"></div>';
718 $out.='<div style="float:right;">';
719 $out.='<p>Details...</p>';
722 $out.='<p style="color:#000000;"><span style="font-size:20px;">'.$appinfo['name']."</span>";
723 $out.="<br>".$appinfo['summary']."</p>\n";
731 function outputEnd() {
736 // Class to output app entries in a compact grid format
743 function FDOutGrid() {
746 function outputStart() {
747 return "\n".'<table border="0" width="100%"><tr>'."\n";
750 function outputEntry($query_vars, $appinfo) {
751 $link=makelink($query_vars, array('fdid'=>$appinfo['id']));
755 if($this->itemCount%4 == 0 && $this->itemCount > 0)
757 $out.='</tr><tr>'."\n";
760 $out.='<td align="center" valign="top" style="background-color:#F8F8F8;">';
762 $out.='<div id="appheader" style="text-align:center;width:110px;">';
764 $out.='<a href="'.$link.'" style="border-bottom-style:none;">';
765 $out.='<img src="' . site_url() . '/repo/icons/'.$appinfo['icon'].'" style="width:48px;border-width:0;padding-top:5px;padding-bottom:5px;"><br/>';
766 $out.=$appinfo['name'].'<br/>';
777 function outputEnd() {
778 return '</tr></table>'."\n";
782 function permissions_cmp($a, $b) {
783 global $permissions_data;
785 $aProtectionLevel = $permissions_data['permission'][$a]['protectionLevel'];
786 $bProtectionLevel = $permissions_data['permission'][$b]['protectionLevel'];
788 if($aProtectionLevel != $bProtectionLevel) {
789 if(strlen($aProtectionLevel)==0) return 1;
790 if(strlen($bProtectionLevel)==0) return -1;
792 return strcmp($aProtectionLevel, $bProtectionLevel);
795 $aGroup = $permissions_data['permission'][$a]['permissionGroup'];
796 $bGroup = $permissions_data['permission'][$b]['permissionGroup'];
798 if($aGroup != $bGroup) {
799 return strcmp($aGroup, $bGroup);
802 return strcmp($a, $b);
805 // Make a link to this page, with the current query vars attached and desired params added/modified
806 function makelink($query_vars, $params=array()) {
807 $link=get_permalink();
809 $p = array_merge($query_vars, $params);
811 // Page 1 is the default, don't clutter urls with it...
812 if($p['fdpage'] == 1)
814 // Likewise for list style...
815 if($p['fdstyle'] == 'list')
816 unset($p['fdstyle']);
821 if(strpos($link,'?')===false)
828 // Return the key value pairs in http-get-parameter format as a string
829 function linkify($vars) {
831 foreach($vars as $k => $v) {
832 if($k!==null && $v!==null && $v!='')
833 $retvar .= $k.'='.$v.'&';
835 return substr($retvar,0,-1);
838 function widget_fdroidlatest($args) {
841 echo $before_title . 'Latest Apps' . $after_title;
843 $handle = fopen(getenv('DOCUMENT_ROOT').'/repo/latestapps.dat', 'r');
845 while (($buffer = fgets($handle, 4096)) !== false) {
846 $app = explode("\t", $buffer);
847 echo '<a href="/repository/browse/?fdid='.$app[0].'">';
848 if(isset($app[2]) && trim($app[2])) {
849 echo '<img src="' . site_url() . '/repo/icons/'.$app[2].'" style="width:32px;border:none;float:right;" />';
851 echo $app[1].'<br />';
852 if(isset($app[3]) && trim($app[3])) {
853 echo '<span style="color:#BBBBBB;">'.$app[3].'</span>';
855 echo '</a><br style="clear:both;" />';
863 $wp_fdroid = new FDroid();