Merge pull request 'add CSV teplate closes #329' (#347) from feature/329_csv_specification into main
All checks were successful
continuous-integration/drone/push Build is passing
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #347
This commit is contained in:
commit
22d50d6c46
10 changed files with 69 additions and 161 deletions
|
|
@ -191,6 +191,26 @@ A **basic CSV member import feature** that allows administrators to upload a CSV
|
||||||
- `/templates/member_import_de.csv`
|
- `/templates/member_import_de.csv`
|
||||||
- In LiveView, link them using Phoenix static path helpers (e.g. `~p` or `Routes.static_path/2`, depending on Phoenix version).
|
- In LiveView, link them using Phoenix static path helpers (e.g. `~p` or `Routes.static_path/2`, depending on Phoenix version).
|
||||||
|
|
||||||
|
**Example Usage in LiveView Templates:**
|
||||||
|
|
||||||
|
```heex
|
||||||
|
<!-- Using ~p sigil (Phoenix 1.7+) -->
|
||||||
|
<.link href={~p"/templates/member_import_en.csv"} download>
|
||||||
|
<%= gettext("Download English Template") %>
|
||||||
|
</.link>
|
||||||
|
|
||||||
|
<.link href={~p"/templates/member_import_de.csv"} download>
|
||||||
|
<%= gettext("Download German Template") %>
|
||||||
|
</.link>
|
||||||
|
|
||||||
|
<!-- Alternative: Using Routes.static_path/2 -->
|
||||||
|
<.link href={Routes.static_path(MvWeb.Endpoint, "/templates/member_import_en.csv")} download>
|
||||||
|
<%= gettext("Download English Template") %>
|
||||||
|
</.link>
|
||||||
|
```
|
||||||
|
|
||||||
|
**Note:** The `templates` directory must be included in `MvWeb.static_paths()` (configured in `lib/mv_web.ex`) for the files to be served.
|
||||||
|
|
||||||
### File Limits
|
### File Limits
|
||||||
|
|
||||||
- **Max file size:** 10 MB
|
- **Max file size:** 10 MB
|
||||||
|
|
|
||||||
|
|
@ -17,7 +17,7 @@ defmodule MvWeb do
|
||||||
those modules here.
|
those modules here.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def static_paths, do: ~w(assets fonts images favicon.ico robots.txt)
|
def static_paths, do: ~w(assets fonts images favicon.ico robots.txt templates)
|
||||||
|
|
||||||
def router do
|
def router do
|
||||||
quote do
|
quote do
|
||||||
|
|
|
||||||
|
|
@ -148,9 +148,9 @@ defmodule MvWeb.MemberLive.Show do
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<%!-- Custom Fields Section --%>
|
<%!-- Custom Fields Section --%>
|
||||||
<%= if Enum.any?(@custom_fields) do %>
|
<%= if is_list(@custom_fields) && Enum.any?(@custom_fields) do %>
|
||||||
<div>
|
<div>
|
||||||
<.section_box title={gettext("Custom Fields")}>
|
<.section_box title={gettext("Additional Data Fields")}>
|
||||||
<div class="grid grid-cols-2 gap-4">
|
<div class="grid grid-cols-2 gap-4">
|
||||||
<%= for custom_field <- @custom_fields do %>
|
<%= for custom_field <- @custom_fields do %>
|
||||||
<% cfv = find_custom_field_value(@member.custom_field_values, custom_field.id) %>
|
<% cfv = find_custom_field_value(@member.custom_field_values, custom_field.id) %>
|
||||||
|
|
@ -233,13 +233,15 @@ defmodule MvWeb.MemberLive.Show do
|
||||||
|
|
||||||
@impl true
|
@impl true
|
||||||
def handle_params(%{"id" => id}, _, socket) do
|
def handle_params(%{"id" => id}, _, socket) do
|
||||||
# Load custom fields once using assign_new to avoid repeated queries
|
# Load custom fields for display
|
||||||
socket =
|
# Note: Each page load starts a new LiveView process, so caching with
|
||||||
assign_new(socket, :custom_fields, fn ->
|
# assign_new is not necessary here (mount creates a fresh socket each time)
|
||||||
|
custom_fields =
|
||||||
Mv.Membership.CustomField
|
Mv.Membership.CustomField
|
||||||
|> Ash.Query.sort(name: :asc)
|
|> Ash.Query.sort(name: :asc)
|
||||||
|> Ash.read!()
|
|> Ash.read!()
|
||||||
end)
|
|
||||||
|
socket = assign(socket, :custom_fields, custom_fields)
|
||||||
|
|
||||||
query =
|
query =
|
||||||
Mv.Membership.Member
|
Mv.Membership.Member
|
||||||
|
|
|
||||||
|
|
@ -633,7 +633,6 @@ msgstr "Bitte wähle zuerst ein Benutzerdefiniertes Feld"
|
||||||
|
|
||||||
#: lib/mv_web/components/layouts/sidebar.ex
|
#: lib/mv_web/components/layouts/sidebar.ex
|
||||||
#: lib/mv_web/live/member_live/form.ex
|
#: lib/mv_web/live/member_live/form.ex
|
||||||
#: lib/mv_web/live/member_live/show.ex
|
|
||||||
#, elixir-autogen, elixir-format
|
#, elixir-autogen, elixir-format
|
||||||
msgid "Custom Fields"
|
msgid "Custom Fields"
|
||||||
msgstr "Benutzerdefinierte Felder"
|
msgstr "Benutzerdefinierte Felder"
|
||||||
|
|
@ -2059,6 +2058,11 @@ msgstr ""
|
||||||
msgid "read_only - Read access to all data"
|
msgid "read_only - Read access to all data"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
|
#: lib/mv_web/live/member_live/show.ex
|
||||||
|
#, elixir-autogen, elixir-format
|
||||||
|
msgid "Additional Data Fields"
|
||||||
|
msgstr "Zusätzliche Datenfelder"
|
||||||
|
|
||||||
#~ #: lib/mv_web/live/custom_field_live/show.ex
|
#~ #: lib/mv_web/live/custom_field_live/show.ex
|
||||||
#~ #, elixir-autogen, elixir-format
|
#~ #, elixir-autogen, elixir-format
|
||||||
#~ msgid "Auto-generated identifier (immutable)"
|
#~ msgid "Auto-generated identifier (immutable)"
|
||||||
|
|
|
||||||
|
|
@ -634,7 +634,6 @@ msgstr ""
|
||||||
|
|
||||||
#: lib/mv_web/components/layouts/sidebar.ex
|
#: lib/mv_web/components/layouts/sidebar.ex
|
||||||
#: lib/mv_web/live/member_live/form.ex
|
#: lib/mv_web/live/member_live/form.ex
|
||||||
#: lib/mv_web/live/member_live/show.ex
|
|
||||||
#, elixir-autogen, elixir-format
|
#, elixir-autogen, elixir-format
|
||||||
msgid "Custom Fields"
|
msgid "Custom Fields"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
@ -2059,3 +2058,8 @@ msgstr ""
|
||||||
#, elixir-autogen, elixir-format
|
#, elixir-autogen, elixir-format
|
||||||
msgid "read_only - Read access to all data"
|
msgid "read_only - Read access to all data"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
|
#: lib/mv_web/live/member_live/show.ex
|
||||||
|
#, elixir-autogen, elixir-format
|
||||||
|
msgid "Additional Data Fields"
|
||||||
|
msgstr ""
|
||||||
|
|
|
||||||
|
|
@ -634,7 +634,6 @@ msgstr ""
|
||||||
|
|
||||||
#: lib/mv_web/components/layouts/sidebar.ex
|
#: lib/mv_web/components/layouts/sidebar.ex
|
||||||
#: lib/mv_web/live/member_live/form.ex
|
#: lib/mv_web/live/member_live/form.ex
|
||||||
#: lib/mv_web/live/member_live/show.ex
|
|
||||||
#, elixir-autogen, elixir-format, fuzzy
|
#, elixir-autogen, elixir-format, fuzzy
|
||||||
msgid "Custom Fields"
|
msgid "Custom Fields"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
@ -2060,6 +2059,11 @@ msgstr ""
|
||||||
msgid "read_only - Read access to all data"
|
msgid "read_only - Read access to all data"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
|
#: lib/mv_web/live/member_live/show.ex
|
||||||
|
#, elixir-autogen, elixir-format
|
||||||
|
msgid "Additional Data Fields"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
#~ #: lib/mv_web/live/components/payment_filter_component.ex
|
#~ #: lib/mv_web/live/components/payment_filter_component.ex
|
||||||
#~ #, elixir-autogen, elixir-format
|
#~ #, elixir-autogen, elixir-format
|
||||||
#~ msgid "All payment statuses"
|
#~ msgid "All payment statuses"
|
||||||
|
|
|
||||||
2
priv/static/templates/member_import_de.csv
Normal file
2
priv/static/templates/member_import_de.csv
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
Vorname;Nachname;E-Mail;Straße;PLZ;Stadt
|
||||||
|
Max;Mustermann;max.mustermann@example.com;Hauptstraße;10115;Berlin
|
||||||
|
2
priv/static/templates/member_import_en.csv
Normal file
2
priv/static/templates/member_import_en.csv
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
first_name;last_name;email;street;postal_code;city
|
||||||
|
John;Doe;john.doe@example.com;Main Street;12345;Berlin
|
||||||
|
|
|
@ -1,141 +0,0 @@
|
||||||
defmodule MvWeb.Helpers.MemberHelpersTest do
|
|
||||||
@moduledoc """
|
|
||||||
Tests for the display_name/1 helper function in MemberHelpers.
|
|
||||||
"""
|
|
||||||
use Mv.DataCase, async: true
|
|
||||||
|
|
||||||
alias Mv.Membership.Member
|
|
||||||
alias MvWeb.Helpers.MemberHelpers
|
|
||||||
|
|
||||||
describe "display_name/1" do
|
|
||||||
test "returns full name when both first_name and last_name are present" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: "John",
|
|
||||||
last_name: "Doe",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "John Doe"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns email when both first_name and last_name are nil" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: nil,
|
|
||||||
last_name: nil,
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "john@example.com"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns first_name only when last_name is nil" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: "John",
|
|
||||||
last_name: nil,
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "John"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns last_name only when first_name is nil" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: nil,
|
|
||||||
last_name: "Doe",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "Doe"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns email when first_name and last_name are empty strings" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: "",
|
|
||||||
last_name: "",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "john@example.com"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns email when first_name and last_name are whitespace only" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: " ",
|
|
||||||
last_name: " \t ",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "john@example.com"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "trims whitespace from name parts" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: " John ",
|
|
||||||
last_name: " Doe ",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "John Doe"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "handles one empty string and one nil" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: "",
|
|
||||||
last_name: nil,
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "john@example.com"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "handles one nil and one empty string" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: nil,
|
|
||||||
last_name: "",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "john@example.com"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "handles one whitespace and one nil" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: " ",
|
|
||||||
last_name: nil,
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "john@example.com"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "handles one valid name and one whitespace" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: "John",
|
|
||||||
last_name: " ",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "John"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "handles member with only first_name containing whitespace" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: " John ",
|
|
||||||
last_name: nil,
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "John"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "handles member with only last_name containing whitespace" do
|
|
||||||
member = %Member{
|
|
||||||
first_name: nil,
|
|
||||||
last_name: " Doe ",
|
|
||||||
email: "john@example.com"
|
|
||||||
}
|
|
||||||
|
|
||||||
assert MemberHelpers.display_name(member) == "Doe"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
@ -7,13 +7,13 @@ defmodule MvWeb.MemberLive.ShowTest do
|
||||||
- Custom Fields section visibility (Issue #282 regression test)
|
- Custom Fields section visibility (Issue #282 regression test)
|
||||||
- Custom field values formatting
|
- Custom field values formatting
|
||||||
|
|
||||||
## Note on async: false
|
## Note on async
|
||||||
Tests use `async: false` (not `async: true`) to prevent PostgreSQL deadlocks
|
Tests can run with `async: true` because:
|
||||||
when creating members and custom fields concurrently. This is intentional and
|
- Each test explicitly manages its own custom fields (creates/deletes as needed)
|
||||||
documented here to avoid confusion in commit messages.
|
- The SQL Sandbox provides proper isolation between parallel tests
|
||||||
|
- Custom field cleanup in tests ensures no interference between tests
|
||||||
"""
|
"""
|
||||||
# async: false to prevent PostgreSQL deadlocks when creating members and custom fields
|
use MvWeb.ConnCase, async: true
|
||||||
use MvWeb.ConnCase, async: false
|
|
||||||
import Phoenix.LiveViewTest
|
import Phoenix.LiveViewTest
|
||||||
require Ash.Query
|
require Ash.Query
|
||||||
use Gettext, backend: MvWeb.Gettext
|
use Gettext, backend: MvWeb.Gettext
|
||||||
|
|
@ -113,11 +113,22 @@ defmodule MvWeb.MemberLive.ShowTest do
|
||||||
conn: conn,
|
conn: conn,
|
||||||
member: member
|
member: member
|
||||||
} do
|
} do
|
||||||
|
# Ensure no custom fields exist for this test
|
||||||
|
# This ensures test isolation even if previous tests created custom fields
|
||||||
|
existing_custom_fields = Ash.read!(CustomField)
|
||||||
|
|
||||||
|
for cf <- existing_custom_fields do
|
||||||
|
Ash.destroy!(cf)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Verify no custom fields exist
|
||||||
|
assert Ash.read!(CustomField) == []
|
||||||
|
|
||||||
conn = conn_with_oidc_user(conn)
|
conn = conn_with_oidc_user(conn)
|
||||||
{:ok, _view, html} = live(conn, ~p"/members/#{member}")
|
{:ok, _view, html} = live(conn, ~p"/members/#{member}")
|
||||||
|
|
||||||
# Custom Fields section should NOT be visible
|
# Custom Fields section should NOT be visible
|
||||||
refute html =~ gettext("Custom Fields")
|
refute html =~ gettext("Additional Data Fields")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue