diff --git a/mod/hcard.php b/mod/hcard.php
deleted file mode 100644
index 013619bcb..000000000
--- a/mod/hcard.php
+++ /dev/null
@@ -1,61 +0,0 @@
-argc > 1) {
- $which = $a->argv[1];
- } else {
- throw new \Friendica\Network\HTTPException\NotFoundException(L10n::t('No profile'));
- }
-
- $profile = 0;
- if ((local_user()) && ($a->argc > 2) && ($a->argv[2] === 'view')) {
- $which = $a->user['nickname'];
- $profile = $a->argv[1];
- }
-
- Profile::load($a, $which, $profile);
-
- if (!empty($a->profile['page-flags']) && ($a->profile['page-flags'] == User::PAGE_FLAGS_COMMUNITY)) {
- $a->page['htmlhead'] .= '';
- }
- if (!empty($a->profile['openidserver'])) {
- $a->page['htmlhead'] .= '' . "\r\n";
- }
- if (!empty($a->profile['openid'])) {
- $delegate = ((strstr($a->profile['openid'], '://')) ? $a->profile['openid'] : 'http://' . $a->profile['openid']);
- $a->page['htmlhead'] .= '' . "\r\n";
- }
-
- if (!$blocked) {
- $keywords = $a->profile['pub_keywords'] ?? '';
- $keywords = str_replace([',',' ',',,'], [' ',',',','], $keywords);
- if (strlen($keywords)) {
- $a->page['htmlhead'] .= '' . "\r\n";
- }
- }
-
- $a->page['htmlhead'] .= '' . "\r\n";
- $a->page['htmlhead'] .= '' . "\r\n";
- $uri = urlencode('acct:' . $a->profile['nickname'] . '@' . $a->getHostName() . (($a->getURLPath()) ? '/' . $a->getURLPath() : ''));
- $a->page['htmlhead'] .= '' . "\r\n";
- header('Link: <' . System::baseUrl() . '/xrd/?uri=' . $uri . '>; rel="lrdd"; type="application/xrd+xml"', false);
-
- $dfrn_pages = ['request', 'confirm', 'notify', 'poll'];
- foreach ($dfrn_pages as $dfrn) {
- $a->page['htmlhead'] .= "\r\n";
- }
-}
diff --git a/src/Module/HoverCard.php b/src/Module/HoverCard.php
new file mode 100644
index 000000000..210680922
--- /dev/null
+++ b/src/Module/HoverCard.php
@@ -0,0 +1,80 @@
+user['nickname'];
+ $profile = $parameters['profile'];
+ } elseif (empty($parameters['action'])) {
+ // Show the profile hovercard
+ $nickname = $parameters['profile'];
+ $profile = 0;
+ } else {
+ /** @var L10n $l10n */
+ $l10n = self::getClass(L10n::class);
+ throw new NotFoundException($l10n->t('No profile'));
+ }
+
+ Profile::load($a, $nickname, $profile);
+
+ /** @var Page $page */
+ $page = self::getClass(Page::class);
+
+ if (!empty($a->profile['page-flags']) && ($a->profile['page-flags'] == User::PAGE_FLAGS_COMMUNITY)) {
+ $page['htmlhead'] .= '';
+ }
+ if (!empty($a->profile['openidserver'])) {
+ $page['htmlhead'] .= '' . "\r\n";
+ }
+ if (!empty($a->profile['openid'])) {
+ $delegate = ((strstr($a->profile['openid'], '://')) ? $a->profile['openid'] : 'http://' . $a->profile['openid']);
+ $page['htmlhead'] .= '' . "\r\n";
+ }
+
+ /** @var Configuration $config */
+ $config = self::getClass(Configuration::class);
+ // check if blocked
+ if ($config->get('system', 'block_public') && !Session::isAuthenticated()) {
+ $keywords = $a->profile['pub_keywords'] ?? '';
+ $keywords = str_replace([',', ' ', ',,'], [' ', ',', ','], $keywords);
+ if (strlen($keywords)) {
+ $page['htmlhead'] .= '' . "\r\n";
+ }
+ }
+
+ /** @var BaseURL $baseUrl */
+ $baseUrl = self::getClass(BaseURL::class);
+
+ $uri = urlencode('acct:' . $a->profile['nickname'] . '@' . $baseUrl->getHostname() . ($baseUrl->getUrlPath() ? '/' . $baseUrl->getUrlPath() : ''));
+
+ $page['htmlhead'] .= '' . "\r\n";
+ $page['htmlhead'] .= '' . "\r\n";
+ $page['htmlhead'] .= '' . "\r\n";
+ header('Link: <' . $baseUrl->get() . '/xrd/?uri=' . $uri . '>; rel="lrdd"; type="application/xrd+xml"', false);
+
+ foreach (['request', 'confirm', 'notify', 'poll'] as $dfrn) {
+ $page['htmlhead'] .= "get() . "/dfrn_{$dfrn}/{$nickname}\" />\r\n";
+ }
+ }
+}
diff --git a/static/routes.config.php b/static/routes.config.php
index 9dced3310..115b8f5f6 100644
--- a/static/routes.config.php
+++ b/static/routes.config.php
@@ -139,11 +139,12 @@ return [
'/{group:\d+}/add/{contact:\d+}' => [Module\Group::class, [R::GET, R::POST]],
'/{group:\d+}/remove/{contact:\d+}' => [Module\Group::class, [R::GET, R::POST]],
],
- '/hashtag' => [Module\Hashtag::class, [R::GET]],
- '/home' => [Module\Home::class, [R::GET]],
- '/help[/{doc:.+}]' => [Module\Help::class, [R::GET]],
- '/inbox[/{nickname}]' => [Module\Inbox::class, [R::GET, R::POST]],
- '/invite' => [Module\Invite::class, [R::GET, R::POST]],
+ '/hashtag' => [Module\Hashtag::class, [R::GET]],
+ '/help[/{doc:.+}]' => [Module\Help::class, [R::GET]],
+ '/home' => [Module\Home::class, [R::GET]],
+ '/hcard/{profile}[/{action}]' => [Module\HoverCard::class, [R::GET]],
+ '/inbox[/{nickname}]' => [Module\Inbox::class, [R::GET, R::POST]],
+ '/invite' => [Module\Invite::class, [R::GET, R::POST]],
'/install' => [
'[/]' => [Module\Install::class, [R::GET, R::POST]],