diff --git a/lib/pleroma/web/static_fe/static_fe_controller.ex b/lib/pleroma/web/static_fe/static_fe_controller.ex
new file mode 100644
index 000000000..067af9816
--- /dev/null
+++ b/lib/pleroma/web/static_fe/static_fe_controller.ex
@@ -0,0 +1,42 @@
+# Pleroma: A lightweight social networking server
+# Copyright © 2017-2019 Pleroma Authors
+# SPDX-License-Identifier: AGPL-3.0-only
+
+defmodule Pleroma.Web.StaticFE.StaticFEController do
+ use Pleroma.Web, :controller
+
+ alias Pleroma.Repo
+ alias Pleroma.Activity
+ alias Pleroma.Object
+ alias Pleroma.User
+ alias Pleroma.Web.ActivityPub.Visibility
+
+ require Logger
+
+ def show_notice(conn, %{"notice_id" => notice_id}) do
+ with %Activity{} = activity <- Repo.get(Activity, notice_id),
+ true <- Visibility.is_public?(activity),
+ %User{} = user <- User.get_or_fetch(activity.data["actor"]),
+ %Object{} = object <- Object.normalize(activity.data["object"]) do
+ conn
+ |> put_layout(:static_fe)
+ |> put_status(200)
+ |> put_view(Pleroma.Web.StaticFE.StaticFEView)
+ |> render("notice.html", %{notice: activity, object: object, user: user})
+ else
+ _ ->
+ conn
+ |> put_status(404)
+ |> text("Not found")
+ end
+ end
+
+ def show(%{path_info: ["notice", notice_id]} = conn, _params), do: show_notice(conn, %{"notice_id" => notice_id})
+
+ # Fallback for unhandled types
+ def show(conn, _params) do
+ conn
+ |> put_status(404)
+ |> text("Not found")
+ end
+end
diff --git a/lib/pleroma/web/static_fe/static_fe_view.ex b/lib/pleroma/web/static_fe/static_fe_view.ex
new file mode 100644
index 000000000..7f58e1b2d
--- /dev/null
+++ b/lib/pleroma/web/static_fe/static_fe_view.ex
@@ -0,0 +1,19 @@
+# Pleroma: A lightweight social networking server
+# Copyright © 2017-2019 Pleroma Authors
+# SPDX-License-Identifier: AGPL-3.0-only
+
+defmodule Pleroma.Web.StaticFE.StaticFEView do
+ use Pleroma.Web, :view
+
+ alias Pleroma.User
+ alias Pleroma.Web.MediaProxy
+ alias Pleroma.Formatter
+
+ def emoji_for_user(%User{} = user) do
+ (user.info.source_data["tag"] || [])
+ |> Enum.filter(fn %{"type" => t} -> t == "Emoji" end)
+ |> Enum.map(fn %{"icon" => %{"url" => url}, "name" => name} ->
+ {String.trim(name, ":"), url}
+ end)
+ end
+end
diff --git a/lib/pleroma/web/templates/layout/static_fe.html.eex b/lib/pleroma/web/templates/layout/static_fe.html.eex
new file mode 100644
index 000000000..c20958162
--- /dev/null
+++ b/lib/pleroma/web/templates/layout/static_fe.html.eex
@@ -0,0 +1,150 @@
+
+
+
+
+
+
+ <%= Application.get_env(:pleroma, :instance)[:name] %>
+
+
+
+
+
+ <%= render @view_module, @view_template, assigns %>
+
+
+
diff --git a/lib/pleroma/web/templates/static_fe/static_fe/notice.html.eex b/lib/pleroma/web/templates/static_fe/static_fe/notice.html.eex
new file mode 100644
index 000000000..9957697ad
--- /dev/null
+++ b/lib/pleroma/web/templates/static_fe/static_fe/notice.html.eex
@@ -0,0 +1,6 @@
+
+ <%= render("user_card.html", %{user: @user}) %>
+
+
<%= @object.data["content"] %>
+
+
diff --git a/lib/pleroma/web/templates/static_fe/static_fe/user_card.html.eex b/lib/pleroma/web/templates/static_fe/static_fe/user_card.html.eex
new file mode 100644
index 000000000..8b397c639
--- /dev/null
+++ b/lib/pleroma/web/templates/static_fe/static_fe/user_card.html.eex
@@ -0,0 +1,11 @@
+