mirror of
https://github.com/plausible/analytics.git
synced 2024-12-29 20:42:01 +03:00
22ecbe7bc7
* move format_price to Plausible.Billing * move PlausibleWeb.Components.Billing file to subfolder * extract new Notice module * rename test file and module name * move growth_grandfathered notice to notice.ex * extract a PlanBenefits module * extract PlanBox component * extract PageviewSlider component * fix plan benefits text color
289 lines
9.3 KiB
Elixir
289 lines
9.3 KiB
Elixir
defmodule PlausibleWeb.Components.Billing do
|
|
@moduledoc false
|
|
|
|
use Phoenix.Component
|
|
import PlausibleWeb.Components.Generic
|
|
require Plausible.Billing.Subscription.Status
|
|
alias PlausibleWeb.Router.Helpers, as: Routes
|
|
alias Plausible.Billing.{Subscription, Subscriptions}
|
|
|
|
def render_monthly_pageview_usage(%{usage: usage} = assigns)
|
|
when is_map_key(usage, :last_30_days) do
|
|
~H"""
|
|
<.monthly_pageview_usage_table usage={@usage.last_30_days} limit={@limit} period={:last_30_days} />
|
|
"""
|
|
end
|
|
|
|
def render_monthly_pageview_usage(assigns) do
|
|
~H"""
|
|
<article id="monthly_pageview_usage_container" x-data="{ tab: 'current_cycle' }" class="mt-8">
|
|
<h1 class="text-xl mb-6 font-bold dark:text-gray-100">Monthly pageviews usage</h1>
|
|
<div class="mb-3">
|
|
<ol class="divide-y divide-gray-300 dark:divide-gray-600 rounded-md border dark:border-gray-600 md:flex md:flex-row-reverse md:divide-y-0 md:overflow-hidden">
|
|
<.billing_cycle_tab
|
|
name="Ongoing cycle"
|
|
tab={:current_cycle}
|
|
date_range={@usage.current_cycle.date_range}
|
|
with_separator={true}
|
|
/>
|
|
<.billing_cycle_tab
|
|
name="Last cycle"
|
|
tab={:last_cycle}
|
|
date_range={@usage.last_cycle.date_range}
|
|
disabled={@usage.last_cycle.total == 0 && @usage.penultimate_cycle.total == 0}
|
|
with_separator={true}
|
|
/>
|
|
<.billing_cycle_tab
|
|
name="Penultimate cycle"
|
|
tab={:penultimate_cycle}
|
|
date_range={@usage.penultimate_cycle.date_range}
|
|
disabled={@usage.penultimate_cycle.total == 0}
|
|
/>
|
|
</ol>
|
|
</div>
|
|
<div x-show="tab === 'current_cycle'">
|
|
<.monthly_pageview_usage_table
|
|
usage={@usage.current_cycle}
|
|
limit={@limit}
|
|
period={:current_cycle}
|
|
/>
|
|
</div>
|
|
<div x-show="tab === 'last_cycle'">
|
|
<.monthly_pageview_usage_table usage={@usage.last_cycle} limit={@limit} period={:last_cycle} />
|
|
</div>
|
|
<div x-show="tab === 'penultimate_cycle'">
|
|
<.monthly_pageview_usage_table
|
|
usage={@usage.penultimate_cycle}
|
|
limit={@limit}
|
|
period={:penultimate_cycle}
|
|
/>
|
|
</div>
|
|
</article>
|
|
"""
|
|
end
|
|
|
|
attr(:usage, :map, required: true)
|
|
attr(:limit, :any, required: true)
|
|
attr(:period, :atom, required: true)
|
|
|
|
defp monthly_pageview_usage_table(assigns) do
|
|
~H"""
|
|
<.usage_and_limits_table>
|
|
<.usage_and_limits_row
|
|
id={"total_pageviews_#{@period}"}
|
|
title={"Total billable pageviews#{if @period == :last_30_days, do: " (last 30 days)"}"}
|
|
usage={@usage.total}
|
|
limit={@limit}
|
|
/>
|
|
<.usage_and_limits_row
|
|
id={"pageviews_#{@period}"}
|
|
pad
|
|
title="Pageviews"
|
|
usage={@usage.pageviews}
|
|
class="font-normal text-gray-500 dark:text-gray-400"
|
|
/>
|
|
<.usage_and_limits_row
|
|
id={"custom_events_#{@period}"}
|
|
pad
|
|
title="Custom events"
|
|
usage={@usage.custom_events}
|
|
class="font-normal text-gray-500 dark:text-gray-400"
|
|
/>
|
|
</.usage_and_limits_table>
|
|
"""
|
|
end
|
|
|
|
attr(:name, :string, required: true)
|
|
attr(:date_range, :any, required: true)
|
|
attr(:tab, :atom, required: true)
|
|
attr(:disabled, :boolean, default: false)
|
|
attr(:with_separator, :boolean, default: false)
|
|
|
|
defp billing_cycle_tab(assigns) do
|
|
~H"""
|
|
<li id={"billing_cycle_tab_#{@tab}"} class="relative md:w-1/3">
|
|
<button
|
|
class={["w-full group", @disabled && "pointer-events-none opacity-50 dark:opacity-25"]}
|
|
x-on:click={"tab = '#{@tab}'"}
|
|
>
|
|
<span
|
|
class="absolute left-0 top-0 h-full w-1 md:bottom-0 md:top-auto md:h-1 md:w-full"
|
|
x-bind:class={"tab === '#{@tab}' ? 'bg-indigo-500' : 'bg-transparent group-hover:bg-gray-200 dark:group-hover:bg-gray-700 '"}
|
|
aria-hidden="true"
|
|
>
|
|
</span>
|
|
<div class={"flex items-center justify-between md:flex-col md:items-start py-2 pr-2 #{if @with_separator, do: "pl-2 md:pl-4", else: "pl-2"}"}>
|
|
<span
|
|
class="text-sm dark:text-gray-100"
|
|
x-bind:class={"tab === '#{@tab}' ? 'text-indigo-600 dark:text-indigo-500 font-semibold' : 'font-medium'"}
|
|
>
|
|
<%= @name %>
|
|
</span>
|
|
<span class="flex text-xs text-gray-500 dark:text-gray-400">
|
|
<%= if @disabled,
|
|
do: "Not available",
|
|
else: PlausibleWeb.TextHelpers.format_date_range(@date_range) %>
|
|
</span>
|
|
</div>
|
|
</button>
|
|
<div
|
|
:if={@with_separator}
|
|
class="absolute inset-0 left-0 top-0 w-3 hidden md:block"
|
|
aria-hidden="true"
|
|
>
|
|
<svg
|
|
class="h-full w-full text-gray-300 dark:text-gray-600"
|
|
viewBox="0 0 12 82"
|
|
fill="none"
|
|
preserveAspectRatio="none"
|
|
>
|
|
<path
|
|
d="M0.5 0V31L10.5 41L0.5 51V82"
|
|
stroke="currentcolor"
|
|
vector-effect="non-scaling-stroke"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
</li>
|
|
"""
|
|
end
|
|
|
|
slot(:inner_block, required: true)
|
|
attr(:rest, :global)
|
|
|
|
def usage_and_limits_table(assigns) do
|
|
~H"""
|
|
<table class="min-w-full text-gray-900 dark:text-gray-100" {@rest}>
|
|
<tbody class="divide-y divide-gray-200 dark:divide-gray-600">
|
|
<%= render_slot(@inner_block) %>
|
|
</tbody>
|
|
</table>
|
|
"""
|
|
end
|
|
|
|
attr(:title, :string, required: true)
|
|
attr(:usage, :integer, required: true)
|
|
attr(:limit, :integer, default: nil)
|
|
attr(:pad, :boolean, default: false)
|
|
attr(:rest, :global)
|
|
|
|
def usage_and_limits_row(assigns) do
|
|
~H"""
|
|
<tr {@rest}>
|
|
<td class={["py-4 pr-1 text-sm sm:whitespace-nowrap text-left", @pad && "pl-6"]}>
|
|
<%= @title %>
|
|
</td>
|
|
<td class="py-4 text-sm sm:whitespace-nowrap text-right">
|
|
<%= Cldr.Number.to_string!(@usage) %>
|
|
<%= if is_number(@limit), do: "/ #{Cldr.Number.to_string!(@limit)}" %>
|
|
</td>
|
|
</tr>
|
|
"""
|
|
end
|
|
|
|
def monthly_quota_box(assigns) do
|
|
~H"""
|
|
<div
|
|
id="monthly-quota-box"
|
|
class="h-32 px-2 py-4 my-4 text-center bg-gray-100 rounded dark:bg-gray-900"
|
|
style="width: 11.75rem;"
|
|
>
|
|
<h4 class="font-black dark:text-gray-100">Monthly quota</h4>
|
|
<div class="py-2 text-xl font-medium dark:text-gray-100">
|
|
<%= PlausibleWeb.AuthView.subscription_quota(@subscription, format: :long) %>
|
|
</div>
|
|
<.styled_link
|
|
:if={
|
|
not (Plausible.Auth.enterprise_configured?(@user) && Subscriptions.halted?(@subscription))
|
|
}
|
|
id="#upgrade-or-change-plan-link"
|
|
href={Routes.billing_path(PlausibleWeb.Endpoint, :choose_plan)}
|
|
>
|
|
<%= change_plan_or_upgrade_text(@subscription) %>
|
|
</.styled_link>
|
|
</div>
|
|
"""
|
|
end
|
|
|
|
def present_enterprise_plan(assigns) do
|
|
~H"""
|
|
<ul class="w-full py-4">
|
|
<li>
|
|
Up to <b><%= present_limit(@plan, :monthly_pageview_limit) %></b> monthly pageviews
|
|
</li>
|
|
<li>
|
|
Up to <b><%= present_limit(@plan, :site_limit) %></b> sites
|
|
</li>
|
|
<li>
|
|
Up to <b><%= present_limit(@plan, :hourly_api_request_limit) %></b> hourly api requests
|
|
</li>
|
|
</ul>
|
|
"""
|
|
end
|
|
|
|
defp present_limit(enterprise_plan, key) do
|
|
enterprise_plan
|
|
|> Map.fetch!(key)
|
|
|> PlausibleWeb.StatsView.large_number_format()
|
|
end
|
|
|
|
attr :id, :string, required: true
|
|
attr :paddle_product_id, :string, required: true
|
|
attr :checkout_disabled, :boolean, default: false
|
|
attr :user, :map, required: true
|
|
attr :confirm_message, :any, default: nil
|
|
slot :inner_block, required: true
|
|
|
|
def paddle_button(assigns) do
|
|
confirmed =
|
|
if assigns.confirm_message, do: "confirm(\"#{assigns.confirm_message}\")", else: "true"
|
|
|
|
assigns = assign(assigns, :confirmed, confirmed)
|
|
|
|
~H"""
|
|
<button
|
|
id={@id}
|
|
onclick={"if (#{@confirmed}) {Paddle.Checkout.open(#{Jason.encode!(%{product: @paddle_product_id, email: @user.email, disableLogout: true, passthrough: @user.id, success: Routes.billing_path(PlausibleWeb.Endpoint, :upgrade_success), theme: "none"})})}"}
|
|
class={[
|
|
"w-full mt-6 block rounded-md py-2 px-3 text-center text-sm font-semibold leading-6 text-white",
|
|
!@checkout_disabled && "bg-indigo-600 hover:bg-indigo-500",
|
|
@checkout_disabled && "pointer-events-none bg-gray-400 dark:bg-gray-600"
|
|
]}
|
|
>
|
|
<%= render_slot(@inner_block) %>
|
|
</button>
|
|
"""
|
|
end
|
|
|
|
def paddle_script(assigns) do
|
|
~H"""
|
|
<script type="text/javascript" src="https://cdn.paddle.com/paddle/paddle.js">
|
|
</script>
|
|
<script :if={Application.get_env(:plausible, :environment) == "dev"}>
|
|
Paddle.Environment.set('sandbox')
|
|
</script>
|
|
<script>
|
|
Paddle.Setup({vendor: <%= Application.get_env(:plausible, :paddle) |> Keyword.fetch!(:vendor_id) %> })
|
|
</script>
|
|
"""
|
|
end
|
|
|
|
def upgrade_link(assigns) do
|
|
~H"""
|
|
<PlausibleWeb.Components.Generic.button_link
|
|
id="upgrade-link-2"
|
|
href={Routes.billing_path(PlausibleWeb.Endpoint, :choose_plan)}
|
|
>
|
|
Upgrade
|
|
</PlausibleWeb.Components.Generic.button_link>
|
|
"""
|
|
end
|
|
|
|
defp change_plan_or_upgrade_text(nil), do: "Upgrade"
|
|
|
|
defp change_plan_or_upgrade_text(%Subscription{status: Subscription.Status.deleted()}),
|
|
do: "Upgrade"
|
|
|
|
defp change_plan_or_upgrade_text(_subscription), do: "Change plan"
|
|
end
|