Compare commits
	
		
			65 Commits
		
	
	
		
			0.2.1
			...
			119f2af6bb
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 119f2af6bb | ||
| 9db6b2c316 | |||
| 22281486e0 | |||
| 1b7546aede | |||
|  | e153893a5b | ||
|  | 2f7c17aad3 | ||
| 664c65d136 | |||
|  | ef76eb002d | ||
|  | 5b40ac4137 | ||
|  | ed20cdd858 | ||
|  | 1e0ec82f3e | ||
|  | d07ac801aa | ||
|  | 420e7c2d71 | ||
|  | 40877d1ac0 | ||
|  | 1dd30e6a5b | ||
|  | eedaf33e25 | ||
| b0a100cd6c | |||
| 6455e2710d | |||
| 6523b28aa2 | |||
| bad1a23dfe | |||
| e0ddefe1d7 | |||
| 5d6ecba9f7 | |||
| a2d1ff9b89 | |||
| 34288a0070 | |||
| f9b08222e1 | |||
| c0179f48bd | |||
| 8bb4aab49c | |||
| 9f2cc54738 | |||
| 0309e9d714 | |||
| af4af84515 | |||
| ec6946068e | |||
| f120e54c3e | |||
| 3d115c6383 | |||
| 1d4622a285 | |||
| d9e7948bb0 | |||
| da8c788992 | |||
| bd20820361 | |||
| d0ee81093a | |||
| 6080fdbe64 | |||
| f42aaf9099 | |||
| a6aa6f3386 | |||
| 8405513337 | |||
| d0857eccc1 | |||
| 4c9e707181 | |||
| e6a4fbcfb5 | |||
| a6b2c6181e | |||
| d79d0fa179 | |||
| 46387e8d7a | |||
| dccea1b9de | |||
| 5ca21c64fd | |||
| 9773ccc6ff | |||
| 9cd2bc574b | |||
| ee28de1178 | |||
| 91cf9d0eb5 | |||
| 3ce8eda712 | |||
| 1a78e88b34 | |||
| 968abd04ee | |||
| dc209fa192 | |||
| a80df49fdd | |||
| 92d1d21d00 | |||
| 917f627933 | |||
| 4946a6b119 | |||
| 9f784c3190 | |||
| aa08e212ee | |||
| 80ad939aab | 
							
								
								
									
										14
									
								
								.credo.exs
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								.credo.exs
									
									
									
									
									
								
							| @@ -157,17 +157,17 @@ | |||||||
|         # |         # | ||||||
|         # Controversial and experimental checks (opt-in, just replace `false` with `[]`) |         # Controversial and experimental checks (opt-in, just replace `false` with `[]`) | ||||||
|         # |         # | ||||||
|         {Credo.Check.Consistency.MultiAliasImportRequireUse, false}, |         {Credo.Check.Consistency.MultiAliasImportRequireUse, []}, | ||||||
|         {Credo.Check.Consistency.UnusedVariableNames, false}, |         {Credo.Check.Consistency.UnusedVariableNames, [force: :meaningful]}, | ||||||
|         {Credo.Check.Design.DuplicatedCode, false}, |         {Credo.Check.Design.DuplicatedCode, false}, | ||||||
|         {Credo.Check.Readability.AliasAs, false}, |         {Credo.Check.Readability.AliasAs, false}, | ||||||
|         {Credo.Check.Readability.BlockPipe, false}, |         {Credo.Check.Readability.BlockPipe, false}, | ||||||
|         {Credo.Check.Readability.ImplTrue, false}, |         {Credo.Check.Readability.ImplTrue, false}, | ||||||
|         {Credo.Check.Readability.MultiAlias, false}, |         {Credo.Check.Readability.MultiAlias, false}, | ||||||
|         {Credo.Check.Readability.SeparateAliasRequire, false}, |         {Credo.Check.Readability.SeparateAliasRequire, []}, | ||||||
|         {Credo.Check.Readability.SinglePipe, false}, |         {Credo.Check.Readability.SinglePipe, false}, | ||||||
|         {Credo.Check.Readability.Specs, false}, |         {Credo.Check.Readability.Specs, false}, | ||||||
|         {Credo.Check.Readability.StrictModuleLayout, false}, |         {Credo.Check.Readability.StrictModuleLayout, []}, | ||||||
|         {Credo.Check.Readability.WithCustomTaggedTuple, false}, |         {Credo.Check.Readability.WithCustomTaggedTuple, false}, | ||||||
|         {Credo.Check.Refactor.ABCSize, false}, |         {Credo.Check.Refactor.ABCSize, false}, | ||||||
|         {Credo.Check.Refactor.AppendSingleItem, false}, |         {Credo.Check.Refactor.AppendSingleItem, false}, | ||||||
| @@ -176,9 +176,9 @@ | |||||||
|         {Credo.Check.Refactor.NegatedIsNil, false}, |         {Credo.Check.Refactor.NegatedIsNil, false}, | ||||||
|         {Credo.Check.Refactor.PipeChainStart, false}, |         {Credo.Check.Refactor.PipeChainStart, false}, | ||||||
|         {Credo.Check.Refactor.VariableRebinding, false}, |         {Credo.Check.Refactor.VariableRebinding, false}, | ||||||
|         {Credo.Check.Warning.LeakyEnvironment, false}, |         {Credo.Check.Warning.LeakyEnvironment, []}, | ||||||
|         {Credo.Check.Warning.MapGetUnsafePass, false}, |         {Credo.Check.Warning.MapGetUnsafePass, []}, | ||||||
|         {Credo.Check.Warning.UnsafeToAtom, false} |         {Credo.Check.Warning.UnsafeToAtom, []} | ||||||
|  |  | ||||||
|         # |         # | ||||||
|         # Custom checks can be created using `mix credo.gen.check`. |         # Custom checks can be created using `mix credo.gen.check`. | ||||||
|   | |||||||
| @@ -35,9 +35,7 @@ steps: | |||||||
|       from_secret: docker_username |       from_secret: docker_username | ||||||
|     password: |     password: | ||||||
|       from_secret: docker_password |       from_secret: docker_password | ||||||
|     tags: |     tags: latest | ||||||
|       - latest |  | ||||||
|       - ${DRONE_TAG} |  | ||||||
|   when: |   when: | ||||||
|     branch: |     branch: | ||||||
|     - stable |     - stable | ||||||
| @@ -79,7 +77,7 @@ services: | |||||||
| volumes: | volumes: | ||||||
|   - name: cache |   - name: cache | ||||||
|     host: |     host: | ||||||
|       path: /tmp/drone-cache |       path: /run/media/default/ssdsrv/gitea/drone-cache | ||||||
|   - name: docker_sock |   - name: docker_sock | ||||||
|     host: |     host: | ||||||
|       path: /var/run/docker.sock |       path: /var/run/docker.sock | ||||||
|   | |||||||
							
								
								
									
										28
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,31 @@ | |||||||
|  | # v0.4.1 | ||||||
|  | - Fix button and tag text wrapping | ||||||
|  | - Code quality fixes | ||||||
|  |  | ||||||
|  | # v0.4.0 | ||||||
|  | - Make tables sortable | ||||||
|  | - Add link to changelog from version number | ||||||
|  | - Fix some elements flashing with black background | ||||||
|  | - Fix bug with moving ammo group to new container | ||||||
|  | - Fix bug with no error showing up for create ammo group form | ||||||
|  |  | ||||||
|  | # v0.3.0 | ||||||
|  | - Fix ammo type counts not showing when count is 0 | ||||||
|  | - Add prompt to create first container before first ammo group | ||||||
|  | - Edit and delete shot groups from ammo group show page | ||||||
|  | - Use today's date when adding new shot groups | ||||||
|  | - Create multiple ammo groups at one time | ||||||
|  |  | ||||||
|  | # v0.2.3 | ||||||
|  | - Fix modals with overflowing forms | ||||||
|  | - Fix grids having uneven margins in phone mode | ||||||
|  | - Add page titles to registration and setting pages | ||||||
|  |  | ||||||
|  | # v0.2.2 | ||||||
|  | - Fix loading and reconnecting pages not being fixed | ||||||
|  | - Fix closing modal in some cases not triggering a page reload | ||||||
|  | - Fix error when display container and tag edit routes from a fresh reload | ||||||
|  |  | ||||||
| # v0.2.1 | # v0.2.1 | ||||||
| - Fix checkbox spacing for mobile view | - Fix checkbox spacing for mobile view | ||||||
| - Fix spacing with form elements in mobile view | - Fix spacing with form elements in mobile view | ||||||
|   | |||||||
| @@ -25,6 +25,7 @@ | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   .btn { |   .btn { | ||||||
|  |     @apply inline-block break-all min-w-4; | ||||||
|     @apply focus:outline-none px-4 py-2 rounded-lg; |     @apply focus:outline-none px-4 py-2 rounded-lg; | ||||||
|     @apply shadow-sm focus:shadow-lg; |     @apply shadow-sm focus:shadow-lg; | ||||||
|     @apply transition-all duration-300 ease-in-out; |     @apply transition-all duration-300 ease-in-out; | ||||||
| @@ -51,6 +52,7 @@ | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   .link { |   .link { | ||||||
|  |     @apply inline-block break-all min-w-4; | ||||||
|     @apply hover:underline; |     @apply hover:underline; | ||||||
|     @apply transition-colors duration-500 ease-in-out; |     @apply transition-colors duration-500 ease-in-out; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -2,7 +2,10 @@ | |||||||
| // update. https://github.com/phoenixframework/phoenix_live_view/issues/1011 | // update. https://github.com/phoenixframework/phoenix_live_view/issues/1011 | ||||||
|  |  | ||||||
| export default { | export default { | ||||||
|   attrs () { return this.el.getAttribute('data-attrs').split(', ') }, |   attrs () { | ||||||
|  |     const attrs = this.el.getAttribute('data-attrs') | ||||||
|  |     if (attrs) { return attrs.split(', ') } else { return [] } | ||||||
|  |   }, | ||||||
|   beforeUpdate () { this.prevAttrs = this.attrs().map(name => [name, this.el.getAttribute(name)]) }, |   beforeUpdate () { this.prevAttrs = this.attrs().map(name => [name, this.el.getAttribute(name)]) }, | ||||||
|   updated () { this.prevAttrs.forEach(([name, val]) => this.el.setAttribute(name, val)) } |   updated () { this.prevAttrs.forEach(([name, val]) => this.el.setAttribute(name, val)) } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -28,6 +28,20 @@ module.exports = { | |||||||
|         128: '32rem', |         128: '32rem', | ||||||
|         192: '48rem', |         192: '48rem', | ||||||
|         256: '64rem' |         256: '64rem' | ||||||
|  |       }, | ||||||
|  |       minWidth: { | ||||||
|  |         4: '1rem', | ||||||
|  |         8: '2rem', | ||||||
|  |         12: '3rem', | ||||||
|  |         16: '4rem', | ||||||
|  |         20: '8rem' | ||||||
|  |       }, | ||||||
|  |       maxWidth: { | ||||||
|  |         4: '1rem', | ||||||
|  |         8: '2rem', | ||||||
|  |         12: '3rem', | ||||||
|  |         16: '4rem', | ||||||
|  |         20: '8rem' | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -196,7 +196,7 @@ defmodule Cannery.Accounts do | |||||||
|          {:ok, _} <- Repo.transaction(user_email_multi(user, email, context)) do |          {:ok, _} <- Repo.transaction(user_email_multi(user, email, context)) do | ||||||
|       :ok |       :ok | ||||||
|     else |     else | ||||||
|       _ -> :error |       _error_tuple -> :error | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
| @@ -265,7 +265,7 @@ defmodule Cannery.Accounts do | |||||||
|     |> Repo.transaction() |     |> Repo.transaction() | ||||||
|     |> case do |     |> case do | ||||||
|       {:ok, %{user: user}} -> {:ok, user} |       {:ok, %{user: user}} -> {:ok, user} | ||||||
|       {:error, :user, changeset, _} -> {:error, changeset} |       {:error, :user, changeset, _changes_so_far} -> {:error, changeset} | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
| @@ -372,7 +372,7 @@ defmodule Cannery.Accounts do | |||||||
|          {:ok, %{user: user}} <- Repo.transaction(confirm_user_multi(user)) do |          {:ok, %{user: user}} <- Repo.transaction(confirm_user_multi(user)) do | ||||||
|       {:ok, user} |       {:ok, user} | ||||||
|     else |     else | ||||||
|       _ -> :error |       _error_tuple -> :error | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
| @@ -420,7 +420,7 @@ defmodule Cannery.Accounts do | |||||||
|          %User{} = user <- Repo.one(query) do |          %User{} = user <- Repo.one(query) do | ||||||
|       user |       user | ||||||
|     else |     else | ||||||
|       _ -> nil |       _error_tuple -> nil | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
| @@ -444,7 +444,7 @@ defmodule Cannery.Accounts do | |||||||
|     |> Repo.transaction() |     |> Repo.transaction() | ||||||
|     |> case do |     |> case do | ||||||
|       {:ok, %{user: user}} -> {:ok, user} |       {:ok, %{user: user}} -> {:ok, user} | ||||||
|       {:error, :user, changeset, _} -> {:error, changeset} |       {:error, :user, changeset, _changes_so_far} -> {:error, changeset} | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -171,7 +171,7 @@ defmodule Cannery.Accounts.User do | |||||||
|     Bcrypt.verify_pass(password, hashed_password) |     Bcrypt.verify_pass(password, hashed_password) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def valid_password?(_, _) do |   def valid_password?(_invalid_user, _invalid_password) do | ||||||
|     Bcrypt.no_user_verify() |     Bcrypt.no_user_verify() | ||||||
|     false |     false | ||||||
|   end |   end | ||||||
|   | |||||||
| @@ -8,6 +8,8 @@ defmodule Cannery.Ammo do | |||||||
|   alias Cannery.Ammo.{AmmoGroup, AmmoType} |   alias Cannery.Ammo.{AmmoGroup, AmmoType} | ||||||
|   alias Ecto.Changeset |   alias Ecto.Changeset | ||||||
|  |  | ||||||
|  |   @ammo_group_create_limit 10_000 | ||||||
|  |  | ||||||
|   @doc """ |   @doc """ | ||||||
|   Returns the list of ammo_types. |   Returns the list of ammo_types. | ||||||
|  |  | ||||||
| @@ -90,7 +92,7 @@ defmodule Cannery.Ammo do | |||||||
|       from ag in AmmoGroup, |       from ag in AmmoGroup, | ||||||
|         where: ag.ammo_type_id == ^ammo_type_id, |         where: ag.ammo_type_id == ^ammo_type_id, | ||||||
|         select: sum(ag.count) |         select: sum(ag.count) | ||||||
|     ) |     ) || 0 | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @doc """ |   @doc """ | ||||||
| @@ -117,7 +119,7 @@ defmodule Cannery.Ammo do | |||||||
|         left_join: sg in assoc(ag, :shot_groups), |         left_join: sg in assoc(ag, :shot_groups), | ||||||
|         where: ag.ammo_type_id == ^ammo_type_id, |         where: ag.ammo_type_id == ^ammo_type_id, | ||||||
|         select: sum(sg.count) |         select: sum(sg.count) | ||||||
|     ) |     ) || 0 | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @doc """ |   @doc """ | ||||||
| @@ -306,7 +308,7 @@ defmodule Cannery.Ammo do | |||||||
|   def get_used_count(%AmmoGroup{} = ammo_group) do |   def get_used_count(%AmmoGroup{} = ammo_group) do | ||||||
|     ammo_group |     ammo_group | ||||||
|     |> Repo.preload(:shot_groups) |     |> Repo.preload(:shot_groups) | ||||||
|     |> Map.get(:shot_groups) |     |> Map.fetch!(:shot_groups) | ||||||
|     |> Enum.map(fn %{count: count} -> count end) |     |> Enum.map(fn %{count: count} -> count end) | ||||||
|     |> Enum.sum() |     |> Enum.sum() | ||||||
|   end |   end | ||||||
| @@ -327,36 +329,61 @@ defmodule Cannery.Ammo do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   @doc """ |   @doc """ | ||||||
|   Creates a ammo_group. |   Creates multiple ammo_groups at once. | ||||||
|  |  | ||||||
|   ## Examples |   ## Examples | ||||||
|  |  | ||||||
|       iex> create_ammo_group(%{field: value}, %User{id: 123}) |       iex> create_ammo_groups(%{field: value}, 3, %User{id: 123}) | ||||||
|       {:ok, %AmmoGroup{}} |       {:ok, {3, [%AmmoGroup{}]}} | ||||||
|  |  | ||||||
|       iex> create_ammo_group(%{field: bad_value}, %User{id: 123}) |       iex> create_ammo_groups(%{field: bad_value}, 3, %User{id: 123}) | ||||||
|       {:error, %Changeset{}} |       {:error, %Changeset{}} | ||||||
|  |  | ||||||
|   """ |   """ | ||||||
|   @spec create_ammo_group(attrs :: map(), User.t()) :: |   @spec create_ammo_groups(attrs :: map(), multiplier :: non_neg_integer(), User.t()) :: | ||||||
|           {:ok, AmmoGroup.t()} | {:error, Changeset.t(AmmoGroup.new_ammo_group())} |           {:ok, {count :: non_neg_integer(), [AmmoGroup.t()] | nil}} | ||||||
|   def create_ammo_group( |           | {:error, Changeset.t(AmmoGroup.new_ammo_group())} | ||||||
|  |   def create_ammo_groups( | ||||||
|         %{"ammo_type_id" => ammo_type_id, "container_id" => container_id} = attrs, |         %{"ammo_type_id" => ammo_type_id, "container_id" => container_id} = attrs, | ||||||
|  |         multiplier, | ||||||
|         %User{id: user_id} = user |         %User{id: user_id} = user | ||||||
|       ) do |       ) | ||||||
|  |       when multiplier >= 1 and multiplier <= @ammo_group_create_limit do | ||||||
|     # validate ammo type and container ids belong to user |     # validate ammo type and container ids belong to user | ||||||
|     _valid_ammo_type = get_ammo_type!(ammo_type_id, user) |     _valid_ammo_type = get_ammo_type!(ammo_type_id, user) | ||||||
|     _valid_container = Containers.get_container!(container_id, user) |     _valid_container = Containers.get_container!(container_id, user) | ||||||
|  |  | ||||||
|     %AmmoGroup{} |     now = NaiveDateTime.utc_now() |> NaiveDateTime.truncate(:second) | ||||||
|     |> AmmoGroup.create_changeset(attrs |> Map.put("user_id", user_id)) |  | ||||||
|     |> Repo.insert() |     changesets = | ||||||
|  |       Enum.map(1..multiplier, fn _count -> | ||||||
|  |         %AmmoGroup{} |> AmmoGroup.create_changeset(attrs |> Map.put("user_id", user_id)) | ||||||
|  |       end) | ||||||
|  |  | ||||||
|  |     if changesets |> Enum.all?(fn %{valid?: valid} -> valid end) do | ||||||
|  |       {count, inserted_ammo_groups} = | ||||||
|  |         Repo.insert_all( | ||||||
|  |           AmmoGroup, | ||||||
|  |           changesets | ||||||
|  |           |> Enum.map(fn changeset -> | ||||||
|  |             changeset | ||||||
|  |             |> Map.get(:changes) | ||||||
|  |             |> Map.merge(%{inserted_at: now, updated_at: now}) | ||||||
|  |           end), | ||||||
|  |           returning: true | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |       {:ok, {count, inserted_ammo_groups}} | ||||||
|  |     else | ||||||
|  |       changesets | ||||||
|  |       |> Enum.reject(fn %{valid?: valid} -> valid end) | ||||||
|  |       |> List.first() | ||||||
|  |       |> Changeset.apply_action(:insert) | ||||||
|  |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def create_ammo_group(invalid_attrs, _user) do |   def create_ammo_groups(invalid_attrs, _multiplier, _user) do | ||||||
|     %AmmoGroup{} |     {:error, %AmmoGroup{} |> AmmoGroup.create_changeset(invalid_attrs)} | ||||||
|     |> AmmoGroup.create_changeset(invalid_attrs |> Map.put("user_id", "-1")) |  | ||||||
|     |> Repo.insert() |  | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @doc """ |   @doc """ | ||||||
|   | |||||||
| @@ -215,7 +215,7 @@ defmodule Cannery.Containers do | |||||||
|   def get_container_rounds!(%Container{} = container) do |   def get_container_rounds!(%Container{} = container) do | ||||||
|     container |     container | ||||||
|     |> Repo.preload(:ammo_groups) |     |> Repo.preload(:ammo_groups) | ||||||
|     |> Map.get(:ammo_groups) |     |> Map.fetch!(:ammo_groups) | ||||||
|     |> Enum.map(fn %{count: count} -> count end) |     |> Enum.map(fn %{count: count} -> count end) | ||||||
|     |> Enum.sum() |     |> Enum.sum() | ||||||
|   end |   end | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ defmodule Cannery.Release do | |||||||
|  |  | ||||||
|   def rollback(repo, version) do |   def rollback(repo, version) do | ||||||
|     load_app() |     load_app() | ||||||
|     {:ok, _, _} = Ecto.Migrator.with_repo(repo, &Ecto.Migrator.run(&1, :down, to: version)) |     {:ok, _fun, _opts} = Ecto.Migrator.with_repo(repo, &Ecto.Migrator.run(&1, :down, to: version)) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp load_app do |   defp load_app do | ||||||
| @@ -20,7 +20,7 @@ defmodule Cannery.Release do | |||||||
|     load_app() |     load_app() | ||||||
|  |  | ||||||
|     for repo <- Application.fetch_env!(@app, :ecto_repos) do |     for repo <- Application.fetch_env!(@app, :ecto_repos) do | ||||||
|       {:ok, _, _} = Ecto.Migrator.with_repo(repo, &Ecto.Migrator.run(&1, :up, all: true)) |       {:ok, _fun, _opts} = Ecto.Migrator.with_repo(repo, &Ecto.Migrator.run(&1, :up, all: true)) | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -6,11 +6,11 @@ defmodule Cannery.Repo.Migrator do | |||||||
|   use GenServer |   use GenServer | ||||||
|   require Logger |   require Logger | ||||||
|  |  | ||||||
|   def start_link(_) do |   def start_link(_opts) do | ||||||
|     GenServer.start_link(__MODULE__, [], []) |     GenServer.start_link(__MODULE__, [], []) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def init(_) do |   def init(_opts) do | ||||||
|     migrate!() |     migrate!() | ||||||
|     {:ok, nil} |     {:ok, nil} | ||||||
|   end |   end | ||||||
|   | |||||||
| @@ -71,6 +71,7 @@ defmodule CanneryWeb do | |||||||
|     quote do |     quote do | ||||||
|       use Phoenix.Router |       use Phoenix.Router | ||||||
|  |  | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       import Plug.Conn |       import Plug.Conn | ||||||
|       import Phoenix.Controller |       import Phoenix.Controller | ||||||
|       import Phoenix.LiveView.Router |       import Phoenix.LiveView.Router | ||||||
| @@ -79,7 +80,9 @@ defmodule CanneryWeb do | |||||||
|  |  | ||||||
|   def channel do |   def channel do | ||||||
|     quote do |     quote do | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       use Phoenix.Channel |       use Phoenix.Channel | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       import CanneryWeb.Gettext |       import CanneryWeb.Gettext | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
| @@ -87,14 +90,18 @@ defmodule CanneryWeb do | |||||||
|   defp view_helpers do |   defp view_helpers do | ||||||
|     quote do |     quote do | ||||||
|       # Use all HTML functionality (forms, tags, etc) |       # Use all HTML functionality (forms, tags, etc) | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       use Phoenix.HTML |       use Phoenix.HTML | ||||||
|  |  | ||||||
|       # Import LiveView and .heex helpers (live_render, live_patch, <.form>, etc) |       # Import LiveView and .heex helpers (live_render, live_patch, <.form>, etc) | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       import Phoenix.LiveView.Helpers |       import Phoenix.LiveView.Helpers | ||||||
|  |  | ||||||
|       # Import basic rendering functionality (render, render_layout, etc) |       # Import basic rendering functionality (render, render_layout, etc) | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       import Phoenix.View |       import Phoenix.View | ||||||
|  |  | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       import CanneryWeb.{ErrorHelpers, Gettext, LiveHelpers, ViewHelpers} |       import CanneryWeb.{ErrorHelpers, Gettext, LiveHelpers, ViewHelpers} | ||||||
|       alias CanneryWeb.Router.Helpers, as: Routes |       alias CanneryWeb.Router.Helpers, as: Routes | ||||||
|     end |     end | ||||||
|   | |||||||
| @@ -36,7 +36,11 @@ | |||||||
|     <%= error_tag(f, :notes, "col-span-3") %> |     <%= error_tag(f, :notes, "col-span-3") %> | ||||||
|  |  | ||||||
|     <%= label(f, :date, gettext("Date (UTC)"), class: "title text-lg text-primary-600") %> |     <%= label(f, :date, gettext("Date (UTC)"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= date_input(f, :date, class: "input input-primary col-span-2") %> |     <%= date_input(f, :date, | ||||||
|  |       class: "input input-primary col-span-2", | ||||||
|  |       phx_update: "ignore", | ||||||
|  |       value: Date.utc_today() | ||||||
|  |     ) %> | ||||||
|     <%= error_tag(f, :notes, "col-span-3") %> |     <%= error_tag(f, :notes, "col-span-3") %> | ||||||
|  |  | ||||||
|     <%= submit(dgettext("actions", "Save"), |     <%= submit(dgettext("actions", "Save"), | ||||||
|   | |||||||
| @@ -14,11 +14,11 @@ defmodule CanneryWeb.Components.ContainerCard do | |||||||
|     ~H""" |     ~H""" | ||||||
|     <div |     <div | ||||||
|       id={"container-#{@container.id}"} |       id={"container-#{@container.id}"} | ||||||
|       class="mx-4 my-2 px-8 py-4 flex flex-col justify-center items-center space-y-4 |       class="overflow-hidden max-w-full mx-4 my-2 px-8 py-4 flex flex-col justify-center items-center space-y-4 | ||||||
|         border border-gray-400 rounded-lg shadow-lg hover:shadow-md |         border border-gray-400 rounded-lg shadow-lg hover:shadow-md | ||||||
|         transition-all duration-300 ease-in-out" |         transition-all duration-300 ease-in-out" | ||||||
|     > |     > | ||||||
|       <div class="mb-4 flex flex-col justify-center items-center space-y-2"> |       <div class="max-w-full mb-4 flex flex-col justify-center items-center space-y-2"> | ||||||
|         <%= live_redirect to: Routes.container_show_path(Endpoint, :show, @container), |         <%= live_redirect to: Routes.container_show_path(Endpoint, :show, @container), | ||||||
|                       class: "link" do %> |                       class: "link" do %> | ||||||
|           <h1 class="px-4 py-2 rounded-lg title text-xl"> |           <h1 class="px-4 py-2 rounded-lg title text-xl"> | ||||||
|   | |||||||
| @@ -4,7 +4,8 @@ defmodule CanneryWeb.Components.MoveAmmoGroupComponent do | |||||||
|   """ |   """ | ||||||
|  |  | ||||||
|   use CanneryWeb, :live_component |   use CanneryWeb, :live_component | ||||||
|   alias Cannery.{Accounts.User, Ammo, Ammo.AmmoGroup, Containers} |   alias Cannery.{Accounts.User, Ammo, Ammo.AmmoGroup, Containers, Containers.Container} | ||||||
|  |   alias CanneryWeb.Endpoint | ||||||
|   alias Phoenix.LiveView.Socket |   alias Phoenix.LiveView.Socket | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
| @@ -27,7 +28,12 @@ defmodule CanneryWeb.Components.MoveAmmoGroupComponent do | |||||||
|       Containers.list_containers(current_user) |       Containers.list_containers(current_user) | ||||||
|       |> Enum.reject(fn %{id: id} -> id == container_id end) |       |> Enum.reject(fn %{id: id} -> id == container_id end) | ||||||
|  |  | ||||||
|     {:ok, socket |> assign(assigns) |> assign(changeset: changeset, containers: containers)} |     socket = | ||||||
|  |       socket | ||||||
|  |       |> assign(assigns) | ||||||
|  |       |> assign(changeset: changeset, containers: containers) | ||||||
|  |  | ||||||
|  |     {:ok, socket} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
| @@ -54,4 +60,76 @@ defmodule CanneryWeb.Components.MoveAmmoGroupComponent do | |||||||
|  |  | ||||||
|     {:noreply, socket} |     {:noreply, socket} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  |   @impl true | ||||||
|  |   def render(%{containers: containers} = assigns) do | ||||||
|  |     columns = [ | ||||||
|  |       %{label: gettext("Container"), key: "name"}, | ||||||
|  |       %{label: gettext("Type"), key: "type"}, | ||||||
|  |       %{label: gettext("Location"), key: "location"}, | ||||||
|  |       %{label: nil, key: "actions", sortable: false} | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     rows = containers |> get_rows_for_containers(assigns, columns) | ||||||
|  |  | ||||||
|  |     assigns = assigns |> Map.merge(%{columns: columns, rows: rows}) | ||||||
|  |  | ||||||
|  |     ~H""" | ||||||
|  |     <div class="w-full flex flex-col space-y-8 justify-center items-center"> | ||||||
|  |       <h2 class="mb-8 text-center title text-xl text-primary-600"> | ||||||
|  |         <%= gettext("Move ammo") %> | ||||||
|  |       </h2> | ||||||
|  |  | ||||||
|  |       <%= if @containers |> Enum.empty?() do %> | ||||||
|  |         <h2 class="title text-xl text-primary-600"> | ||||||
|  |           <%= gettext("No other containers") %> | ||||||
|  |           <%= display_emoji("😔") %> | ||||||
|  |         </h2> | ||||||
|  |  | ||||||
|  |         <%= live_patch(dgettext("actions", "Add another container!"), | ||||||
|  |           to: Routes.container_index_path(Endpoint, :new), | ||||||
|  |           class: "btn btn-primary" | ||||||
|  |         ) %> | ||||||
|  |       <% else %> | ||||||
|  |         <.live_component | ||||||
|  |           module={CanneryWeb.Components.TableComponent} | ||||||
|  |           id="move_ammo_group_table" | ||||||
|  |           columns={@columns} | ||||||
|  |           rows={@rows} | ||||||
|  |         /> | ||||||
|  |       <% end %> | ||||||
|  |     </div> | ||||||
|  |     """ | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   @spec get_rows_for_containers([Container.t()], map(), [map()]) :: [map()] | ||||||
|  |   defp get_rows_for_containers(containers, assigns, columns) do | ||||||
|  |     containers | ||||||
|  |     |> Enum.map(fn container -> | ||||||
|  |       columns | ||||||
|  |       |> Enum.into(%{}, fn %{key: key} -> {key, get_row_value_by_key(key, container, assigns)} end) | ||||||
|  |     end) | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   @spec get_row_value_by_key(String.t(), Container.t(), map()) :: any() | ||||||
|  |   defp get_row_value_by_key("actions", container, assigns) do | ||||||
|  |     assigns = assigns |> Map.put(:container, container) | ||||||
|  |  | ||||||
|  |     ~H""" | ||||||
|  |     <div class="px-4 py-2 space-x-4 flex justify-center items-center"> | ||||||
|  |       <button | ||||||
|  |         type="button" | ||||||
|  |         class="btn btn-primary" | ||||||
|  |         phx-click="move" | ||||||
|  |         phx-target={@myself} | ||||||
|  |         phx-value-container_id={container.id} | ||||||
|  |       > | ||||||
|  |         <%= dgettext("actions", "Select") %> | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |     """ | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp get_row_value_by_key(key, container, _assigns), | ||||||
|  |     do: container |> Map.get(key |> String.to_existing_atom()) | ||||||
| end | end | ||||||
|   | |||||||
| @@ -1,70 +0,0 @@ | |||||||
| <div class="w-full flex flex-col space-y-8 justify-center items-center"> |  | ||||||
|   <h2 class="mb-8 text-center title text-xl text-primary-600"> |  | ||||||
|     <%= gettext("Move ammo") %> |  | ||||||
|   </h2> |  | ||||||
|  |  | ||||||
|   <%= if @containers |> Enum.empty?() do %> |  | ||||||
|     <h2 class="title text-xl text-primary-600"> |  | ||||||
|       <%= gettext("No other containers") %> |  | ||||||
|       <%= display_emoji("😔") %> |  | ||||||
|     </h2> |  | ||||||
|  |  | ||||||
|     <%= live_patch(dgettext("actions", "Add another container!"), |  | ||||||
|       to: Routes.container_index_path(Endpoint, :new), |  | ||||||
|       class: "btn btn-primary" |  | ||||||
|     ) %> |  | ||||||
|   <% else %> |  | ||||||
|     <div class="w-full overflow-x-auto border border-gray-600 rounded-lg shadow-lg bg-black"> |  | ||||||
|       <table class="min-w-full table-auto text-center bg-white"> |  | ||||||
|         <thead class="border-b border-primary-600"> |  | ||||||
|           <tr> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Container") %> |  | ||||||
|             </th> |  | ||||||
|  |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Type") %> |  | ||||||
|             </th> |  | ||||||
|  |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Location") %> |  | ||||||
|             </th> |  | ||||||
|  |  | ||||||
|             <th class="p-2"></th> |  | ||||||
|           </tr> |  | ||||||
|         </thead> |  | ||||||
|         <tbody id="containers"> |  | ||||||
|           <%= for container <- @containers do %> |  | ||||||
|             <tr id={"container-#{container.id}"}> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= container.name %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= container.type %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= container.location %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <div class="px-4 py-2 space-x-4 flex justify-center items-center"> |  | ||||||
|                   <button |  | ||||||
|                     type="button" |  | ||||||
|                     class="btn btn-primary" |  | ||||||
|                     phx-click="move" |  | ||||||
|                     phx-target={@myself} |  | ||||||
|                     phx-value-container_id={container.id} |  | ||||||
|                   > |  | ||||||
|                     <%= dgettext("actions", "Select") %> |  | ||||||
|                   </button> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           <% end %> |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|   <% end %> |  | ||||||
| </div> |  | ||||||
							
								
								
									
										87
									
								
								lib/cannery_web/components/table_component.ex
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								lib/cannery_web/components/table_component.ex
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | |||||||
|  | defmodule CanneryWeb.Components.TableComponent do | ||||||
|  |   @moduledoc """ | ||||||
|  |   Livecomponent that presents a resortable table | ||||||
|  |  | ||||||
|  |   It takes the following required assigns: | ||||||
|  |     - `:columns`: An array of maps containing the following keys | ||||||
|  |       - `:label`: A gettext'd or otherwise user-facing string label for the | ||||||
|  |         column. Can be nil | ||||||
|  |       - `:key`: A string key used for sorting | ||||||
|  |       - `:class`: Extra classes to be applied to the column element, if desired. | ||||||
|  |         Optional | ||||||
|  |       - `:sortable`: If false, will prevent the user from sorting with it. | ||||||
|  |         Optional | ||||||
|  |     - `:values`: An array of maps containing data for each row. Each map is | ||||||
|  |       string-keyed with the associated column key to the following values: | ||||||
|  |       - A single element, like string, integer or Phoenix.LiveView.Rendered | ||||||
|  |         object, like returned from the ~H sigil | ||||||
|  |       - A tuple, containing a custom value used for sorting, and the displayed | ||||||
|  |         content. | ||||||
|  |   """ | ||||||
|  |  | ||||||
|  |   use CanneryWeb, :live_component | ||||||
|  |   alias Phoenix.LiveView.Socket | ||||||
|  |  | ||||||
|  |   @impl true | ||||||
|  |   @spec update( | ||||||
|  |           %{ | ||||||
|  |             required(:columns) => | ||||||
|  |               list(%{ | ||||||
|  |                 required(:label) => String.t() | nil, | ||||||
|  |                 required(:key) => String.t() | nil, | ||||||
|  |                 optional(:class) => String.t(), | ||||||
|  |                 optional(:sortable) => false | ||||||
|  |               }), | ||||||
|  |             required(:rows) => | ||||||
|  |               list(%{ | ||||||
|  |                 (key :: String.t()) => any() | {custom_sort_value :: String.t(), value :: any()} | ||||||
|  |               }), | ||||||
|  |             optional(any()) => any() | ||||||
|  |           }, | ||||||
|  |           Socket.t() | ||||||
|  |         ) :: {:ok, Socket.t()} | ||||||
|  |   def update(%{columns: columns, rows: rows} = assigns, socket) do | ||||||
|  |     initial_key = columns |> List.first() |> Map.get(:key) | ||||||
|  |     rows = rows |> Enum.sort_by(fn row -> row |> Map.get(initial_key) end, :asc) | ||||||
|  |  | ||||||
|  |     socket = | ||||||
|  |       socket | ||||||
|  |       |> assign(assigns) | ||||||
|  |       |> assign(columns: columns, rows: rows, last_sort_key: initial_key, sort_mode: :asc) | ||||||
|  |  | ||||||
|  |     {:ok, socket} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   @impl true | ||||||
|  |   def handle_event( | ||||||
|  |         "sort_by", | ||||||
|  |         %{"sort-key" => key}, | ||||||
|  |         %{assigns: %{rows: rows, last_sort_key: key, sort_mode: sort_mode}} = socket | ||||||
|  |       ) do | ||||||
|  |     sort_mode = if sort_mode == :asc, do: :desc, else: :asc | ||||||
|  |     rows = rows |> sort_by_custom_sort_value_or_value(key, sort_mode) | ||||||
|  |     {:noreply, socket |> assign(sort_mode: sort_mode, rows: rows)} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   def handle_event( | ||||||
|  |         "sort_by", | ||||||
|  |         %{"sort-key" => key}, | ||||||
|  |         %{assigns: %{rows: rows}} = socket | ||||||
|  |       ) do | ||||||
|  |     rows = rows |> sort_by_custom_sort_value_or_value(key, :asc) | ||||||
|  |     {:noreply, socket |> assign(last_sort_key: key, sort_mode: :asc, rows: rows)} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp sort_by_custom_sort_value_or_value(rows, key, sort_mode) do | ||||||
|  |     rows | ||||||
|  |     |> Enum.sort_by( | ||||||
|  |       fn row -> | ||||||
|  |         case row |> Map.get(key) do | ||||||
|  |           {custom_sort_key, _value} -> custom_sort_key | ||||||
|  |           value -> value | ||||||
|  |         end | ||||||
|  |       end, | ||||||
|  |       sort_mode | ||||||
|  |     ) | ||||||
|  |   end | ||||||
|  | end | ||||||
							
								
								
									
										52
									
								
								lib/cannery_web/components/table_component.html.heex
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								lib/cannery_web/components/table_component.html.heex
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | <div class="w-full overflow-x-auto border border-gray-600 rounded-lg shadow-lg bg-black"> | ||||||
|  |   <table class="min-w-full table-auto text-center bg-white"> | ||||||
|  |     <thead class="border-b border-primary-600"> | ||||||
|  |       <tr> | ||||||
|  |         <%= for %{key: key, label: label} = column <- @columns do %> | ||||||
|  |           <%= if column |> Map.get(:sortable, true) do %> | ||||||
|  |             <th class={"p-2 #{column[:class]}"}> | ||||||
|  |               <span | ||||||
|  |                 class="cursor-pointer" | ||||||
|  |                 phx-click="sort_by" | ||||||
|  |                 phx-value-sort-key={key} | ||||||
|  |                 phx-target={@myself} | ||||||
|  |               > | ||||||
|  |                 <span class="underline"><%= label %></span> | ||||||
|  |                 <%= if @last_sort_key == key do %> | ||||||
|  |                   <%= case @sort_mode do %> | ||||||
|  |                     <% :asc -> %> | ||||||
|  |                       <i class="fas fa-sm fa-chevron-down"></i> | ||||||
|  |                     <% :desc -> %> | ||||||
|  |                       <i class="fas fa-sm fa-chevron-up"></i> | ||||||
|  |                   <% end %> | ||||||
|  |                 <% else %> | ||||||
|  |                   <i class="fas fa-sm fa-chevron-up opacity-0"></i> | ||||||
|  |                 <% end %> | ||||||
|  |               </span> | ||||||
|  |             </th> | ||||||
|  |           <% else %> | ||||||
|  |             <th class={"p-2 #{column[:class]}"}> | ||||||
|  |               <%= label %> | ||||||
|  |             </th> | ||||||
|  |           <% end %> | ||||||
|  |         <% end %> | ||||||
|  |       </tr> | ||||||
|  |     </thead> | ||||||
|  |     <tbody> | ||||||
|  |       <%= for values <- @rows do %> | ||||||
|  |         <tr> | ||||||
|  |           <%= for %{key: key} = value <- @columns do %> | ||||||
|  |             <td class={"p-2 #{value[:class]}"}> | ||||||
|  |               <%= case values |> Map.get(key) do %> | ||||||
|  |                 <% {_custom_sort_value, value} -> %> | ||||||
|  |                   <%= value %> | ||||||
|  |                 <% value -> %> | ||||||
|  |                   <%= value %> | ||||||
|  |               <% end %> | ||||||
|  |             </td> | ||||||
|  |           <% end %> | ||||||
|  |         </tr> | ||||||
|  |       <% end %> | ||||||
|  |     </tbody> | ||||||
|  |   </table> | ||||||
|  | </div> | ||||||
| @@ -22,7 +22,7 @@ defmodule CanneryWeb.Components.TagCard do | |||||||
|   def simple_tag_card(assigns) do |   def simple_tag_card(assigns) do | ||||||
|     ~H""" |     ~H""" | ||||||
|     <h1 |     <h1 | ||||||
|       class="mx-2 my-1 px-4 py-2 rounded-lg title text-xl" |       class="inline-block break-all mx-2 my-1 px-4 py-2 rounded-lg title text-xl" | ||||||
|       style={"color: #{@tag.text_color}; background-color: #{@tag.bg_color}"} |       style={"color: #{@tag.text_color}; background-color: #{@tag.bg_color}"} | ||||||
|     > |     > | ||||||
|       <%= @tag.name %> |       <%= @tag.name %> | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| defmodule CanneryWeb.UserConfirmationController do | defmodule CanneryWeb.UserConfirmationController do | ||||||
|   use CanneryWeb, :controller |   use CanneryWeb, :controller | ||||||
|  |  | ||||||
|  |   import CanneryWeb.Gettext | ||||||
|   alias Cannery.Accounts |   alias Cannery.Accounts | ||||||
|  |  | ||||||
|   def new(conn, _params) do |   def new(conn, _params) do | ||||||
|     render(conn, "new.html") |     render(conn, "new.html", page_title: gettext("Confirm your account")) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def create(conn, %{"user" => %{"email" => email}}) do |   def create(conn, %{"user" => %{"email" => email}}) do | ||||||
|   | |||||||
| @@ -29,8 +29,11 @@ defmodule CanneryWeb.UserRegistrationController do | |||||||
|  |  | ||||||
|   # renders new user registration page |   # renders new user registration page | ||||||
|   defp render_new(conn, invite \\ nil) do |   defp render_new(conn, invite \\ nil) do | ||||||
|     changeset = Accounts.change_user_registration(%User{}) |     render(conn, "new.html", | ||||||
|     conn |> render("new.html", changeset: changeset, invite: invite) |       changeset: Accounts.change_user_registration(%User{}), | ||||||
|  |       invite: invite, | ||||||
|  |       page_title: gettext("Register") | ||||||
|  |     ) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def create(conn, %{"user" => %{"invite_token" => invite_token}} = attrs) do |   def create(conn, %{"user" => %{"invite_token" => invite_token}} = attrs) do | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ defmodule CanneryWeb.UserResetPasswordController do | |||||||
|   plug :get_user_by_reset_password_token when action in [:edit, :update] |   plug :get_user_by_reset_password_token when action in [:edit, :update] | ||||||
|  |  | ||||||
|   def new(conn, _params) do |   def new(conn, _params) do | ||||||
|     render(conn, "new.html") |     render(conn, "new.html", page_title: gettext("Forgot your password?")) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def create(conn, %{"user" => %{"email" => email}}) do |   def create(conn, %{"user" => %{"email" => email}}) do | ||||||
| @@ -31,7 +31,10 @@ defmodule CanneryWeb.UserResetPasswordController do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   def edit(conn, _params) do |   def edit(conn, _params) do | ||||||
|     render(conn, "edit.html", changeset: Accounts.change_user_password(conn.assigns.user)) |     render(conn, "edit.html", | ||||||
|  |       changeset: Accounts.change_user_password(conn.assigns.user), | ||||||
|  |       page_title: gettext("Reset your password") | ||||||
|  |     ) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   # Do not log in the user after reset password to avoid a |   # Do not log in the user after reset password to avoid a | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ defmodule CanneryWeb.UserSessionController do | |||||||
|   alias CanneryWeb.UserAuth |   alias CanneryWeb.UserAuth | ||||||
|  |  | ||||||
|   def new(conn, _params) do |   def new(conn, _params) do | ||||||
|     render(conn, "new.html", error_message: nil) |     render(conn, "new.html", error_message: nil, page_title: gettext("Log in")) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def create(conn, %{"user" => user_params}) do |   def create(conn, %{"user" => user_params}) do | ||||||
|   | |||||||
| @@ -1,13 +1,13 @@ | |||||||
| defmodule CanneryWeb.UserSettingsController do | defmodule CanneryWeb.UserSettingsController do | ||||||
|   use CanneryWeb, :controller |   use CanneryWeb, :controller | ||||||
|  |   import CanneryWeb.Gettext | ||||||
|   alias Cannery.Accounts |   alias Cannery.Accounts | ||||||
|   alias CanneryWeb.{HomeLive, UserAuth} |   alias CanneryWeb.{HomeLive, UserAuth} | ||||||
|  |  | ||||||
|   plug :assign_email_and_password_changesets |   plug :assign_email_and_password_changesets | ||||||
|  |  | ||||||
|   def edit(conn, _params) do |   def edit(conn, _params) do | ||||||
|     render(conn, "edit.html") |     render(conn, "edit.html", page_title: gettext("Settings")) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def update(conn, %{"action" => "update_email"} = params) do |   def update(conn, %{"action" => "update_email"} = params) do | ||||||
|   | |||||||
| @@ -9,6 +9,8 @@ defmodule CanneryWeb.AmmoGroupLive.FormComponent do | |||||||
|   alias Ecto.Changeset |   alias Ecto.Changeset | ||||||
|   alias Phoenix.LiveView.Socket |   alias Phoenix.LiveView.Socket | ||||||
|  |  | ||||||
|  |   @ammo_group_create_limit 10_000 | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   @spec update( |   @spec update( | ||||||
|           %{:ammo_group => AmmoGroup.t(), :current_user => User.t(), optional(any) => any}, |           %{:ammo_group => AmmoGroup.t(), :current_user => User.t(), optional(any) => any}, | ||||||
| @@ -20,20 +22,37 @@ defmodule CanneryWeb.AmmoGroupLive.FormComponent do | |||||||
|  |  | ||||||
|   @spec update(Socket.t()) :: {:ok, Socket.t()} |   @spec update(Socket.t()) :: {:ok, Socket.t()} | ||||||
|   def update(%{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket) do |   def update(%{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket) do | ||||||
|     changeset = Ammo.change_ammo_group(ammo_group) |     socket = | ||||||
|     containers = Containers.list_containers(current_user) |       socket | ||||||
|     ammo_types = Ammo.list_ammo_types(current_user) |       |> assign(:ammo_group_create_limit, @ammo_group_create_limit) | ||||||
|     {:ok, socket |> assign(changeset: changeset, containers: containers, ammo_types: ammo_types)} |       |> assign(:changeset, Ammo.change_ammo_group(ammo_group)) | ||||||
|  |       |> assign(:ammo_types, Ammo.list_ammo_types(current_user)) | ||||||
|  |       |> assign_new(:containers, fn -> Containers.list_containers(current_user) end) | ||||||
|  |  | ||||||
|  |     {:ok, socket} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_event( |   def handle_event( | ||||||
|         "validate", |         "validate", | ||||||
|         %{"ammo_group" => ammo_group_params}, |         %{"ammo_group" => ammo_group_params}, | ||||||
|         %{assigns: %{ammo_group: ammo_group}} = socket |         %{assigns: %{action: action, ammo_group: ammo_group}} = socket | ||||||
|       ) do |       ) do | ||||||
|     socket = socket |> assign(:changeset, ammo_group |> Ammo.change_ammo_group(ammo_group_params)) |     changeset_action = | ||||||
|     {:noreply, socket} |       case action do | ||||||
|  |         :new -> :insert | ||||||
|  |         :edit -> :update | ||||||
|  |       end | ||||||
|  |  | ||||||
|  |     changeset = ammo_group |> Ammo.change_ammo_group(ammo_group_params) | ||||||
|  |  | ||||||
|  |     changeset = | ||||||
|  |       case changeset |> Changeset.apply_action(changeset_action) do | ||||||
|  |         {:ok, _data} -> changeset | ||||||
|  |         {:error, changeset} -> changeset | ||||||
|  |       end | ||||||
|  |  | ||||||
|  |     {:noreply, socket |> assign(:changeset, changeset)} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   def handle_event( |   def handle_event( | ||||||
| @@ -77,20 +96,65 @@ defmodule CanneryWeb.AmmoGroupLive.FormComponent do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp save_ammo_group( |   defp save_ammo_group( | ||||||
|          %{assigns: %{current_user: current_user, return_to: return_to}} = socket, |          %{assigns: %{changeset: changeset}} = socket, | ||||||
|          :new, |          :new, | ||||||
|          ammo_group_params |          %{"multiplier" => multiplier_str} = ammo_group_params | ||||||
|        ) do |        ) do | ||||||
|     socket = |     socket = | ||||||
|       case Ammo.create_ammo_group(ammo_group_params, current_user) do |       case multiplier_str |> Integer.parse() do | ||||||
|         {:ok, _ammo_group} -> |         {multiplier, _remainder} | ||||||
|           prompt = dgettext("prompts", "Ammo group created successfully") |         when multiplier >= 1 and multiplier <= @ammo_group_create_limit -> | ||||||
|           socket |> put_flash(:info, prompt) |> push_redirect(to: return_to) |           socket |> create_multiple(ammo_group_params, multiplier) | ||||||
|  |  | ||||||
|         {:error, %Changeset{} = changeset} -> |         {multiplier, _remainder} -> | ||||||
|           socket |> assign(changeset: changeset) |           error_msg = | ||||||
|  |             dgettext( | ||||||
|  |               "errors", | ||||||
|  |               "Invalid number of copies, must be between 1 and %{max}. Was %{multiplier}", | ||||||
|  |               max: @ammo_group_create_limit, | ||||||
|  |               multiplier: multiplier | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |           {:error, changeset} = | ||||||
|  |             changeset | ||||||
|  |             |> Changeset.add_error(:multiplier, error_msg) | ||||||
|  |             |> Changeset.apply_action(:insert) | ||||||
|  |  | ||||||
|  |           socket |> assign(:changeset, changeset) | ||||||
|  |  | ||||||
|  |         :error -> | ||||||
|  |           error_msg = dgettext("errors", "Could not parse number of copies") | ||||||
|  |  | ||||||
|  |           {:error, changeset} = | ||||||
|  |             changeset | ||||||
|  |             |> Changeset.add_error(:multiplier, error_msg) | ||||||
|  |             |> Changeset.apply_action(:insert) | ||||||
|  |  | ||||||
|  |           socket |> assign(:changeset, changeset) | ||||||
|       end |       end | ||||||
|  |  | ||||||
|     {:noreply, socket} |     {:noreply, socket} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  |   defp create_multiple( | ||||||
|  |          %{assigns: %{current_user: current_user, return_to: return_to}} = socket, | ||||||
|  |          ammo_group_params, | ||||||
|  |          multiplier | ||||||
|  |        ) do | ||||||
|  |     case Ammo.create_ammo_groups(ammo_group_params, multiplier, current_user) do | ||||||
|  |       {:ok, {count, _ammo_groups}} -> | ||||||
|  |         prompt = | ||||||
|  |           dngettext( | ||||||
|  |             "prompts", | ||||||
|  |             "Ammo group created successfully", | ||||||
|  |             "Ammo groups created successfully", | ||||||
|  |             count | ||||||
|  |           ) | ||||||
|  |  | ||||||
|  |         socket |> put_flash(:info, prompt) |> push_redirect(to: return_to) | ||||||
|  |  | ||||||
|  |       {:error, %Changeset{} = changeset} -> | ||||||
|  |         socket |> assign(changeset: changeset) | ||||||
|  |     end | ||||||
|  |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -18,42 +18,62 @@ | |||||||
|       </div> |       </div> | ||||||
|     <% end %> |     <% end %> | ||||||
|  |  | ||||||
|     <%= label(f, :ammo_type_id, gettext("Ammo type"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :ammo_type_id, gettext("Ammo type"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= select(f, :ammo_type_id, ammo_type_options(@ammo_types), |     <%= select(f, :ammo_type_id, ammo_type_options(@ammo_types), | ||||||
|       class: "text-center col-span-2 input input-primary" |       class: "text-center col-span-2 input input-primary" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :ammo_type_id, "col-span-3 text-center") %> |     <%= error_tag(f, :ammo_type_id, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :count, gettext("Count"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :count, gettext("Count"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= number_input(f, :count, |     <%= number_input(f, :count, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       min: 1 |       min: 1 | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :count, "col-span-3 text-center") %> |     <%= error_tag(f, :count, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :price_paid, gettext("Price paid"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :price_paid, gettext("Price paid"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= number_input(f, :price_paid, |     <%= number_input(f, :price_paid, | ||||||
|       step: "0.01", |       step: 0.01, | ||||||
|       class: "text-center col-span-2 input input-primary" |       class: "text-center col-span-2 input input-primary" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :price_paid, "col-span-3 text-center") %> |     <%= error_tag(f, :price_paid, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :notes, gettext("Notes"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :notes, gettext("Notes"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= textarea(f, :notes, |     <%= textarea(f, :notes, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       phx_hook: "MaintainAttrs" |       phx_hook: "MaintainAttrs" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :notes, "col-span-3 text-center") %> |     <%= error_tag(f, :notes, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :container, gettext("Container"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :container, gettext("Container"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= select(f, :container_id, container_options(@containers), |     <%= select(f, :container_id, container_options(@containers), | ||||||
|       class: "text-center col-span-2 input input-primary" |       class: "text-center col-span-2 input input-primary" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :container_id, "col-span-3 text-center") %> |     <%= error_tag(f, :container_id, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= submit(dgettext("actions", "Save"), |     <%= case @action do %> | ||||||
|       phx_disable_with: dgettext("prompts", "Saving..."), |       <% :new -> %> | ||||||
|       class: "mx-auto col-span-3 btn btn-primary" |         <hr class="hr col-span-3" /> | ||||||
|     ) %> |  | ||||||
|  |         <%= label(f, :multiplier, gettext("Copies"), class: "title text-lg text-primary-600") %> | ||||||
|  |         <%= number_input(f, :multiplier, | ||||||
|  |           max: @ammo_group_create_limit, | ||||||
|  |           class: "text-center input input-primary", | ||||||
|  |           value: 1, | ||||||
|  |           phx_update: "ignore" | ||||||
|  |         ) %> | ||||||
|  |  | ||||||
|  |         <%= submit(dgettext("actions", "Create"), | ||||||
|  |           phx_disable_with: dgettext("prompts", "Creating..."), | ||||||
|  |           class: "mx-auto btn btn-primary" | ||||||
|  |         ) %> | ||||||
|  |  | ||||||
|  |         <%= error_tag(f, :multiplier, "col-span-3 text-center") %> | ||||||
|  |       <% :edit -> %> | ||||||
|  |         <%= submit(dgettext("actions", "Save"), | ||||||
|  |           phx_disable_with: dgettext("prompts", "Saving..."), | ||||||
|  |           class: "mx-auto col-span-3 btn btn-primary" | ||||||
|  |         ) %> | ||||||
|  |     <% end %> | ||||||
|   </.form> |   </.form> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ defmodule CanneryWeb.AmmoGroupLive.Index do | |||||||
|   """ |   """ | ||||||
|  |  | ||||||
|   use CanneryWeb, :live_view |   use CanneryWeb, :live_view | ||||||
|   alias Cannery.{Ammo, Ammo.AmmoGroup, Repo} |   alias Cannery.{Ammo, Ammo.AmmoGroup, Containers, Repo} | ||||||
|   alias CanneryWeb.Endpoint |   alias CanneryWeb.Endpoint | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
| @@ -17,9 +17,11 @@ defmodule CanneryWeb.AmmoGroupLive.Index do | |||||||
|     {:noreply, apply_action(socket, live_action, params)} |     {:noreply, apply_action(socket, live_action, params)} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp apply_action(%{assigns: %{current_user: current_user}} = socket, :add_shot_group, %{ |   defp apply_action( | ||||||
|          "id" => id |          %{assigns: %{current_user: current_user}} = socket, | ||||||
|        }) do |          :add_shot_group, | ||||||
|  |          %{"id" => id} | ||||||
|  |        ) do | ||||||
|     socket |     socket | ||||||
|     |> assign(:page_title, gettext("Record shots")) |     |> assign(:page_title, gettext("Record shots")) | ||||||
|     |> assign(:ammo_group, Ammo.get_ammo_group!(id, current_user)) |     |> assign(:ammo_group, Ammo.get_ammo_group!(id, current_user)) | ||||||
| @@ -72,6 +74,121 @@ defmodule CanneryWeb.AmmoGroupLive.Index do | |||||||
|  |  | ||||||
|   defp display_ammo_groups(%{assigns: %{current_user: current_user}} = socket) do |   defp display_ammo_groups(%{assigns: %{current_user: current_user}} = socket) do | ||||||
|     ammo_groups = Ammo.list_ammo_groups(current_user) |> Repo.preload([:ammo_type, :container]) |     ammo_groups = Ammo.list_ammo_groups(current_user) |> Repo.preload([:ammo_type, :container]) | ||||||
|     socket |> assign(:ammo_groups, ammo_groups) |     containers = Containers.list_containers(current_user) | ||||||
|  |  | ||||||
|  |     columns = [ | ||||||
|  |       %{label: gettext("Ammo type"), key: "ammo_type"}, | ||||||
|  |       %{label: gettext("Count"), key: "count"}, | ||||||
|  |       %{label: gettext("Price paid"), key: "price_paid"}, | ||||||
|  |       %{label: gettext("% left"), key: "remaining"}, | ||||||
|  |       %{label: gettext("Range"), key: "range"}, | ||||||
|  |       %{label: gettext("Container"), key: "container"}, | ||||||
|  |       %{label: nil, key: "actions", sortable: false} | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     rows = | ||||||
|  |       ammo_groups | ||||||
|  |       |> Enum.map(fn ammo_group -> ammo_group |> get_row_data_for_ammo_group(columns) end) | ||||||
|  |  | ||||||
|  |     socket | ||||||
|  |     |> assign(ammo_groups: ammo_groups, containers: containers, columns: columns, rows: rows) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  |   @spec get_row_data_for_ammo_group(AmmoGroup.t(), [map()]) :: [map()] | ||||||
|  |   defp get_row_data_for_ammo_group(ammo_group, columns) do | ||||||
|  |     ammo_group = ammo_group |> Repo.preload([:ammo_type, :container]) | ||||||
|  |  | ||||||
|  |     columns | ||||||
|  |     |> Enum.into(%{}, fn %{key: key} -> {key, get_value_for_key(key, ammo_group)} end) | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   @spec get_value_for_key(String.t(), AmmoGroup.t()) :: any() | ||||||
|  |   defp get_value_for_key("ammo_type", %{ammo_type: ammo_type}) do | ||||||
|  |     {ammo_type.name, | ||||||
|  |      live_patch(ammo_type.name, | ||||||
|  |        to: Routes.ammo_type_show_path(Endpoint, :show, ammo_type), | ||||||
|  |        class: "link" | ||||||
|  |      )} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("price_paid", %{price_paid: nil}), do: {"a", nil} | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("price_paid", %{price_paid: price_paid}), | ||||||
|  |     do: gettext("$%{amount}", amount: price_paid |> :erlang.float_to_binary(decimals: 2)) | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("range", %{staged: staged} = ammo_group) do | ||||||
|  |     assigns = %{ammo_group: ammo_group} | ||||||
|  |  | ||||||
|  |     {staged, | ||||||
|  |      ~H""" | ||||||
|  |      <div class="min-w-20 py-2 px-4 h-full flex flex-col justify-center items-center"> | ||||||
|  |        <button | ||||||
|  |          type="button" | ||||||
|  |          class="mx-2 my-1 btn btn-primary" | ||||||
|  |          phx-click="toggle_staged" | ||||||
|  |          phx-value-ammo_group_id={ammo_group.id} | ||||||
|  |        > | ||||||
|  |          <%= if ammo_group.staged, do: gettext("Unstage"), else: gettext("Stage") %> | ||||||
|  |        </button> | ||||||
|  |  | ||||||
|  |        <%= live_patch(dgettext("actions", "Record shots"), | ||||||
|  |          to: Routes.ammo_group_index_path(Endpoint, :add_shot_group, ammo_group), | ||||||
|  |          class: "mx-2 my-1 btn btn-primary" | ||||||
|  |        ) %> | ||||||
|  |      </div> | ||||||
|  |      """} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("remaining", ammo_group), | ||||||
|  |     do: "#{ammo_group |> Ammo.get_percentage_remaining()}%" | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("actions", ammo_group) do | ||||||
|  |     assigns = %{ammo_group: ammo_group} | ||||||
|  |  | ||||||
|  |     ~H""" | ||||||
|  |     <div class="py-2 px-4 h-full space-x-4 flex justify-center items-center"> | ||||||
|  |       <%= live_redirect to: Routes.ammo_group_show_path(Endpoint, :show, ammo_group), | ||||||
|  |                     class: "text-primary-600 link", | ||||||
|  |                     data: [qa: "view-#{ammo_group.id}"] do %> | ||||||
|  |         <i class="fa-fw fa-lg fas fa-eye"></i> | ||||||
|  |       <% end %> | ||||||
|  |  | ||||||
|  |       <%= live_patch to: Routes.ammo_group_index_path(Endpoint, :edit, ammo_group), | ||||||
|  |                   class: "text-primary-600 link", | ||||||
|  |                   data: [qa: "edit-#{ammo_group.id}"] do %> | ||||||
|  |         <i class="fa-fw fa-lg fas fa-edit"></i> | ||||||
|  |       <% end %> | ||||||
|  |  | ||||||
|  |       <%= link to: "#", | ||||||
|  |             class: "text-primary-600 link", | ||||||
|  |             phx_click: "delete", | ||||||
|  |             phx_value_id: ammo_group.id, | ||||||
|  |             data: [ | ||||||
|  |               confirm: dgettext("prompts", "Are you sure you want to delete this ammo?"), | ||||||
|  |               qa: "delete-#{ammo_group.id}" | ||||||
|  |             ] do %> | ||||||
|  |         <i class="fa-fw fa-lg fas fa-trash"></i> | ||||||
|  |       <% end %> | ||||||
|  |     </div> | ||||||
|  |     """ | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("container", %{container: nil}), do: {nil, nil} | ||||||
|  |  | ||||||
|  |   defp get_value_for_key("container", %{container: %{name: container_name}} = ammo_group) do | ||||||
|  |     assigns = %{ammo_group: ammo_group} | ||||||
|  |  | ||||||
|  |     {container_name, | ||||||
|  |      ~H""" | ||||||
|  |      <div class="min-w-20 py-2 px-4 h-full space-x-4 flex justify-center items-center"> | ||||||
|  |        <%= live_patch(@ammo_group.container.name, | ||||||
|  |          to: Routes.ammo_group_index_path(Endpoint, :move, @ammo_group), | ||||||
|  |          class: "btn btn-primary" | ||||||
|  |        ) %> | ||||||
|  |      </div> | ||||||
|  |      """} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp get_value_for_key(key, ammo_group), | ||||||
|  |     do: ammo_group |> Map.get(key |> String.to_existing_atom()) | ||||||
| end | end | ||||||
|   | |||||||
| @@ -9,126 +9,49 @@ | |||||||
|       <%= display_emoji("😔") %> |       <%= display_emoji("😔") %> | ||||||
|     </h2> |     </h2> | ||||||
|  |  | ||||||
|     <%= live_patch(dgettext("actions", "Add your first box!"), |     <%= if @containers |> Enum.empty?() do %> | ||||||
|       to: Routes.ammo_group_index_path(Endpoint, :new), |       <div class="flex justify-center items-center"> | ||||||
|       class: "btn btn-primary" |         <h2 class="m-2 title text-md text-primary-600"> | ||||||
|     ) %> |           <%= dgettext("prompts", "You'll need to") %> | ||||||
|  |         </h2> | ||||||
|  |  | ||||||
|  |         <%= live_patch(dgettext("actions", "add a container first"), | ||||||
|  |           to: Routes.container_index_path(Endpoint, :new), | ||||||
|  |           class: "btn btn-primary" | ||||||
|  |         ) %> | ||||||
|  |       </div> | ||||||
|  |     <% else %> | ||||||
|  |       <%= live_patch(dgettext("actions", "Add your first box!"), | ||||||
|  |         to: Routes.ammo_group_index_path(Endpoint, :new), | ||||||
|  |         class: "btn btn-primary" | ||||||
|  |       ) %> | ||||||
|  |     <% end %> | ||||||
|   <% else %> |   <% else %> | ||||||
|     <%= live_patch(dgettext("actions", "New Ammo group"), |     <%= if @containers |> Enum.empty?() do %> | ||||||
|       to: Routes.ammo_group_index_path(Endpoint, :new), |       <div class="flex justify-center items-center"> | ||||||
|       class: "btn btn-primary" |         <h2 class="m-2 title text-md text-primary-600"> | ||||||
|     ) %> |           <%= dgettext("prompts", "You'll need to") %> | ||||||
|  |         </h2> | ||||||
|  |  | ||||||
|     <div class="w-full overflow-x-auto border border-gray-600 rounded-lg shadow-lg bg-black"> |         <%= live_patch(dgettext("actions", "add a container first"), | ||||||
|       <table class="min-w-full table-auto text-center bg-white"> |           to: Routes.container_index_path(Endpoint, :new), | ||||||
|         <thead class="border-b border-primary-600"> |           class: "btn btn-primary" | ||||||
|           <tr> |         ) %> | ||||||
|             <th class="p-2"> |       </div> | ||||||
|               <%= gettext("Ammo type") %> |     <% else %> | ||||||
|             </th> |       <%= live_patch(dgettext("actions", "New Ammo group"), | ||||||
|             <th class="p-2"> |         to: Routes.ammo_group_index_path(Endpoint, :new), | ||||||
|               <%= gettext("Count") %> |         class: "btn btn-primary" | ||||||
|             </th> |       ) %> | ||||||
|             <th class="p-2"> |     <% end %> | ||||||
|               <%= gettext("Price paid") %> |  | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("% left") %> |  | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Range") %> |  | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Container") %> |  | ||||||
|             </th> |  | ||||||
|  |  | ||||||
|             <th class="p-2"></th> |     <.live_component | ||||||
|           </tr> |       module={CanneryWeb.Components.TableComponent} | ||||||
|         </thead> |       id="ammo_groups_index_table" | ||||||
|         <tbody id="ammo_groups"> |       action={@live_action} | ||||||
|           <%= for ammo_group <- @ammo_groups do %> |       columns={@columns} | ||||||
|             <tr id={"ammo_group-#{ammo_group.id}"}> |       rows={@rows} | ||||||
|               <td class="p-2"> |     /> | ||||||
|                 <%= live_patch(ammo_group.ammo_type.name, |  | ||||||
|                   to: Routes.ammo_type_show_path(Endpoint, :show, ammo_group.ammo_type), |  | ||||||
|                   class: "link" |  | ||||||
|                 ) %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= ammo_group.count %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= if ammo_group.price_paid do %> |  | ||||||
|                   <%= gettext("$%{amount}", |  | ||||||
|                     amount: ammo_group.price_paid |> :erlang.float_to_binary(decimals: 2) |  | ||||||
|                   ) %> |  | ||||||
|                 <% end %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= "#{ammo_group |> Ammo.get_percentage_remaining()}%" %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <div class="px-4 py-2 space-x-4 flex justify-center items-center"> |  | ||||||
|                   <button |  | ||||||
|                     type="button" |  | ||||||
|                     class="btn btn-primary" |  | ||||||
|                     phx-click="toggle_staged" |  | ||||||
|                     phx-value-ammo_group_id={ammo_group.id} |  | ||||||
|                   > |  | ||||||
|                     <%= if ammo_group.staged, do: gettext("Unstage"), else: gettext("Stage") %> |  | ||||||
|                   </button> |  | ||||||
|  |  | ||||||
|                   <%= live_patch(dgettext("actions", "Record shots"), |  | ||||||
|                     to: Routes.ammo_group_index_path(Endpoint, :add_shot_group, ammo_group), |  | ||||||
|                     class: "btn btn-primary" |  | ||||||
|                   ) %> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= if ammo_group.container do %> |  | ||||||
|                   <%= live_patch(ammo_group.container.name, |  | ||||||
|                     to: Routes.ammo_group_index_path(Endpoint, :move, ammo_group), |  | ||||||
|                     class: "btn btn-primary" |  | ||||||
|                   ) %> |  | ||||||
|                 <% end %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <div class="px-4 py-2 space-x-4 flex justify-center items-center"> |  | ||||||
|                   <%= live_redirect to: Routes.ammo_group_show_path(Endpoint, :show, ammo_group), |  | ||||||
|                                 class: "text-primary-600 link", |  | ||||||
|                                 data: [qa: "view-#{ammo_group.id}"] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-eye"></i> |  | ||||||
|                   <% end %> |  | ||||||
|  |  | ||||||
|                   <%= live_patch to: Routes.ammo_group_index_path(Endpoint, :edit, ammo_group), |  | ||||||
|                              class: "text-primary-600 link", |  | ||||||
|                              data: [qa: "edit-#{ammo_group.id}"] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-edit"></i> |  | ||||||
|                   <% end %> |  | ||||||
|  |  | ||||||
|                   <%= link to: "#", |  | ||||||
|                        class: "text-primary-600 link", |  | ||||||
|                        phx_click: "delete", |  | ||||||
|                        phx_value_id: ammo_group.id, |  | ||||||
|                        data: [ |  | ||||||
|                          confirm: dgettext("prompts", "Are you sure you want to delete this ammo?"), |  | ||||||
|                          qa: "delete-#{ammo_group.id}" |  | ||||||
|                        ] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-trash"></i> |  | ||||||
|                   <% end %> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           <% end %> |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|   <% end %> |   <% end %> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| @@ -143,6 +66,7 @@ | |||||||
|         ammo_group={@ammo_group} |         ammo_group={@ammo_group} | ||||||
|         return_to={Routes.ammo_group_index_path(Endpoint, :index)} |         return_to={Routes.ammo_group_index_path(Endpoint, :index)} | ||||||
|         current_user={@current_user} |         current_user={@current_user} | ||||||
|  |         containers={@containers} | ||||||
|       /> |       /> | ||||||
|     </.modal> |     </.modal> | ||||||
|   <% @live_action == :add_shot_group -> %> |   <% @live_action == :add_shot_group -> %> | ||||||
| @@ -170,4 +94,5 @@ | |||||||
|       /> |       /> | ||||||
|     </.modal> |     </.modal> | ||||||
|   <% true -> %> |   <% true -> %> | ||||||
|   <% end %> |     <%= nil %> | ||||||
|  | <% end %> | ||||||
|   | |||||||
| @@ -5,8 +5,9 @@ defmodule CanneryWeb.AmmoGroupLive.Show do | |||||||
|  |  | ||||||
|   use CanneryWeb, :live_view |   use CanneryWeb, :live_view | ||||||
|   import CanneryWeb.Components.ContainerCard |   import CanneryWeb.Components.ContainerCard | ||||||
|   alias Cannery.{Ammo, Repo} |   alias Cannery.{ActivityLog, ActivityLog.ShotGroup, Ammo, Ammo.AmmoGroup, Repo} | ||||||
|   alias CanneryWeb.Endpoint |   alias CanneryWeb.Endpoint | ||||||
|  |   alias Phoenix.LiveView.Socket | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def mount(_params, session, socket) do |   def mount(_params, session, socket) do | ||||||
| @@ -15,18 +16,35 @@ defmodule CanneryWeb.AmmoGroupLive.Show do | |||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_params( |   def handle_params( | ||||||
|         %{"id" => id}, |         %{"id" => id, "shot_group_id" => shot_group_id}, | ||||||
|         _url, |         _url, | ||||||
|         %{assigns: %{live_action: live_action, current_user: current_user}} = socket |         %{assigns: %{live_action: live_action, current_user: current_user}} = socket | ||||||
|       ) do |       ) do | ||||||
|     ammo_group = Ammo.get_ammo_group!(id, current_user) |> Repo.preload([:container, :ammo_type]) |     shot_group = ActivityLog.get_shot_group!(shot_group_id, current_user) | ||||||
|     {:noreply, socket |> assign(page_title: page_title(live_action), ammo_group: ammo_group)} |  | ||||||
|  |     socket = | ||||||
|  |       socket | ||||||
|  |       |> assign(page_title: page_title(live_action), shot_group: shot_group) | ||||||
|  |       |> display_ammo_group(id) | ||||||
|  |  | ||||||
|  |     {:noreply, socket} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  |   @impl true | ||||||
|  |   def handle_params(%{"id" => id}, _url, %{assigns: %{live_action: live_action}} = socket) do | ||||||
|  |     {:noreply, socket |> assign(page_title: page_title(live_action)) |> display_ammo_group(id)} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp page_title(:add_shot_group), do: gettext("Record Shots") | ||||||
|  |   defp page_title(:edit_shot_group), do: gettext("Edit Shot Records") | ||||||
|  |   defp page_title(:move), do: gettext("Move Ammo group") | ||||||
|  |   defp page_title(:show), do: gettext("Show Ammo group") | ||||||
|  |   defp page_title(:edit), do: gettext("Edit Ammo group") | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_event( |   def handle_event( | ||||||
|         "delete", |         "delete", | ||||||
|         _, |         _params, | ||||||
|         %{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket |         %{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket | ||||||
|       ) do |       ) do | ||||||
|     ammo_group |> Ammo.delete_ammo_group!(current_user) |     ammo_group |> Ammo.delete_ammo_group!(current_user) | ||||||
| @@ -40,17 +58,90 @@ defmodule CanneryWeb.AmmoGroupLive.Show do | |||||||
|   @impl true |   @impl true | ||||||
|   def handle_event( |   def handle_event( | ||||||
|         "toggle_staged", |         "toggle_staged", | ||||||
|         _, |         _params, | ||||||
|         %{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket |         %{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket | ||||||
|       ) do |       ) do | ||||||
|     {:ok, ammo_group} = |     {:ok, ammo_group} = | ||||||
|       ammo_group |> Ammo.update_ammo_group(%{"staged" => !ammo_group.staged}, current_user) |       ammo_group |> Ammo.update_ammo_group(%{"staged" => !ammo_group.staged}, current_user) | ||||||
|  |  | ||||||
|     {:noreply, socket |> assign(ammo_group: ammo_group)} |     {:noreply, socket |> display_ammo_group(ammo_group)} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp page_title(:add_shot_group), do: gettext("Add Shot group") |   @impl true | ||||||
|   defp page_title(:move), do: gettext("Move Ammo group") |   def handle_event( | ||||||
|   defp page_title(:show), do: gettext("Show Ammo group") |         "delete_shot_group", | ||||||
|   defp page_title(:edit), do: gettext("Edit Ammo group") |         %{"id" => id}, | ||||||
|  |         %{assigns: %{ammo_group: ammo_group, current_user: current_user}} = socket | ||||||
|  |       ) do | ||||||
|  |     {:ok, _} = | ||||||
|  |       ActivityLog.get_shot_group!(id, current_user) | ||||||
|  |       |> ActivityLog.delete_shot_group(current_user) | ||||||
|  |  | ||||||
|  |     prompt = dgettext("prompts", "Shot records deleted succesfully") | ||||||
|  |     {:noreply, socket |> put_flash(:info, prompt) |> display_ammo_group(ammo_group)} | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   @spec display_ammo_group(Socket.t(), AmmoGroup.t() | AmmoGroup.id()) :: Socket.t() | ||||||
|  |   defp display_ammo_group(socket, %AmmoGroup{} = ammo_group) do | ||||||
|  |     ammo_group = ammo_group |> Repo.preload([:container, :ammo_type, :shot_groups], force: true) | ||||||
|  |  | ||||||
|  |     columns = [ | ||||||
|  |       %{label: gettext("Rounds shot"), key: "count"}, | ||||||
|  |       %{label: gettext("Notes"), key: "notes"}, | ||||||
|  |       %{label: gettext("Date"), key: "date"}, | ||||||
|  |       %{label: nil, key: "actions", sortable: false} | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     rows = | ||||||
|  |       ammo_group.shot_groups | ||||||
|  |       |> Enum.map(fn shot_group -> | ||||||
|  |         ammo_group |> get_table_row_for_shot_group(shot_group, columns) | ||||||
|  |       end) | ||||||
|  |  | ||||||
|  |     socket |> assign(ammo_group: ammo_group, columns: columns, rows: rows) | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp display_ammo_group(%{assigns: %{current_user: current_user}} = socket, id), | ||||||
|  |     do: display_ammo_group(socket, Ammo.get_ammo_group!(id, current_user)) | ||||||
|  |  | ||||||
|  |   @spec get_table_row_for_shot_group(AmmoGroup.t(), ShotGroup.t(), [map()]) :: [map()] | ||||||
|  |   defp get_table_row_for_shot_group(ammo_group, %{date: date} = shot_group, columns) do | ||||||
|  |     assigns = %{ammo_group: ammo_group, shot_group: shot_group} | ||||||
|  |  | ||||||
|  |     columns | ||||||
|  |     |> Enum.into(%{}, fn %{key: key} -> | ||||||
|  |       value = | ||||||
|  |         case key do | ||||||
|  |           "date" -> | ||||||
|  |             {date, date |> display_date()} | ||||||
|  |  | ||||||
|  |           "actions" -> | ||||||
|  |             ~H""" | ||||||
|  |             <div class="px-4 py-2 space-x-4 flex justify-center items-center"> | ||||||
|  |               <%= live_patch to: Routes.ammo_group_show_path(Endpoint, :edit_shot_group, @ammo_group, shot_group), | ||||||
|  |                           class: "text-primary-600 link", | ||||||
|  |                           data: [qa: "edit-#{shot_group.id}"] do %> | ||||||
|  |                 <i class="fa-fw fa-lg fas fa-edit"></i> | ||||||
|  |               <% end %> | ||||||
|  |  | ||||||
|  |               <%= link to: "#", | ||||||
|  |                     class: "text-primary-600 link", | ||||||
|  |                     phx_click: "delete_shot_group", | ||||||
|  |                     phx_value_id: shot_group.id, | ||||||
|  |                     data: [ | ||||||
|  |                       confirm: dgettext("prompts", "Are you sure you want to delete this shot record?"), | ||||||
|  |                       qa: "delete-#{shot_group.id}" | ||||||
|  |                     ] do %> | ||||||
|  |                 <i class="fa-fw fa-lg fas fa-trash"></i> | ||||||
|  |               <% end %> | ||||||
|  |             </div> | ||||||
|  |             """ | ||||||
|  |  | ||||||
|  |           key -> | ||||||
|  |             shot_group |> Map.get(key |> String.to_existing_atom()) | ||||||
|  |         end | ||||||
|  |  | ||||||
|  |       {key, value} | ||||||
|  |     end) | ||||||
|  |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -111,38 +111,12 @@ | |||||||
|       <%= gettext("Rounds used") %> |       <%= gettext("Rounds used") %> | ||||||
|     </h1> |     </h1> | ||||||
|  |  | ||||||
|     <div class="w-full overflow-x-auto border border-gray-600 rounded-lg shadow-lg bg-black"> |     <.live_component | ||||||
|       <table class="min-w-full table-auto text-center bg-white"> |       module={CanneryWeb.Components.TableComponent} | ||||||
|         <thead class="border-b border-primary-600"> |       id="ammo_group_shot_groups_table" | ||||||
|           <tr> |       columns={@columns} | ||||||
|             <th class="p-2"> |       rows={@rows} | ||||||
|               <%= gettext("Rounds shot") %> |     /> | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Notes") %> |  | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Date") %> |  | ||||||
|             </th> |  | ||||||
|           </tr> |  | ||||||
|         </thead> |  | ||||||
|         <tbody id="shot_groups"> |  | ||||||
|           <%= for shot_group <- @ammo_group.shot_groups do %> |  | ||||||
|             <tr id={"shot_group-#{shot_group.id}"}> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= shot_group.count %> |  | ||||||
|               </td> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= shot_group.notes %> |  | ||||||
|               </td> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= shot_group.date |> display_date() %> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           <% end %> |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|   <% end %> |   <% end %> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| @@ -159,6 +133,18 @@ | |||||||
|         current_user={@current_user} |         current_user={@current_user} | ||||||
|       /> |       /> | ||||||
|     </.modal> |     </.modal> | ||||||
|  |   <% :edit_shot_group -> %> | ||||||
|  |     <.modal return_to={Routes.ammo_group_show_path(Endpoint, :show, @ammo_group)}> | ||||||
|  |       <.live_component | ||||||
|  |         module={CanneryWeb.RangeLive.FormComponent} | ||||||
|  |         id={@shot_group.id} | ||||||
|  |         title={@page_title} | ||||||
|  |         action={@live_action} | ||||||
|  |         shot_group={@shot_group} | ||||||
|  |         return_to={Routes.ammo_group_show_path(Endpoint, :show, @ammo_group)} | ||||||
|  |         current_user={@current_user} | ||||||
|  |       /> | ||||||
|  |     </.modal> | ||||||
|   <% :add_shot_group -> %> |   <% :add_shot_group -> %> | ||||||
|     <.modal return_to={Routes.ammo_group_show_path(Endpoint, :show, @ammo_group)}> |     <.modal return_to={Routes.ammo_group_show_path(Endpoint, :show, @ammo_group)}> | ||||||
|       <.live_component |       <.live_component | ||||||
|   | |||||||
| @@ -17,11 +17,11 @@ | |||||||
|       </div> |       </div> | ||||||
|     <% end %> |     <% end %> | ||||||
|  |  | ||||||
|     <%= label(f, :name, gettext("Name"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :name, gettext("Name"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= text_input(f, :name, class: "text-center col-span-2 input input-primary") %> |     <%= text_input(f, :name, class: "text-center col-span-2 input input-primary") %> | ||||||
|     <%= error_tag(f, :name, "col-span-3 text-center") %> |     <%= error_tag(f, :name, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :desc, gettext("Description"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :desc, gettext("Description"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= textarea(f, :desc, |     <%= textarea(f, :desc, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       phx_hook: "MaintainAttrs" |       phx_hook: "MaintainAttrs" | ||||||
| @@ -34,50 +34,42 @@ | |||||||
|     > |     > | ||||||
|       <%= gettext("Example bullet type abbreviations") %> |       <%= gettext("Example bullet type abbreviations") %> | ||||||
|     </a> |     </a> | ||||||
|     <%= label(f, :bullet_type, gettext("Bullet type"), |     <%= label(f, :bullet_type, gettext("Bullet type"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :bullet_type, |     <%= text_input(f, :bullet_type, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: gettext("FMJ") |       placeholder: gettext("FMJ") | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :bullet_type, "col-span-3 text-center") %> |     <%= error_tag(f, :bullet_type, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :bullet_core, gettext("Bullet core"), |     <%= label(f, :bullet_core, gettext("Bullet core"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :bullet_core, |     <%= text_input(f, :bullet_core, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: gettext("Steel") |       placeholder: gettext("Steel") | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :bullet_core, "col-span-3 text-center") %> |     <%= error_tag(f, :bullet_core, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :cartridge, gettext("Cartridge"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :cartridge, gettext("Cartridge"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= text_input(f, :cartridge, |     <%= text_input(f, :cartridge, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: "5.56x46mm NATO" |       placeholder: "5.56x46mm NATO" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :cartridge, "col-span-3 text-center") %> |     <%= error_tag(f, :cartridge, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :caliber, gettext("Caliber"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :caliber, gettext("Caliber"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= text_input(f, :caliber, |     <%= text_input(f, :caliber, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: ".223" |       placeholder: ".223" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :caliber, "col-span-3 text-center") %> |     <%= error_tag(f, :caliber, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :case_material, gettext("Case material"), |     <%= label(f, :case_material, gettext("Case material"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :case_material, |     <%= text_input(f, :case_material, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: gettext("Brass") |       placeholder: gettext("Brass") | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :case_material, "col-span-3 text-center") %> |     <%= error_tag(f, :case_material, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :jacket_type, gettext("Jacket type"), |     <%= label(f, :jacket_type, gettext("Jacket type"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :jacket_type, |     <%= text_input(f, :jacket_type, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: gettext("Bimetal") |       placeholder: gettext("Bimetal") | ||||||
| @@ -85,7 +77,7 @@ | |||||||
|     <%= error_tag(f, :case_material, "col-span-3 text-center") %> |     <%= error_tag(f, :case_material, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :muzzle_velocity, gettext("Muzzle velocity"), |     <%= label(f, :muzzle_velocity, gettext("Muzzle velocity"), | ||||||
|       class: "mr-4 title text-lg text-primary-600" |       class: "title text-lg text-primary-600" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= number_input(f, :muzzle_velocity, |     <%= number_input(f, :muzzle_velocity, | ||||||
|       step: "1", |       step: "1", | ||||||
| @@ -94,14 +86,12 @@ | |||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :muzzle_velocity, "col-span-3 text-center") %> |     <%= error_tag(f, :muzzle_velocity, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :powder_type, gettext("Powder type"), |     <%= label(f, :powder_type, gettext("Powder type"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :powder_type, class: "text-center col-span-2 input input-primary") %> |     <%= text_input(f, :powder_type, class: "text-center col-span-2 input input-primary") %> | ||||||
|     <%= error_tag(f, :powder_type, "col-span-3 text-center") %> |     <%= error_tag(f, :powder_type, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :powder_grains_per_charge, gettext("Powder grains per charge"), |     <%= label(f, :powder_grains_per_charge, gettext("Powder grains per charge"), | ||||||
|       class: "mr-4 title text-lg text-primary-600" |       class: "title text-lg text-primary-600" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= number_input(f, :powder_grains_per_charge, |     <%= number_input(f, :powder_grains_per_charge, | ||||||
|       step: "1", |       step: "1", | ||||||
| @@ -110,7 +100,7 @@ | |||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :powder_grains_per_charge, "col-span-3 text-center") %> |     <%= error_tag(f, :powder_grains_per_charge, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :grains, gettext("Grains"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :grains, gettext("Grains"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= number_input(f, :grains, |     <%= number_input(f, :grains, | ||||||
|       step: "1", |       step: "1", | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
| @@ -118,54 +108,48 @@ | |||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :grains, "col-span-3 text-center") %> |     <%= error_tag(f, :grains, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :pressure, gettext("Pressure"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :pressure, gettext("Pressure"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= text_input(f, :pressure, |     <%= text_input(f, :pressure, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: "+P" |       placeholder: "+P" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :pressure, "col-span-3 text-center") %> |     <%= error_tag(f, :pressure, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :primer_type, gettext("Primer type"), |     <%= label(f, :primer_type, gettext("Primer type"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :primer_type, |     <%= text_input(f, :primer_type, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: "Boxer" |       placeholder: "Boxer" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :primer_type, "col-span-3 text-center") %> |     <%= error_tag(f, :primer_type, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :firing_type, gettext("Firing type"), |     <%= label(f, :firing_type, gettext("Firing type"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :firing_type, |     <%= text_input(f, :firing_type, | ||||||
|       class: "text-center col-span-2 input input-primary", |       class: "text-center col-span-2 input input-primary", | ||||||
|       placeholder: "Centerfire" |       placeholder: "Centerfire" | ||||||
|     ) %> |     ) %> | ||||||
|     <%= error_tag(f, :firing_type, "col-span-3 text-center") %> |     <%= error_tag(f, :firing_type, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :tracer, gettext("Tracer"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :tracer, gettext("Tracer"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= checkbox(f, :tracer, class: "text-center col-span-2 checkbox") %> |     <%= checkbox(f, :tracer, class: "text-center col-span-2 checkbox") %> | ||||||
|     <%= error_tag(f, :tracer, "col-span-3 text-center") %> |     <%= error_tag(f, :tracer, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :incendiary, gettext("Incendiary"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :incendiary, gettext("Incendiary"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= checkbox(f, :incendiary, class: "text-center col-span-2 checkbox") %> |     <%= checkbox(f, :incendiary, class: "text-center col-span-2 checkbox") %> | ||||||
|     <%= error_tag(f, :incendiary, "col-span-3 text-center") %> |     <%= error_tag(f, :incendiary, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :blank, gettext("Blank"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :blank, gettext("Blank"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= checkbox(f, :blank, class: "text-center col-span-2 checkbox") %> |     <%= checkbox(f, :blank, class: "text-center col-span-2 checkbox") %> | ||||||
|     <%= error_tag(f, :blank, "col-span-3 text-center") %> |     <%= error_tag(f, :blank, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :corrosive, gettext("Corrosive"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :corrosive, gettext("Corrosive"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= checkbox(f, :corrosive, class: "text-center col-span-2 checkbox") %> |     <%= checkbox(f, :corrosive, class: "text-center col-span-2 checkbox") %> | ||||||
|     <%= error_tag(f, :corrosive, "col-span-3 text-center") %> |     <%= error_tag(f, :corrosive, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :manufacturer, gettext("Manufacturer"), |     <%= label(f, :manufacturer, gettext("Manufacturer"), class: "title text-lg text-primary-600") %> | ||||||
|       class: "mr-4 title text-lg text-primary-600" |  | ||||||
|     ) %> |  | ||||||
|     <%= text_input(f, :manufacturer, class: "text-center col-span-2 input input-primary") %> |     <%= text_input(f, :manufacturer, class: "text-center col-span-2 input input-primary") %> | ||||||
|     <%= error_tag(f, :manufacturer, "col-span-3 text-center") %> |     <%= error_tag(f, :manufacturer, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|     <%= label(f, :upc, gettext("UPC"), class: "mr-4 title text-lg text-primary-600") %> |     <%= label(f, :upc, gettext("UPC"), class: "title text-lg text-primary-600") %> | ||||||
|     <%= text_input(f, :upc, class: "text-center col-span-2 input input-primary") %> |     <%= text_input(f, :upc, class: "text-center col-span-2 input input-primary") %> | ||||||
|     <%= error_tag(f, :upc, "col-span-3 text-center") %> |     <%= error_tag(f, :upc, "col-span-3 text-center") %> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ defmodule CanneryWeb.AmmoTypeLive.Index do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp apply_action(socket, :index, _params) do |   defp apply_action(socket, :index, _params) do | ||||||
|     socket |> assign(:page_title, gettext("Listing Ammo types")) |> assign(:ammo_type, nil) |     socket |> assign(:page_title, gettext("Ammo types")) |> assign(:ammo_type, nil) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
| @@ -46,37 +46,103 @@ defmodule CanneryWeb.AmmoTypeLive.Index do | |||||||
|   defp list_ammo_types(%{assigns: %{current_user: current_user}} = socket) do |   defp list_ammo_types(%{assigns: %{current_user: current_user}} = socket) do | ||||||
|     ammo_types = Ammo.list_ammo_types(current_user) |     ammo_types = Ammo.list_ammo_types(current_user) | ||||||
|  |  | ||||||
|     columns_to_display = |     columns = | ||||||
|       [ |       [ | ||||||
|         {gettext("Name"), :name, :string}, |         %{label: gettext("Name"), key: "name", type: :string}, | ||||||
|         {gettext("Bullet type"), :bullet_type, :string}, |         %{label: gettext("Bullet type"), key: "bullet_type", type: :string}, | ||||||
|         {gettext("Bullet core"), :bullet_core, :string}, |         %{label: gettext("Bullet core"), key: "bullet_core", type: :string}, | ||||||
|         {gettext("Cartridge"), :cartridge, :string}, |         %{label: gettext("Cartridge"), key: "cartridge", type: :string}, | ||||||
|         {gettext("Caliber"), :caliber, :string}, |         %{label: gettext("Caliber"), key: "caliber", type: :string}, | ||||||
|         {gettext("Case material"), :case_material, :string}, |         %{label: gettext("Case material"), key: "case_material", type: :string}, | ||||||
|         {gettext("Jacket type"), :jacket_type, :string}, |         %{label: gettext("Jacket type"), key: "jacket_type", type: :string}, | ||||||
|         {gettext("Muzzle velocity"), :muzzle_velocity, :string}, |         %{label: gettext("Muzzle velocity"), key: "muzzle_velocity", type: :string}, | ||||||
|         {gettext("Powder type"), :powder_type, :string}, |         %{label: gettext("Powder type"), key: "powder_type", type: :string}, | ||||||
|         {gettext("Powder grains per charge"), :powder_grains_per_charge, :string}, |         %{ | ||||||
|         {gettext("Grains"), :grains, :string}, |           label: gettext("Powder grains per charge"), | ||||||
|         {gettext("Pressure"), :pressure, :string}, |           key: "powder_grains_per_charge", | ||||||
|         {gettext("Primer type"), :primer_type, :string}, |           type: :string | ||||||
|         {gettext("Firing type"), :firing_type, :string}, |         }, | ||||||
|         {gettext("Tracer"), :tracer, :boolean}, |         %{label: gettext("Grains"), key: "grains", type: :string}, | ||||||
|         {gettext("Incendiary"), :incendiary, :boolean}, |         %{label: gettext("Pressure"), key: "pressure", type: :string}, | ||||||
|         {gettext("Blank"), :blank, :boolean}, |         %{label: gettext("Primer type"), key: "primer_type", type: :string}, | ||||||
|         {gettext("Corrosive"), :corrosive, :boolean}, |         %{label: gettext("Firing type"), key: "firing_type", type: :string}, | ||||||
|         {gettext("Manufacturer"), :manufacturer, :string}, |         %{label: gettext("Tracer"), key: "tracer", type: :boolean}, | ||||||
|         {gettext("UPC"), :upc, :string} |         %{label: gettext("Incendiary"), key: "incendiary", type: :boolean}, | ||||||
|  |         %{label: gettext("Blank"), key: "blank", type: :boolean}, | ||||||
|  |         %{label: gettext("Corrosive"), key: "corrosive", type: :boolean}, | ||||||
|  |         %{label: gettext("Manufacturer"), key: "manufacturer", type: :string}, | ||||||
|  |         %{label: gettext("UPC"), key: "upc", type: :string} | ||||||
|       ] |       ] | ||||||
|       |> Enum.filter(fn {_label, field, type} -> |       |> Enum.filter(fn %{key: key, type: type} -> | ||||||
|         # remove columns if all values match defaults |         # remove columns if all values match defaults | ||||||
|         default_value = if type == :boolean, do: false, else: nil |         default_value = if type == :boolean, do: false, else: nil | ||||||
|  |  | ||||||
|         ammo_types |         ammo_types | ||||||
|         |> Enum.any?(fn ammo_type -> not (ammo_type |> Map.get(field) == default_value) end) |         |> Enum.any?(fn ammo_type -> | ||||||
|  |           not (ammo_type |> Map.get(key |> String.to_existing_atom()) == default_value) | ||||||
|  |         end) | ||||||
|       end) |       end) | ||||||
|  |       |> Kernel.++([ | ||||||
|  |         %{label: gettext("Total # of rounds"), key: "round_count", type: :round_count}, | ||||||
|  |         %{label: nil, key: "actions", type: :actions, sortable: false} | ||||||
|  |       ]) | ||||||
|  |  | ||||||
|     socket |> assign(ammo_types: ammo_types, columns_to_display: columns_to_display) |     rows = | ||||||
|  |       ammo_types | ||||||
|  |       |> Enum.map(fn ammo_type -> ammo_type |> get_ammo_type_values(columns, current_user) end) | ||||||
|  |  | ||||||
|  |     socket |> assign(columns: columns, rows: rows) | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   defp get_ammo_type_values(ammo_type, columns, current_user) do | ||||||
|  |     assigns = %{ammo_type: ammo_type} | ||||||
|  |  | ||||||
|  |     columns | ||||||
|  |     |> Enum.into(%{}, fn %{key: key, type: type} -> | ||||||
|  |       value = | ||||||
|  |         case type do | ||||||
|  |           :boolean -> | ||||||
|  |             ammo_type |> Map.get(key |> String.to_existing_atom()) |> humanize() | ||||||
|  |  | ||||||
|  |           :round_count -> | ||||||
|  |             ammo_type |> Ammo.get_round_count_for_ammo_type(current_user) | ||||||
|  |  | ||||||
|  |           :actions -> | ||||||
|  |             ~H""" | ||||||
|  |             <div class="px-4 py-2 space-x-4 flex justify-center items-center"> | ||||||
|  |               <%= live_redirect to: Routes.ammo_type_show_path(Endpoint, :show, ammo_type), | ||||||
|  |                             class: "text-primary-600 link", | ||||||
|  |                             data: [qa: "view-#{ammo_type.id}"] do %> | ||||||
|  |                 <i class="fa-fw fa-lg fas fa-eye"></i> | ||||||
|  |               <% end %> | ||||||
|  |  | ||||||
|  |               <%= live_patch to: Routes.ammo_type_index_path(Endpoint, :edit, ammo_type), | ||||||
|  |                           class: "text-primary-600 link", | ||||||
|  |                           data: [qa: "edit-#{ammo_type.id}"] do %> | ||||||
|  |                 <i class="fa-fw fa-lg fas fa-edit"></i> | ||||||
|  |               <% end %> | ||||||
|  |  | ||||||
|  |               <%= link to: "#", | ||||||
|  |                     class: "text-primary-600 link", | ||||||
|  |                     phx_click: "delete", | ||||||
|  |                     phx_value_id: ammo_type.id, | ||||||
|  |                     data: [ | ||||||
|  |                       confirm: dgettext("prompts", "Are you sure you want to delete this ammo?"), | ||||||
|  |                       qa: "delete-#{ammo_type.id}" | ||||||
|  |                     ] do %> | ||||||
|  |                 <i class="fa-lg fas fa-trash"></i> | ||||||
|  |               <% end %> | ||||||
|  |             </div> | ||||||
|  |             """ | ||||||
|  |  | ||||||
|  |           nil -> | ||||||
|  |             nil | ||||||
|  |  | ||||||
|  |           _other -> | ||||||
|  |             ammo_type |> Map.get(key |> String.to_existing_atom()) | ||||||
|  |         end | ||||||
|  |  | ||||||
|  |       {key, value} | ||||||
|  |     end) | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ | |||||||
|     <%= gettext("Ammo Types") %> |     <%= gettext("Ammo Types") %> | ||||||
|   </h1> |   </h1> | ||||||
|  |  | ||||||
|   <%= if @ammo_types |> Enum.empty?() do %> |   <%= if @rows |> Enum.empty?() do %> | ||||||
|     <h2 class="title text-xl text-primary-600"> |     <h2 class="title text-xl text-primary-600"> | ||||||
|       <%= gettext("No Ammo Types") %> |       <%= gettext("No Ammo Types") %> | ||||||
|       <%= display_emoji("😔") %> |       <%= display_emoji("😔") %> | ||||||
| @@ -19,71 +19,13 @@ | |||||||
|       class: "btn btn-primary" |       class: "btn btn-primary" | ||||||
|     ) %> |     ) %> | ||||||
|  |  | ||||||
|     <div class="w-full overflow-x-auto border border-gray-600 rounded-lg shadow-lg bg-black"> |     <.live_component | ||||||
|       <table class="min-w-full table-auto text-center bg-white"> |       module={CanneryWeb.Components.TableComponent} | ||||||
|         <thead class="border-b border-primary-600"> |       id="ammo_types_index_table" | ||||||
|           <tr> |       action={@live_action} | ||||||
|             <%= for {field_name, _field, _type} <- @columns_to_display do %> |       columns={@columns} | ||||||
|               <th class="p-2"> |       rows={@rows} | ||||||
|                 <%= field_name %> |     /> | ||||||
|               </th> |  | ||||||
|             <% end %> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Total # of rounds") %> |  | ||||||
|             </th> |  | ||||||
|  |  | ||||||
|             <th class="p-2"></th> |  | ||||||
|           </tr> |  | ||||||
|         </thead> |  | ||||||
|         <tbody> |  | ||||||
|           <%= for ammo_type <- @ammo_types do %> |  | ||||||
|             <tr id={"ammo_type-#{ammo_type.id}"}> |  | ||||||
|               <%= for {_label, field, type} <- @columns_to_display do %> |  | ||||||
|                 <td class="p-2"> |  | ||||||
|                   <%= case type do %> |  | ||||||
|                     <% :boolean -> %> |  | ||||||
|                       <%= ammo_type |> Map.get(field) |> humanize() %> |  | ||||||
|                     <% _other -> %> |  | ||||||
|                       <%= ammo_type |> Map.get(field) %> |  | ||||||
|                   <% end %> |  | ||||||
|                 </td> |  | ||||||
|               <% end %> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= ammo_type |> Ammo.get_round_count_for_ammo_type(@current_user) %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <div class="px-4 py-2 space-x-4 flex justify-center items-center"> |  | ||||||
|                   <%= live_redirect to: Routes.ammo_type_show_path(Endpoint, :show, ammo_type), |  | ||||||
|                                 class: "text-primary-600 link", |  | ||||||
|                                 data: [qa: "view-#{ammo_type.id}"] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-eye"></i> |  | ||||||
|                   <% end %> |  | ||||||
|  |  | ||||||
|                   <%= live_patch to: Routes.ammo_type_index_path(Endpoint, :edit, ammo_type), |  | ||||||
|                              class: "text-primary-600 link", |  | ||||||
|                              data: [qa: "edit-#{ammo_type.id}"] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-edit"></i> |  | ||||||
|                   <% end %> |  | ||||||
|  |  | ||||||
|                   <%= link to: "#", |  | ||||||
|                        class: "text-primary-600 link", |  | ||||||
|                        phx_click: "delete", |  | ||||||
|                        phx_value_id: ammo_type.id, |  | ||||||
|                        data: [ |  | ||||||
|                          confirm: dgettext("prompts", "Are you sure you want to delete this ammo?"), |  | ||||||
|                          qa: "delete-#{ammo_type.id}" |  | ||||||
|                        ] do %> |  | ||||||
|                     <i class="fa-lg fas fa-trash"></i> |  | ||||||
|                   <% end %> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           <% end %> |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|   <% end %> |   <% end %> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ defmodule CanneryWeb.AmmoTypeLive.Show do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_params(%{"id" => id}, _, %{assigns: %{current_user: current_user}} = socket) do |   def handle_params(%{"id" => id}, _params, %{assigns: %{current_user: current_user}} = socket) do | ||||||
|     ammo_type = Ammo.get_ammo_type!(id, current_user) |     ammo_type = Ammo.get_ammo_type!(id, current_user) | ||||||
|  |  | ||||||
|     socket = |     socket = | ||||||
| @@ -32,7 +32,7 @@ defmodule CanneryWeb.AmmoTypeLive.Show do | |||||||
|   @impl true |   @impl true | ||||||
|   def handle_event( |   def handle_event( | ||||||
|         "delete", |         "delete", | ||||||
|         _, |         _params, | ||||||
|         %{assigns: %{ammo_type: ammo_type, current_user: current_user}} = socket |         %{assigns: %{ammo_type: ammo_type, current_user: current_user}} = socket | ||||||
|       ) do |       ) do | ||||||
|     %{name: ammo_type_name} = ammo_type |> Ammo.delete_ammo_type!(current_user) |     %{name: ammo_type_name} = ammo_type |> Ammo.delete_ammo_type!(current_user) | ||||||
|   | |||||||
| @@ -100,9 +100,8 @@ | |||||||
|         ) %> |         ) %> | ||||||
|       </span> |       </span> | ||||||
|     <% else %> |     <% else %> | ||||||
|       <h3 class="title text-lg col-span-2"> |       <h3 class="mx-8 my-4 title text-lg text-primary-600 col-span-2"> | ||||||
|         <%= gettext("No cost information") %> |         <%= gettext("No cost information") %> | ||||||
|         <%= display_emoji("😔") %> |  | ||||||
|       </h3> |       </h3> | ||||||
|     <% end %> |     <% end %> | ||||||
|   </div> |   </div> | ||||||
| @@ -111,8 +110,10 @@ | |||||||
|  |  | ||||||
|   <div> |   <div> | ||||||
|     <%= if @ammo_groups |> Enum.empty?() do %> |     <%= if @ammo_groups |> Enum.empty?() do %> | ||||||
|       <%= gettext("No ammo for this type") %> |       <h2 class="mx-8 my-4 title text-lg text-primary-600"> | ||||||
|       <%= display_emoji("😔") %> |         <%= gettext("No ammo for this type") %> | ||||||
|  |         <%= display_emoji("😔") %> | ||||||
|  |       </h2> | ||||||
|     <% else %> |     <% else %> | ||||||
|       <div class="flex flex-wrap justify-center items-center"> |       <div class="flex flex-wrap justify-center items-center"> | ||||||
|         <%= for ammo_group <- @ammo_groups do %> |         <%= for ammo_group <- @ammo_groups do %> | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ defmodule CanneryWeb.ContainerLive.Index do | |||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_params(params, _url, %{assigns: %{live_action: live_action}} = socket) do |   def handle_params(params, _url, %{assigns: %{live_action: live_action}} = socket) do | ||||||
|     {:noreply, apply_action(socket, live_action, params)} |     {:noreply, apply_action(socket, live_action, params) |> display_containers()} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp apply_action(%{assigns: %{current_user: current_user}} = socket, :edit, %{"id" => id}) do |   defp apply_action(%{assigns: %{current_user: current_user}} = socket, :edit, %{"id" => id}) do | ||||||
| @@ -35,7 +35,7 @@ defmodule CanneryWeb.ContainerLive.Index do | |||||||
|  |  | ||||||
|   defp apply_action(socket, :index, _params) do |   defp apply_action(socket, :index, _params) do | ||||||
|     socket |     socket | ||||||
|     |> assign(:page_title, gettext("Listing Containers")) |     |> assign(:page_title, gettext("Containers")) | ||||||
|     |> assign(:container, nil) |     |> assign(:container, nil) | ||||||
|     |> display_containers() |     |> display_containers() | ||||||
|   end |   end | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ | |||||||
|     ) %> |     ) %> | ||||||
|   <% end %> |   <% end %> | ||||||
|  |  | ||||||
|   <div class="flex flex-row flex-wrap justify-center items-center"> |   <div class="max-w-full flex flex-row flex-wrap justify-center items-center"> | ||||||
|     <%= for container <- @containers do %> |     <%= for container <- @containers do %> | ||||||
|       <.container_card container={container}> |       <.container_card container={container}> | ||||||
|         <:tag_actions> |         <:tag_actions> | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ defmodule CanneryWeb.ContainerLive.Show do | |||||||
|   @impl true |   @impl true | ||||||
|   def handle_params( |   def handle_params( | ||||||
|         %{"id" => id}, |         %{"id" => id}, | ||||||
|         _, |         _session, | ||||||
|         %{assigns: %{current_user: current_user}} = socket |         %{assigns: %{current_user: current_user}} = socket | ||||||
|       ) do |       ) do | ||||||
|     {:noreply, socket |> render_container(id, current_user)} |     {:noreply, socket |> render_container(id, current_user)} | ||||||
| @@ -53,7 +53,7 @@ defmodule CanneryWeb.ContainerLive.Show do | |||||||
|   @impl true |   @impl true | ||||||
|   def handle_event( |   def handle_event( | ||||||
|         "delete_container", |         "delete_container", | ||||||
|         _, |         _params, | ||||||
|         %{assigns: %{container: container, current_user: current_user}} = socket |         %{assigns: %{container: container, current_user: current_user}} = socket | ||||||
|       ) do |       ) do | ||||||
|     socket = |     socket = | ||||||
|   | |||||||
| @@ -72,9 +72,11 @@ | |||||||
|  |  | ||||||
|   <hr class="mb-4 hr" /> |   <hr class="mb-4 hr" /> | ||||||
|  |  | ||||||
|   <p> |   <div> | ||||||
|     <%= if @container.ammo_groups |> Enum.empty?() do %> |     <%= if @container.ammo_groups |> Enum.empty?() do %> | ||||||
|       <%= gettext("No ammo groups in this container") %> |       <h2 class="mx-8 my-4 title text-lg text-primary-600"> | ||||||
|  |         <%= gettext("No ammo groups in this container") %> | ||||||
|  |       </h2> | ||||||
|     <% else %> |     <% else %> | ||||||
|       <div class="flex flex-wrap justify-center items-center"> |       <div class="flex flex-wrap justify-center items-center"> | ||||||
|         <%= for ammo_group <- @container.ammo_groups do %> |         <%= for ammo_group <- @container.ammo_groups do %> | ||||||
| @@ -82,7 +84,7 @@ | |||||||
|         <% end %> |         <% end %> | ||||||
|       </div> |       </div> | ||||||
|     <% end %> |     <% end %> | ||||||
|   </p> |   </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| <%= if @live_action in [:edit] do %> | <%= if @live_action in [:edit] do %> | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ defmodule CanneryWeb.HomeLive do | |||||||
|       %{^query => vsn} -> |       %{^query => vsn} -> | ||||||
|         {:noreply, redirect(socket, external: "https://hexdocs.pm/#{query}/#{vsn}")} |         {:noreply, redirect(socket, external: "https://hexdocs.pm/#{query}/#{vsn}")} | ||||||
|  |  | ||||||
|       _ -> |       _no_query -> | ||||||
|         {:noreply, |         {:noreply, | ||||||
|          socket |          socket | ||||||
|          |> put_flash(:error, "No dependencies found matching \"#{query}\"") |          |> put_flash(:error, "No dependencies found matching \"#{query}\"") | ||||||
| @@ -127,11 +127,15 @@ defmodule CanneryWeb.HomeLive do | |||||||
|           </p> |           </p> | ||||||
|         </li> |         </li> | ||||||
|  |  | ||||||
|         <li class="flex flex-row justify-center space-x-2"> |         <li class="flex flex-row justify-center items-center space-x-2"> | ||||||
|           <b>Version:</b> |           <b>Version:</b> | ||||||
|           <p> |           <%= link class: "flex flex-row justify-center items-center space-x-2 hover:underline", | ||||||
|             0.2.1 |                 to: "https://gitea.bubbletea.dev/shibao/cannery/src/branch/stable/CHANGELOG.md", | ||||||
|           </p> |                 target: "_blank", | ||||||
|  |                 rel: "noopener noreferrer" do %> | ||||||
|  |             <p>0.4.1</p> | ||||||
|  |             <i class="fas fa-md fa-info-circle"></i> | ||||||
|  |           <% end %> | ||||||
|         </li> |         </li> | ||||||
|       </ul> |       </ul> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ defmodule CanneryWeb.InviteLive.Index do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp apply_action(socket, :index, _params) do |   defp apply_action(socket, :index, _params) do | ||||||
|     socket |> assign(page_title: gettext("Listing Invites"), invite: nil) |     socket |> assign(page_title: gettext("Invites"), invite: nil) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
| @@ -119,7 +119,7 @@ defmodule CanneryWeb.InviteLive.Index do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_event("copy_to_clipboard", _, socket) do |   def handle_event("copy_to_clipboard", _params, socket) do | ||||||
|     prompt = dgettext("prompts", "Copied to clipboard") |     prompt = dgettext("prompts", "Copied to clipboard") | ||||||
|     {:noreply, socket |> put_flash(:info, prompt)} |     {:noreply, socket |> put_flash(:info, prompt)} | ||||||
|   end |   end | ||||||
|   | |||||||
| @@ -38,32 +38,45 @@ defmodule CanneryWeb.LiveHelpers do | |||||||
|   """ |   """ | ||||||
|   def modal(assigns) do |   def modal(assigns) do | ||||||
|     ~H""" |     ~H""" | ||||||
|  |     <%= live_patch to: @return_to, | ||||||
|  |       id: "modal-bg", | ||||||
|  |       class: | ||||||
|  |         "fade-in fixed z-10 left-0 top-0 | ||||||
|  |          w-full h-full overflow-hidden | ||||||
|  |          p-8 flex flex-col justify-center items-center cursor-auto", | ||||||
|  |       style: "background-color: rgba(0,0,0,0.4);", | ||||||
|  |       phx_remove: hide_modal() | ||||||
|  |     do %> | ||||||
|  |       <span class="hidden"></span> | ||||||
|  |     <% end %> | ||||||
|  |  | ||||||
|     <div |     <div | ||||||
|       id="modal" |       id="modal" | ||||||
|       class="fade-in fixed z-10 left-0 top-0 |       class="fixed z-10 left-0 top-0 pointer-events-none | ||||||
|       w-full h-full overflow-hidden |         w-full h-full overflow-hidden | ||||||
|       p-8 flex flex-col justify-center items-center" |         p-4 sm:p-8 flex flex-col justify-center items-center" | ||||||
|       style="opacity: 1 !important; background-color: rgba(0,0,0,0.4);" |  | ||||||
|       phx-remove={hide_modal()} |  | ||||||
|     > |     > | ||||||
|       <div |       <div | ||||||
|         id="modal-content" |         id="modal-content" | ||||||
|         class="fade-in-scale w-full max-w-3xl max-h-128 relative overflow-y-auto |         class="fade-in-scale w-full max-w-3xl relative | ||||||
|  |         pointer-events-auto overflow-hidden | ||||||
|  |         px-8 py-4 sm:py-8 flex flex-col justify-center items-center | ||||||
|         flex flex-col justify-start items-center |         flex flex-col justify-start items-center | ||||||
|         bg-white border-2 rounded-lg" |         bg-white border-2 rounded-lg" | ||||||
|         phx-click-away={hide_modal()} |  | ||||||
|         phx-window-keydown={hide_modal()} |  | ||||||
|         phx-key="escape" |  | ||||||
|       > |       > | ||||||
|         <%= live_patch to: @return_to, |         <%= live_patch to: @return_to, | ||||||
|                    id: "close", |                    id: "close", | ||||||
|                    class: |                    class: | ||||||
|                      "absolute top-8 right-10 text-gray-500 hover:text-gray-800 transition-all duration-500 ease-in-out", |                      "absolute top-8 right-10 | ||||||
|                    phx_click: hide_modal() do %> |                       text-gray-500 hover:text-gray-800 | ||||||
|  |                       transition-all duration-500 ease-in-out", | ||||||
|  |                    phx_remove: hide_modal() do %> | ||||||
|           <i class="fa-fw fa-lg fas fa-times"></i> |           <i class="fa-fw fa-lg fas fa-times"></i> | ||||||
|         <% end %> |         <% end %> | ||||||
|  |  | ||||||
|         <div class="w-full p-8 flex flex-col space-y-4 justify-start items-center"> |         <div | ||||||
|  |           class="overflow-x-hidden overflow-y-auto w-full p-8 flex flex-col space-y-4 justify-start items-center" | ||||||
|  |         > | ||||||
|           <%= render_slot(@inner_block) %> |           <%= render_slot(@inner_block) %> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
| @@ -74,6 +87,7 @@ defmodule CanneryWeb.LiveHelpers do | |||||||
|   def hide_modal(js \\ %JS{}) do |   def hide_modal(js \\ %JS{}) do | ||||||
|     js |     js | ||||||
|     |> JS.hide(to: "#modal", transition: "fade-out") |     |> JS.hide(to: "#modal", transition: "fade-out") | ||||||
|  |     |> JS.hide(to: "#modal-bg", transition: "fade-out") | ||||||
|     |> JS.hide(to: "#modal-content", transition: "fade-out-scale") |     |> JS.hide(to: "#modal-content", transition: "fade-out-scale") | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ defmodule CanneryWeb.RangeLive.Index do | |||||||
|          %{"id" => id} |          %{"id" => id} | ||||||
|        ) do |        ) do | ||||||
|     socket |     socket | ||||||
|     |> assign(:page_title, gettext("Record shots")) |     |> assign(:page_title, gettext("Record Shots")) | ||||||
|     |> assign(:ammo_group, Ammo.get_ammo_group!(id, current_user)) |     |> assign(:ammo_group, Ammo.get_ammo_group!(id, current_user)) | ||||||
|   end |   end | ||||||
|  |  | ||||||
| @@ -77,6 +77,69 @@ defmodule CanneryWeb.RangeLive.Index do | |||||||
|       ActivityLog.list_shot_groups(current_user) |> Repo.preload(ammo_group: :ammo_type) |       ActivityLog.list_shot_groups(current_user) |> Repo.preload(ammo_group: :ammo_type) | ||||||
|  |  | ||||||
|     ammo_groups = Ammo.list_staged_ammo_groups(current_user) |     ammo_groups = Ammo.list_staged_ammo_groups(current_user) | ||||||
|     socket |> assign(shot_groups: shot_groups, ammo_groups: ammo_groups) |  | ||||||
|  |     columns = [ | ||||||
|  |       %{label: gettext("Ammo"), key: "name"}, | ||||||
|  |       %{label: gettext("Rounds shot"), key: "count"}, | ||||||
|  |       %{label: gettext("Notes"), key: "notes"}, | ||||||
|  |       %{label: gettext("Date"), key: "date"}, | ||||||
|  |       %{label: nil, key: "actions", sortable: false} | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     rows = | ||||||
|  |       shot_groups | ||||||
|  |       |> Enum.map(fn shot_group -> shot_group |> get_row_data_for_shot_group(columns) end) | ||||||
|  |  | ||||||
|  |     socket | ||||||
|  |     |> assign(ammo_groups: ammo_groups, columns: columns, rows: rows, shot_groups: shot_groups) | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   @spec get_row_data_for_shot_group(ShotGroup.t(), [map()]) :: [map()] | ||||||
|  |   defp get_row_data_for_shot_group(%{date: date} = shot_group, columns) do | ||||||
|  |     shot_group = shot_group |> Repo.preload(ammo_group: :ammo_type) | ||||||
|  |     assigns = %{shot_group: shot_group} | ||||||
|  |  | ||||||
|  |     columns | ||||||
|  |     |> Enum.into(%{}, fn %{key: key} -> | ||||||
|  |       value = | ||||||
|  |         case key do | ||||||
|  |           "name" -> | ||||||
|  |             {shot_group.ammo_group.ammo_type.name, | ||||||
|  |              live_patch(shot_group.ammo_group.ammo_type.name, | ||||||
|  |                to: Routes.ammo_group_show_path(Endpoint, :show, shot_group.ammo_group), | ||||||
|  |                class: "link" | ||||||
|  |              )} | ||||||
|  |  | ||||||
|  |           "date" -> | ||||||
|  |             date |> display_date() | ||||||
|  |  | ||||||
|  |           "actions" -> | ||||||
|  |             ~H""" | ||||||
|  |             <div class="px-4 py-2 space-x-4 flex justify-center items-center"> | ||||||
|  |               <%= live_patch to: Routes.range_index_path(Endpoint, :edit, shot_group), | ||||||
|  |                           class: "text-primary-600 link", | ||||||
|  |                           data: [qa: "edit-#{shot_group.id}"] do %> | ||||||
|  |                 <i class="fa-fw fa-lg fas fa-edit"></i> | ||||||
|  |               <% end %> | ||||||
|  |  | ||||||
|  |               <%= link to: "#", | ||||||
|  |                     class: "text-primary-600 link", | ||||||
|  |                     phx_click: "delete", | ||||||
|  |                     phx_value_id: shot_group.id, | ||||||
|  |                     data: [ | ||||||
|  |                       confirm: dgettext("prompts", "Are you sure you want to delete this shot record?"), | ||||||
|  |                       qa: "delete-#{shot_group.id}" | ||||||
|  |                     ] do %> | ||||||
|  |                 <i class="fa-fw fa-lg fas fa-trash"></i> | ||||||
|  |               <% end %> | ||||||
|  |             </div> | ||||||
|  |             """ | ||||||
|  |  | ||||||
|  |           key -> | ||||||
|  |             shot_group |> Map.get(key |> String.to_existing_atom()) | ||||||
|  |         end | ||||||
|  |  | ||||||
|  |       {key, value} | ||||||
|  |     end) | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -53,70 +53,12 @@ | |||||||
|       <%= gettext("Shot log") %> |       <%= gettext("Shot log") %> | ||||||
|     </h1> |     </h1> | ||||||
|  |  | ||||||
|     <div class="w-full overflow-x-auto border border-gray-600 rounded-lg shadow-lg bg-black"> |     <.live_component | ||||||
|       <table class="min-w-full table-auto text-center bg-white"> |       module={CanneryWeb.Components.TableComponent} | ||||||
|         <thead class="border-b border-primary-600"> |       id="shot_groups_index_table" | ||||||
|           <tr> |       columns={@columns} | ||||||
|             <th class="p-2"> |       rows={@rows} | ||||||
|               <%= gettext("Ammo") %> |     /> | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Rounds shot") %> |  | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Notes") %> |  | ||||||
|             </th> |  | ||||||
|             <th class="p-2"> |  | ||||||
|               <%= gettext("Date") %> |  | ||||||
|             </th> |  | ||||||
|  |  | ||||||
|             <th class="p-2"></th> |  | ||||||
|           </tr> |  | ||||||
|         </thead> |  | ||||||
|         <tbody id="shot_groups"> |  | ||||||
|           <%= for shot_group <- @shot_groups do %> |  | ||||||
|             <tr id={"shot_group-#{shot_group.id}"}> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= live_patch(shot_group.ammo_group.ammo_type.name, |  | ||||||
|                   to: Routes.ammo_group_show_path(Endpoint, :show, shot_group.ammo_group), |  | ||||||
|                   class: "link" |  | ||||||
|                 ) %> |  | ||||||
|               </td> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= shot_group.count %> |  | ||||||
|               </td> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= shot_group.notes %> |  | ||||||
|               </td> |  | ||||||
|               <td class="p-2"> |  | ||||||
|                 <%= shot_group.date |> display_date() %> |  | ||||||
|               </td> |  | ||||||
|  |  | ||||||
|               <td class="p-2 w-full h-full space-x-2 flex justify-center items-center"> |  | ||||||
|                 <div class="px-4 py-2 space-x-4 flex justify-center items-center"> |  | ||||||
|                   <%= live_patch to: Routes.range_index_path(Endpoint, :edit, shot_group), |  | ||||||
|                              class: "text-primary-600 link", |  | ||||||
|                              data: [qa: "edit-#{shot_group.id}"] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-edit"></i> |  | ||||||
|                   <% end %> |  | ||||||
|  |  | ||||||
|                   <%= link to: "#", |  | ||||||
|                        class: "text-primary-600 link", |  | ||||||
|                        phx_click: "delete", |  | ||||||
|                        phx_value_id: shot_group.id, |  | ||||||
|                        data: [ |  | ||||||
|                          confirm: dgettext("prompts", "Are you sure you want to delete this shot record?"), |  | ||||||
|                          qa: "delete-#{shot_group.id}" |  | ||||||
|                        ] do %> |  | ||||||
|                     <i class="fa-fw fa-lg fas fa-trash"></i> |  | ||||||
|                   <% end %> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           <% end %> |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|   <% end %> |   <% end %> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ defmodule CanneryWeb.TagLive.Index do | |||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   def handle_params(params, _url, %{assigns: %{live_action: live_action}} = socket) do |   def handle_params(params, _url, %{assigns: %{live_action: live_action}} = socket) do | ||||||
|     {:noreply, apply_action(socket, live_action, params)} |     {:noreply, apply_action(socket, live_action, params) |> display_tags} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp apply_action(%{assigns: %{current_user: current_user}} = socket, :edit, %{"id" => id}) do |   defp apply_action(%{assigns: %{current_user: current_user}} = socket, :edit, %{"id" => id}) do | ||||||
| @@ -31,7 +31,7 @@ defmodule CanneryWeb.TagLive.Index do | |||||||
|   end |   end | ||||||
|  |  | ||||||
|   defp apply_action(socket, :index, _params) do |   defp apply_action(socket, :index, _params) do | ||||||
|     socket |> assign(:page_title, gettext("Listing Tags")) |> assign(:tag, nil) |     socket |> assign(:page_title, gettext("Tags")) |> assign(:tag, nil) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   @impl true |   @impl true | ||||||
|   | |||||||
| @@ -80,6 +80,7 @@ defmodule CanneryWeb.Router do | |||||||
|     live "/ammo_groups/:id/show/edit", AmmoGroupLive.Show, :edit |     live "/ammo_groups/:id/show/edit", AmmoGroupLive.Show, :edit | ||||||
|     live "/ammo_groups/:id/show/add_shot_group", AmmoGroupLive.Show, :add_shot_group |     live "/ammo_groups/:id/show/add_shot_group", AmmoGroupLive.Show, :add_shot_group | ||||||
|     live "/ammo_groups/:id/show/move", AmmoGroupLive.Show, :move |     live "/ammo_groups/:id/show/move", AmmoGroupLive.Show, :move | ||||||
|  |     live "/ammo_groups/:id/show/:shot_group_id/edit", AmmoGroupLive.Show, :edit_shot_group | ||||||
|  |  | ||||||
|     live "/range", RangeLive.Index, :index |     live "/range", RangeLive.Index, :index | ||||||
|     live "/range/:id/edit", RangeLive.Index, :edit |     live "/range/:id/edit", RangeLive.Index, :edit | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ | |||||||
|  |  | ||||||
| <div | <div | ||||||
|   id="loading" |   id="loading" | ||||||
|   class="absolute opacity-0 top-0 left-0 w-screen h-screen bg-white z-50 |   class="fixed opacity-0 top-0 left-0 w-screen h-screen bg-white z-50 | ||||||
|   flex flex-col justify-center items-center space-y-4 |   flex flex-col justify-center items-center space-y-4 | ||||||
|   transition-opacity ease-in-out duration-500" |   transition-opacity ease-in-out duration-500" | ||||||
| > | > | ||||||
| @@ -42,7 +42,7 @@ | |||||||
|  |  | ||||||
| <div | <div | ||||||
|   id="disconnect" |   id="disconnect" | ||||||
|   class="absolute opacity-0 top-0 left-0 w-screen h-screen bg-white z-50 |   class="fixed opacity-0 top-0 left-0 w-screen h-screen bg-white z-50 | ||||||
|   flex flex-col justify-center items-center space-y-4 |   flex flex-col justify-center items-center space-y-4 | ||||||
|   transition-opacity ease-in-out duration-500" |   transition-opacity ease-in-out duration-500" | ||||||
| > | > | ||||||
|   | |||||||
| @@ -1,15 +1,12 @@ | |||||||
| <!DOCTYPE html> | <!DOCTYPE html> | ||||||
| <html lang="en" class="m-0 p-0 w-full h-full"> | <html lang="en" class="m-0 p-0 w-full h-full bg-white"> | ||||||
|   <head> |   <head> | ||||||
|     <meta charset="utf-8" /> |     <meta charset="utf-8" /> | ||||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge" /> |     <meta http-equiv="X-UA-Compatible" content="IE=edge" /> | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> |     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||||
|     <%= csrf_meta_tag() %> |     <%= csrf_meta_tag() %> | ||||||
|     <%= if(assigns |> Map.has_key?(:page_title), |     <%= if(assigns |> Map.has_key?(:page_title), do: @page_title, else: "Cannery") | ||||||
|       do: "#{assigns.page_title} | Cannery", |     |> live_title_tag(suffix: " | Cannery") %> | ||||||
|       else: "Cannery" |  | ||||||
|     ) |  | ||||||
|     |> live_title_tag(suffix: "") %> |  | ||||||
|     <link phx-track-static rel="stylesheet" href={Routes.static_path(@conn, "/css/app.css")} /> |     <link phx-track-static rel="stylesheet" href={Routes.static_path(@conn, "/css/app.css")} /> | ||||||
|     <script |     <script | ||||||
|       defer |       defer | ||||||
| @@ -20,7 +17,7 @@ | |||||||
|     </script> |     </script> | ||||||
|   </head> |   </head> | ||||||
|  |  | ||||||
|   <body class="m-0 p-0 w-full h-full"> |   <body class="m-0 p-0 w-full h-full bg-white"> | ||||||
|     <%= @inner_content %> |     <%= @inner_content %> | ||||||
|   </body> |   </body> | ||||||
| </html> | </html> | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ defmodule CanneryWeb.EmailView do | |||||||
|   @moduledoc """ |   @moduledoc """ | ||||||
|   A view for email-related helper functions |   A view for email-related helper functions | ||||||
|   """ |   """ | ||||||
|   alias CanneryWeb.{Endpoint, HomeLive} |  | ||||||
|  |  | ||||||
|   use CanneryWeb, :view |   use CanneryWeb, :view | ||||||
|  |   alias CanneryWeb.{Endpoint, HomeLive} | ||||||
| end | end | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ defmodule CanneryWeb.ErrorView do | |||||||
|       case error_path do |       case error_path do | ||||||
|         "404.html" -> dgettext("errors", "Not found") |         "404.html" -> dgettext("errors", "Not found") | ||||||
|         "401.html" -> dgettext("errors", "Unauthorized") |         "401.html" -> dgettext("errors", "Unauthorized") | ||||||
|         _ -> dgettext("errors", "Internal Server Error") |         _other_template -> dgettext("errors", "Internal Server Error") | ||||||
|       end |       end | ||||||
|  |  | ||||||
|     render("error.html", %{error_string: error_string}) |     render("error.html", %{error_string: error_string}) | ||||||
|   | |||||||
| @@ -7,6 +7,8 @@ defmodule CanneryWeb.ViewHelpers do | |||||||
|  |  | ||||||
|   import Phoenix.LiveView.Helpers |   import Phoenix.LiveView.Helpers | ||||||
|  |  | ||||||
|  |   @id_length 16 | ||||||
|  |  | ||||||
|   @doc """ |   @doc """ | ||||||
|   Returns a <time> element that renders the naivedatetime in the user's local |   Returns a <time> element that renders the naivedatetime in the user's local | ||||||
|   timezone with Alpine.js |   timezone with Alpine.js | ||||||
| @@ -16,11 +18,12 @@ defmodule CanneryWeb.ViewHelpers do | |||||||
|  |  | ||||||
|   def display_datetime(datetime) do |   def display_datetime(datetime) do | ||||||
|     assigns = %{ |     assigns = %{ | ||||||
|  |       id: :crypto.strong_rand_bytes(@id_length) |> Base.url_encode64(), | ||||||
|       datetime: datetime |> DateTime.from_naive!("Etc/UTC") |> DateTime.to_iso8601(:extended) |       datetime: datetime |> DateTime.from_naive!("Etc/UTC") |> DateTime.to_iso8601(:extended) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ~H""" |     ~H""" | ||||||
|     <time datetime={@datetime} x-data={"{ |     <time id={@id} datetime={@datetime} x-data={"{ | ||||||
|         date: |         date: | ||||||
|           Intl.DateTimeFormat([], {dateStyle: 'short', timeStyle: 'long'}) |           Intl.DateTimeFormat([], {dateStyle: 'short', timeStyle: 'long'}) | ||||||
|             .format(new Date(\"#{@datetime}\")) |             .format(new Date(\"#{@datetime}\")) | ||||||
| @@ -38,10 +41,13 @@ defmodule CanneryWeb.ViewHelpers do | |||||||
|   def display_date(nil), do: "" |   def display_date(nil), do: "" | ||||||
|  |  | ||||||
|   def display_date(date) do |   def display_date(date) do | ||||||
|     assigns = %{date: date |> Date.to_iso8601(:extended)} |     assigns = %{ | ||||||
|  |       id: :crypto.strong_rand_bytes(@id_length) |> Base.url_encode64(), | ||||||
|  |       date: date |> Date.to_iso8601(:extended) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     ~H""" |     ~H""" | ||||||
|     <time datetime={@date} x-data={"{ |     <time id={@id} datetime={@date} x-data={"{ | ||||||
|         date: |         date: | ||||||
|           Intl.DateTimeFormat([], {timeZone: 'Etc/UTC', dateStyle: 'short'}).format(new Date(\"#{@date}\")) |           Intl.DateTimeFormat([], {timeZone: 'Etc/UTC', dateStyle: 'short'}).format(new Date(\"#{@date}\")) | ||||||
|       }"} x-text="date"> |       }"} x-text="date"> | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								mix.exs
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								mix.exs
									
									
									
									
									
								
							| @@ -4,7 +4,7 @@ defmodule Cannery.MixProject do | |||||||
|   def project do |   def project do | ||||||
|     [ |     [ | ||||||
|       app: :cannery, |       app: :cannery, | ||||||
|       version: "0.2.1", |       version: "0.4.1", | ||||||
|       elixir: "~> 1.12", |       elixir: "~> 1.12", | ||||||
|       elixirc_paths: elixirc_paths(Mix.env()), |       elixirc_paths: elixirc_paths(Mix.env()), | ||||||
|       compilers: [:gettext] ++ Mix.compilers(), |       compilers: [:gettext] ++ Mix.compilers(), | ||||||
|   | |||||||
| @@ -11,12 +11,12 @@ msgid "" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:42 | #: lib/cannery_web/live/ammo_group_live/index.ex:44 | ||||||
| msgid "Add Ammo" | msgid "Add Ammo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:12 | #: lib/cannery_web/live/ammo_group_live/index.html.heex:24 | ||||||
| msgid "Add your first box!" | msgid "Add your first box!" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -81,7 +81,7 @@ msgid "Make your first tag!" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:17 | #: lib/cannery_web/live/ammo_group_live/index.html.heex:42 | ||||||
| msgid "New Ammo group" | msgid "New Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -124,9 +124,9 @@ msgid "Reset password" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:42 | #: lib/cannery_web/components/add_shot_group_component.html.heex:46 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:54 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:73 | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:172 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:156 | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:50 | #: lib/cannery_web/live/container_live/form_component.html.heex:50 | ||||||
| #: lib/cannery_web/live/invite_live/form_component.html.heex:28 | #: lib/cannery_web/live/invite_live/form_component.html.heex:28 | ||||||
| #: lib/cannery_web/live/range_live/form_component.html.heex:40 | #: lib/cannery_web/live/range_live/form_component.html.heex:40 | ||||||
| @@ -160,7 +160,7 @@ msgid "Why not get some ready to shoot?" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:85 | #: lib/cannery_web/live/ammo_group_live/index.ex:134 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:86 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:86 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:36 | #: lib/cannery_web/live/range_live/index.html.heex:36 | ||||||
| msgid "Record shots" | msgid "Record shots" | ||||||
| @@ -172,7 +172,7 @@ msgid "Ammo Details" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:12 | #: lib/cannery_web/components/move_ammo_group_component.ex:89 | ||||||
| msgid "Add another container!" | msgid "Add another container!" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -182,7 +182,7 @@ msgid "Move containers" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:60 | #: lib/cannery_web/components/move_ammo_group_component.ex:127 | ||||||
| msgid "Select" | msgid "Select" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -190,3 +190,14 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/invite_live/index.html.heex:33 | #: lib/cannery_web/live/invite_live/index.html.heex:33 | ||||||
| msgid "Copy to clipboard" | msgid "Copy to clipboard" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:18 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:36 | ||||||
|  | msgid "add a container first" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:66 | ||||||
|  | msgid "Create" | ||||||
|  | msgstr "" | ||||||
|   | |||||||
							
								
								
									
										216
									
								
								priv/gettext/de/LC_MESSAGES/actions.po
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										216
									
								
								priv/gettext/de/LC_MESSAGES/actions.po
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,216 @@ | |||||||
|  | msgid "" | ||||||
|  | msgstr "" | ||||||
|  | "Project-Id-Version: PACKAGE VERSION\n" | ||||||
|  | "Report-Msgid-Bugs-To: \n" | ||||||
|  | "POT-Creation-Date: 2022-04-19 19:32+0000\n" | ||||||
|  | "PO-Revision-Date: 2022-04-19 21:32+0000\n" | ||||||
|  | "Last-Translator: shibao <shibao@bubbletea.dev>\n" | ||||||
|  | "Language-Team: German <https://weblate.bubbletea.dev/projects/cannery/" | ||||||
|  | "actions/de/>\n" | ||||||
|  | "Language: de\n" | ||||||
|  | "MIME-Version: 1.0\n" | ||||||
|  | "Content-Type: text/plain; charset=UTF-8\n" | ||||||
|  | "Content-Transfer-Encoding: 8bit\n" | ||||||
|  | "Plural-Forms: nplurals=2; plural=n != 1;\n" | ||||||
|  | "X-Generator: Weblate 4.11.2\n" | ||||||
|  |  | ||||||
|  | ## This file is a PO Template file. | ||||||
|  | ## | ||||||
|  | ## "msgid"s here are often extracted from source code. | ||||||
|  | ## Add new translations manually only if they're dynamic | ||||||
|  | ## translations that can't be statically extracted. | ||||||
|  | ## | ||||||
|  | ## Run "mix gettext.extract" to bring this file up to | ||||||
|  | ## date. Leave "msgstr"s empty as changing them here has no | ||||||
|  | ## effect: edit them in PO (.po) files instead. | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:44 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Add Ammo" | ||||||
|  | msgstr "Munition hinzufügen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:24 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Add your first box!" | ||||||
|  | msgstr "Fügen Sie ihre erste Box hinzu!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.html.heex:12 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Add your first container!" | ||||||
|  | msgstr "Fügen Sie ihren ersten Behälter hinzu!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.html.heex:12 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Add your first type!" | ||||||
|  | msgstr "Fügen Sie ihre erste Munitionsart hinzu!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:16 | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:45 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Change email" | ||||||
|  | msgstr "Mailadresse ändern" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:60 | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:101 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Change password" | ||||||
|  | msgstr "Passwort ändern" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:17 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Create Invite" | ||||||
|  | msgstr "Einladung erstellen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:108 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Delete User" | ||||||
|  | msgstr "Benutzer löschen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_registration/new.html.heex:43 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/new.html.heex:3 | ||||||
|  | #: lib/cannery_web/templates/user_session/new.html.heex:45 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Forgot your password?" | ||||||
|  | msgstr "Passwort vergessen?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:12 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Invite someone new!" | ||||||
|  | msgstr "Laden Sie jemanden ein!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:108 | ||||||
|  | #: lib/cannery_web/templates/user_confirmation/new.html.heex:30 | ||||||
|  | #: lib/cannery_web/templates/user_registration/new.html.heex:39 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/edit.html.heex:48 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/new.html.heex:30 | ||||||
|  | #: lib/cannery_web/templates/user_session/new.html.heex:3 | ||||||
|  | #: lib/cannery_web/templates/user_session/new.html.heex:33 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Log in" | ||||||
|  | msgstr "Einloggen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/index.html.heex:14 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Make your first tag!" | ||||||
|  | msgstr "Erstellen Sie ihren ersten Tag!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:42 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Ammo group" | ||||||
|  | msgstr "Neue Munitionsgruppe" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.html.heex:17 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Ammo type" | ||||||
|  | msgstr "Neue Munitionsart" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.html.heex:17 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Container" | ||||||
|  | msgstr "Neuer Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/index.html.heex:19 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Tag" | ||||||
|  | msgstr "Neuer Tag" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:101 | ||||||
|  | #: lib/cannery_web/templates/user_confirmation/new.html.heex:25 | ||||||
|  | #: lib/cannery_web/templates/user_registration/new.html.heex:3 | ||||||
|  | #: lib/cannery_web/templates/user_registration/new.html.heex:33 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/edit.html.heex:43 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/new.html.heex:25 | ||||||
|  | #: lib/cannery_web/templates/user_session/new.html.heex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Register" | ||||||
|  | msgstr "Registrieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_confirmation/new.html.heex:3 | ||||||
|  | #: lib/cannery_web/templates/user_confirmation/new.html.heex:16 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Resend confirmation instructions" | ||||||
|  | msgstr "Bestätigungsmail erneut senden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/edit.html.heex:3 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/edit.html.heex:34 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Reset password" | ||||||
|  | msgstr "Passwort zurücksetzen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.html.heex:46 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:73 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:156 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:50 | ||||||
|  | #: lib/cannery_web/live/invite_live/form_component.html.heex:28 | ||||||
|  | #: lib/cannery_web/live/range_live/form_component.html.heex:40 | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:66 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Save" | ||||||
|  | msgstr "Speichern" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/new.html.heex:16 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Send instructions to reset password" | ||||||
|  | msgstr "Anleitung zum Passwort zurücksetzen zusenden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:53 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Why not add one?" | ||||||
|  | msgstr "Warum fügen Sie keine hinzu?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.html.heex:52 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Add" | ||||||
|  | msgstr "Hinzufügen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:17 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Stage ammo" | ||||||
|  | msgstr "Munition markieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:12 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Why not get some ready to shoot?" | ||||||
|  | msgstr "Warum nicht einige für den Schießstand auswählen?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:134 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:86 | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:36 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Record shots" | ||||||
|  | msgstr "Schüsse dokumentieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:50 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo Details" | ||||||
|  | msgstr "Munitionsdetails" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:89 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Add another container!" | ||||||
|  | msgstr "Einen weiteren Behälter hinzufügen!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:80 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Move containers" | ||||||
|  | msgstr "Behälter verschieben" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:127 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Select" | ||||||
|  | msgstr "Markieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:33 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Copy to clipboard" | ||||||
|  | msgstr "In die Zwischenablage kopieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:18 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:36 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "add a container first" | ||||||
|  | msgstr "Zuerst einen Behälter hinzufügen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:66 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Create" | ||||||
|  | msgstr "Erstellen" | ||||||
							
								
								
									
										846
									
								
								priv/gettext/de/LC_MESSAGES/default.po
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										846
									
								
								priv/gettext/de/LC_MESSAGES/default.po
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,846 @@ | |||||||
|  | msgid "" | ||||||
|  | msgstr "" | ||||||
|  | "Project-Id-Version: PACKAGE VERSION\n" | ||||||
|  | "Report-Msgid-Bugs-To: \n" | ||||||
|  | "POT-Creation-Date: 2022-04-19 19:32+0000\n" | ||||||
|  | "PO-Revision-Date: 2022-04-19 21:32+0000\n" | ||||||
|  | "Last-Translator: shibao <shibao@bubbletea.dev>\n" | ||||||
|  | "Language-Team: German <https://weblate.bubbletea.dev/projects/cannery/" | ||||||
|  | "default/de/>\n" | ||||||
|  | "Language: de\n" | ||||||
|  | "MIME-Version: 1.0\n" | ||||||
|  | "Content-Type: text/plain; charset=UTF-8\n" | ||||||
|  | "Content-Transfer-Encoding: 8bit\n" | ||||||
|  | "Plural-Forms: nplurals=2; plural=n != 1;\n" | ||||||
|  | "X-Generator: Weblate 4.11.2\n" | ||||||
|  |  | ||||||
|  | ## This file is a PO Template file. | ||||||
|  | ## | ||||||
|  | ## "msgid"s here are often extracted from source code. | ||||||
|  | ## Add new translations manually only if they're dynamic | ||||||
|  | ## translations that can't be statically extracted. | ||||||
|  | ## | ||||||
|  | ## Run "mix gettext.extract" to bring this file up to | ||||||
|  | ## date. Leave "msgstr"s empty as changing them here has no | ||||||
|  | ## effect: edit them in PO (.po) files instead. | ||||||
|  | #: lib/cannery_web/live/home_live.ex:63 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} lets you easily keep an eye on your ammo levels before and after range day" | ||||||
|  | msgstr "" | ||||||
|  | "Mit %{name} können Sie ihren Munitionsbestand vor und nach dem Schießen " | ||||||
|  | "leicht im Auge behalten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:85 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Access from any internet-capable device" | ||||||
|  | msgstr "Zugriff von jedem Internet-fähigen Gerät" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:90 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Admins" | ||||||
|  | msgstr "Admins" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:99 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Admins:" | ||||||
|  | msgstr "Admins:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:52 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:3 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:82 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo" | ||||||
|  | msgstr "Munition" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:21 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:80 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo type" | ||||||
|  | msgstr "Munitionsarten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:94 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Average Price paid" | ||||||
|  | msgstr "Durchschnittlicher Kaufpreis" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:54 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Background color" | ||||||
|  | msgstr "Hintergrundfarbe" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:140 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:71 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:55 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Blank" | ||||||
|  | msgstr "Knallpatrone" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:68 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Brass" | ||||||
|  | msgstr "Messing" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:44 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:53 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:41 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Bullet core" | ||||||
|  | msgstr "Projektilkern" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:37 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:52 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Bullet type" | ||||||
|  | msgstr "Patronenart" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:58 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:55 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:43 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Caliber" | ||||||
|  | msgstr "Kaliber" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:51 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:54 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:42 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Cartridge" | ||||||
|  | msgstr "Patrone" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:65 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:56 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:44 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Case material" | ||||||
|  | msgstr "Gehäusematerial" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:67 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:48 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:85 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Container" | ||||||
|  | msgstr "Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:46 | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:38 | ||||||
|  | #: lib/cannery_web/live/container_live/index.html.heex:3 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Containers" | ||||||
|  | msgstr "Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:144 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:72 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:56 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Corrosive" | ||||||
|  | msgstr "Korrosiv" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:27 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:81 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Count" | ||||||
|  | msgstr "Anzahl" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/ammo_group_card.ex:29 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Count:" | ||||||
|  | msgstr "Anzahl:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:24 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:27 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Description" | ||||||
|  | msgstr "Beschreibung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/container_card.ex:31 | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Description:" | ||||||
|  | msgstr "Beschreibung:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:59 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Disable" | ||||||
|  | msgstr "Deaktivieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:60 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Easy to Use:" | ||||||
|  | msgstr "Einfache Anwendung:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:38 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:42 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit Ammo group" | ||||||
|  | msgstr "Munitionsgruppe bearbeiten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:23 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.ex:47 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit Ammo type" | ||||||
|  | msgstr "Munitionstyp bearbeiten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:35 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit Invite" | ||||||
|  | msgstr "Einladung bearbeiten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/index.ex:23 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit Tag" | ||||||
|  | msgstr "Tag bearbeiten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:63 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Enable" | ||||||
|  | msgstr "Aktivieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:35 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Example bullet type abbreviations" | ||||||
|  | msgstr "Beispiel Munitionstyp Abkürzungen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "FMJ" | ||||||
|  | msgstr "VM" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:103 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:65 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:49 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Grains" | ||||||
|  | msgstr "Körner" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:136 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:70 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:54 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Incendiary" | ||||||
|  | msgstr "Brandmunition" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:94 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Instance Information" | ||||||
|  | msgstr "Instanzinformationen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/invite_card.ex:27 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Invite Disabled" | ||||||
|  | msgstr "Einladung deaktiviert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:125 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Invite Only" | ||||||
|  | msgstr "Nur mit Einladung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:71 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:43 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:3 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Invites" | ||||||
|  | msgstr "Einladungen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_session/new.html.heex:28 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Keep me logged in for 60 days" | ||||||
|  | msgstr "Für 60 Tage eingeloggt bleiben" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:69 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:42 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Location" | ||||||
|  | msgstr "Standort" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/container_card.ex:43 | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:20 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Location:" | ||||||
|  | msgstr "Standort:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:38 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Magazine, Clip, Ammo Box, etc" | ||||||
|  | msgstr "Magazin, Ladestreifen, Munitionskiste usw." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:58 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Manage" | ||||||
|  | msgstr "Verwalten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:148 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:73 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:57 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Manufacturer" | ||||||
|  | msgstr "Hersteller" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:31 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Metal ammo can with the anime girl sticker" | ||||||
|  | msgstr "Metallene Munitionskiste mit Anime-Girl-Sticker" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:23 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "My cool ammo can" | ||||||
|  | msgstr "Meine coole Munitionskiste" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:20 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:51 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:20 | ||||||
|  | #: lib/cannery_web/live/invite_live/form_component.html.heex:20 | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:50 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Name" | ||||||
|  | msgstr "Name" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:29 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Ammo type" | ||||||
|  | msgstr "Neuer Munitionstyp" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:33 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Container" | ||||||
|  | msgstr "Neuer Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:39 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Invite" | ||||||
|  | msgstr "Neue Einladung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/index.ex:29 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Tag" | ||||||
|  | msgstr "Neuer Tag" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No Ammo" | ||||||
|  | msgstr "Keine Munition" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No Ammo Types" | ||||||
|  | msgstr "Keine Munitionsarten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:114 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No ammo for this type" | ||||||
|  | msgstr "Keine Munition dieser Art" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:78 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No ammo groups in this container" | ||||||
|  | msgstr "Keine Munitionsgruppe in diesem Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No containers" | ||||||
|  | msgstr "Kein Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No invites" | ||||||
|  | msgstr "Keine Einladung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.html.heex:30 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.html.heex:10 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No tags" | ||||||
|  | msgstr "Keine Tags" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.html.heex:30 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:41 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:90 | ||||||
|  | #: lib/cannery_web/live/range_live/form_component.html.heex:29 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:84 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Notes" | ||||||
|  | msgstr "Bemerkungen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/ammo_group_card.ex:35 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:24 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Notes:" | ||||||
|  | msgstr "Bemerkungen:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:46 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "On the bookshelf" | ||||||
|  | msgstr "Auf dem Bücherregal" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:111 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:66 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:50 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Pressure" | ||||||
|  | msgstr "Druck" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:34 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:82 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Price paid" | ||||||
|  | msgstr "Kaufpreis" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/ammo_group_card.ex:42 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Price paid:" | ||||||
|  | msgstr "Kaufpreis:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:118 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:67 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:51 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Primer type" | ||||||
|  | msgstr "Zündertyp" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:124 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Public Signups" | ||||||
|  | msgstr "Öffentliche Registrierung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:72 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Secure:" | ||||||
|  | msgstr "Sicher:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:75 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Self-host your own instance, or use an instance from someone you trust." | ||||||
|  | msgstr "" | ||||||
|  | "Hosten Sie Ihre eigene Instanz oder verwenden Sie eine Instanz, der Sie " | ||||||
|  | "vertrauen." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:79 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Set Unlimited" | ||||||
|  | msgstr "Unbegrenzt setzen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:10 | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:3 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Settings" | ||||||
|  | msgstr "Einstellungen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:41 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Show Ammo group" | ||||||
|  | msgstr "Munitionsgruppen anzeigen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.ex:46 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Show Ammo type" | ||||||
|  | msgstr "Zeige Munitionsarten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:82 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Simple:" | ||||||
|  | msgstr "Einfach:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:47 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Steel" | ||||||
|  | msgstr "Stahl" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:98 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Stored in" | ||||||
|  | msgstr "Gelagert in" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:40 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.ex:34 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.html.heex:3 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Tags" | ||||||
|  | msgstr "Tags" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/index.html.heex:6 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Tags can be added to your containers to help you organize" | ||||||
|  | msgstr "Tags können zur besseren Ordnung einem Behälter hinzugefügt werden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:60 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Text color" | ||||||
|  | msgstr "Textfarbe" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:51 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "The self-hosted firearm tracker website" | ||||||
|  | msgstr "Die selbst-gehostete Website zur Verwaltung von Schusswaffen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:103 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "This ammo group is not in a container" | ||||||
|  | msgstr "Diese Munitionsgruppe ist nicht in einem Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:132 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:69 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:53 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Tracer" | ||||||
|  | msgstr "Leuchtspur" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:68 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:35 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Type" | ||||||
|  | msgstr "Art" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/container_card.ex:37 | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:14 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Type:" | ||||||
|  | msgstr "Art:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:119 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Users" | ||||||
|  | msgstr "Benutzer" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/invite_card.ex:22 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Uses Left:" | ||||||
|  | msgstr "Verbleibende Nutzung:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/form_component.html.heex:24 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Uses left" | ||||||
|  | msgstr "Verbleibende Nutzung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:47 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Welcome to %{name}" | ||||||
|  | msgstr "Willkommen %{name}" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:76 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Your data stays with you, period" | ||||||
|  | msgstr "Ihre Daten bleiben bei Ihnen, Punkt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:49 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No tags for this container" | ||||||
|  | msgstr "Keine Tags für diesen Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:64 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:84 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Range" | ||||||
|  | msgstr "Schießplatz" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:3 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Range day" | ||||||
|  | msgstr "Range Day" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:91 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:85 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Date" | ||||||
|  | msgstr "Datum" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/form_component.html.heex:21 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Shots fired" | ||||||
|  | msgstr "Schüsse abgegeben" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No ammo staged" | ||||||
|  | msgstr "Keine Munition selektiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:77 | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:33 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Stage for range" | ||||||
|  | msgstr "Für Schießplatz selektieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:76 | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:32 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Unstage from range" | ||||||
|  | msgstr "Für Schießplatz deselektieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.html.heex:3 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:26 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Record shots" | ||||||
|  | msgstr "Schüsse dokumentieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.html.heex:3 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo Types" | ||||||
|  | msgstr "Munitionsarten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:49 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo groups" | ||||||
|  | msgstr "Munitionsgruppen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.html.heex:38 | ||||||
|  | #: lib/cannery_web/live/range_live/form_component.html.heex:36 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Date (UTC)" | ||||||
|  | msgstr "Zeit (UTC)" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:39 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:34 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit Shot Records" | ||||||
|  | msgstr "Schießkladde editieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New Shot Records" | ||||||
|  | msgstr "Neue Schießkladde" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:48 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No shots recorded" | ||||||
|  | msgstr "Keine Schüsse dokumentiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.html.heex:21 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Rounds left" | ||||||
|  | msgstr "Patronen verbleibend" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:89 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:83 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Rounds shot" | ||||||
|  | msgstr "Patronen abgefeuert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:46 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Shot Records" | ||||||
|  | msgstr "Schießkladde" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:32 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Move Ammo group" | ||||||
|  | msgstr "Munitionsgruppe verschieben" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:80 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Move ammo" | ||||||
|  | msgstr "Munition verschieben" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:85 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No other containers" | ||||||
|  | msgstr "Kein weiterer Behälter" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:53 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Shot log" | ||||||
|  | msgstr "Schießkladde" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/ammo_group_card.ex:43 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:117 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:32 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:39 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:98 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "$%{amount}" | ||||||
|  | msgstr "$%{amount}" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:75 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Bimetal" | ||||||
|  | msgstr "Bimetall" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:72 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:57 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:45 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Jacket type" | ||||||
|  | msgstr "Patronenhülse" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:79 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:58 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:46 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Muzzle velocity" | ||||||
|  | msgstr "Mündungsgeschwindigkeit" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:93 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:61 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:48 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Powder grains per charge" | ||||||
|  | msgstr "Pulverkörner pro Ladung" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:89 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:59 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:47 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Powder type" | ||||||
|  | msgstr "Pulverart" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:152 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:74 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:58 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "UPC" | ||||||
|  | msgstr "UPC" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:80 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Confirm new password" | ||||||
|  | msgstr "Passwort bestätigen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:33 | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:89 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Current password" | ||||||
|  | msgstr "Derzeitiges Passwort" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:73 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "New password" | ||||||
|  | msgstr "Neues Passwort" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:131 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Stage" | ||||||
|  | msgstr "Markiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:131 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Unstage" | ||||||
|  | msgstr "Demarkiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:125 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:68 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:52 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Firing type" | ||||||
|  | msgstr "Patronenhülsenform" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/layout/live.html.heex:50 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Reconnecting..." | ||||||
|  | msgstr "Neu verbinden..." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/layout/live.html.heex:37 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Loading..." | ||||||
|  | msgstr "Lädt..." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:29 | ||||||
|  | #: lib/cannery_web/live/container_live/show.ex:97 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit %{name}" | ||||||
|  | msgstr "%{name} bearbeiten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:48 | ||||||
|  | #: lib/cannery_web/live/container_live/show.ex:98 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Edit %{name} tags" | ||||||
|  | msgstr "Editiere %{name} Tags" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/container_card.ex:50 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Rounds:" | ||||||
|  | msgstr "Patronen:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/show.ex:96 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Show %{name}" | ||||||
|  | msgstr "Zeige %{name}" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:104 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "No cost information" | ||||||
|  | msgstr "Keine Preisinformationen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:83 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "% left" | ||||||
|  | msgstr "% verbleibend" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:38 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Current value:" | ||||||
|  | msgstr "Derzeitiger Wert:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:31 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Original cost:" | ||||||
|  | msgstr "Originalpreis:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:13 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Original count:" | ||||||
|  | msgstr "Ursprüngliche Anzahl:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:18 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Percentage left:" | ||||||
|  | msgstr "Prozent verbleibend:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:111 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Rounds used" | ||||||
|  | msgstr "Patronen verbraucht" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:77 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Current # of rounds:" | ||||||
|  | msgstr "Derzeitige # an Patronen:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:86 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Total # of rounds" | ||||||
|  | msgstr "Summe aller Patronen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:85 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Total rounds shot:" | ||||||
|  | msgstr "Summe abgegebener Schüsse:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_confirmation_controller.ex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Confirm your account" | ||||||
|  | msgstr "Bestätigen Sie ihr Nutzerkonto" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:9 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Forgot your password?" | ||||||
|  | msgstr "Passwort vergessen?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_session_controller.ex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Log in" | ||||||
|  | msgstr "Einloggen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:35 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Register" | ||||||
|  | msgstr "Registrieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:36 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Reset your password" | ||||||
|  | msgstr "Passwort zurücksetzen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:38 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:28 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Record Shots" | ||||||
|  | msgstr "Schüsse dokumentieren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:58 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Copies" | ||||||
|  | msgstr "Kopien" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:34 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo types" | ||||||
|  | msgstr "Munitionsart" | ||||||
							
								
								
									
										121
									
								
								priv/gettext/de/LC_MESSAGES/emails.po
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								priv/gettext/de/LC_MESSAGES/emails.po
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | |||||||
|  | msgid "" | ||||||
|  | msgstr "" | ||||||
|  | "Project-Id-Version: PACKAGE VERSION\n" | ||||||
|  | "Report-Msgid-Bugs-To: \n" | ||||||
|  | "POT-Creation-Date: 2022-04-19 19:32+0000\n" | ||||||
|  | "PO-Revision-Date: 2022-04-19 21:44+0000\n" | ||||||
|  | "Last-Translator: Kaia Estra <kaia@fedora.email>\n" | ||||||
|  | "Language-Team: German <https://weblate.bubbletea.dev/projects/cannery/emails/" | ||||||
|  | "de/>\n" | ||||||
|  | "Language: de\n" | ||||||
|  | "MIME-Version: 1.0\n" | ||||||
|  | "Content-Type: text/plain; charset=UTF-8\n" | ||||||
|  | "Content-Transfer-Encoding: 8bit\n" | ||||||
|  | "Plural-Forms: nplurals=2; plural=n != 1;\n" | ||||||
|  | "X-Generator: Weblate 4.11.2\n" | ||||||
|  |  | ||||||
|  | ## This file is a PO Template file. | ||||||
|  | ## | ||||||
|  | ## "msgid"s here are often extracted from source code. | ||||||
|  | ## Add new translations manually only if they're dynamic | ||||||
|  | ## translations that can't be statically extracted. | ||||||
|  | ## | ||||||
|  | ## Run "mix gettext.extract" to bring this file up to | ||||||
|  | ## date. Leave "msgstr"s empty as changing them here has no | ||||||
|  | ## effect: edit them in PO (.po) files instead. | ||||||
|  | #: lib/cannery/accounts/email.ex:30 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Confirm your %{name} account" | ||||||
|  | msgstr "Bestätigen Sie ihr %{name} Nutzerkonto" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.html.eex:3 | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.txt.eex:2 | ||||||
|  | #: lib/cannery_web/templates/email/reset_password.html.eex:3 | ||||||
|  | #: lib/cannery_web/templates/email/reset_password.txt.eex:2 | ||||||
|  | #: lib/cannery_web/templates/email/update_email.html.eex:3 | ||||||
|  | #: lib/cannery_web/templates/email/update_email.txt.eex:2 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Hi %{email}," | ||||||
|  | msgstr "Hallo %{email}," | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.txt.eex:10 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "If you didn't create an account at %{url}, please ignore this." | ||||||
|  | msgstr "" | ||||||
|  | "Falls Sie dieses Nutzerkonto bei %{url} nicht erstellt haben, ignorieren Sie " | ||||||
|  | "diese Nachricht bitte." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/reset_password.txt.eex:8 | ||||||
|  | #: lib/cannery_web/templates/email/update_email.txt.eex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "If you didn't request this change from %{url}, please ignore this." | ||||||
|  | msgstr "" | ||||||
|  | "Falls Sie diese Änderung von %{url} nicht angefordert haben, ignorieren Sie " | ||||||
|  | "bitte diese Nachricht." | ||||||
|  |  | ||||||
|  | #: lib/cannery/accounts/email.ex:37 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Reset your %{name} password" | ||||||
|  | msgstr "Passwort für %{name} zurücksetzen" | ||||||
|  |  | ||||||
|  | #: lib/cannery/accounts/email.ex:44 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Update your %{name} email" | ||||||
|  | msgstr "Aktualisieren Sie %{name} Mailadresse" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.html.eex:9 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Welcome to %{name}!" | ||||||
|  | msgstr "Willkommen %{name}!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.txt.eex:4 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Welcome to %{name}%!" | ||||||
|  | msgstr "Willkommen %{name}%!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/update_email.html.eex:8 | ||||||
|  | #: lib/cannery_web/templates/email/update_email.txt.eex:4 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You can change your email by visiting the URL below:" | ||||||
|  | msgstr "Sie können Ihre Mailadresse unter folgender URL ändern:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.html.eex:14 | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.txt.eex:6 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You can confirm your account by visiting the URL below:" | ||||||
|  | msgstr "Sie können Ihr Nutzerkonto unter folgender URL bestätigen:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/reset_password.html.eex:8 | ||||||
|  | #: lib/cannery_web/templates/email/reset_password.txt.eex:4 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You can reset your password by visiting the URL below:" | ||||||
|  | msgstr "Sie können ihr Passwort unter folgender URL zurücksetzen:" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/confirm_email.html.eex:22 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "If you didn't create an account at %{name}, please ignore this." | ||||||
|  | msgstr "" | ||||||
|  | "Falls SIe dieses Nutzerkonto unter %{name}, nicht erstellt haben, ignorieren " | ||||||
|  | "Sie diese Nachricht bitte." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/email/reset_password.html.eex:16 | ||||||
|  | #: lib/cannery_web/templates/email/update_email.html.eex:16 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "If you didn't request this change from %{name}, please ignore this." | ||||||
|  | msgstr "" | ||||||
|  | "Falls Sie die Änderung von %{name} nicht angefragt haben, ignorieren Sie " | ||||||
|  | "diese Nachricht bitte." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/layout/email.txt.eex:9 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "This email was sent from %{name} at %{url}, the self-hosted firearm tracker website." | ||||||
|  | msgstr "" | ||||||
|  | "Diese Nachricht wurde von %{name} unter %{url} gesandt, einem selbst-" | ||||||
|  | "gehosteten Schusswaffenmanager." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/layout/email.html.heex:17 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "This email was sent from %{name}, the self-hosted firearm tracker website." | ||||||
|  | msgstr "" | ||||||
|  | "Diese Nachricht wurde von %{name} gesandt, einem selbst-gehosteten " | ||||||
|  | "Schusswaffenmanager." | ||||||
							
								
								
									
										187
									
								
								priv/gettext/de/LC_MESSAGES/errors.po
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										187
									
								
								priv/gettext/de/LC_MESSAGES/errors.po
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,187 @@ | |||||||
|  | msgid "" | ||||||
|  | msgstr "" | ||||||
|  | "Project-Id-Version: PACKAGE VERSION\n" | ||||||
|  | "Report-Msgid-Bugs-To: \n" | ||||||
|  | "POT-Creation-Date: 2022-04-19 19:32+0000\n" | ||||||
|  | "PO-Revision-Date: 2022-04-19 21:32+0000\n" | ||||||
|  | "Last-Translator: shibao <shibao@bubbletea.dev>\n" | ||||||
|  | "Language-Team: German <https://weblate.bubbletea.dev/projects/cannery/errors/" | ||||||
|  | "de/>\n" | ||||||
|  | "Language: de\n" | ||||||
|  | "MIME-Version: 1.0\n" | ||||||
|  | "Content-Type: text/plain; charset=UTF-8\n" | ||||||
|  | "Content-Transfer-Encoding: 8bit\n" | ||||||
|  | "Plural-Forms: nplurals=2; plural=n != 1;\n" | ||||||
|  | "X-Generator: Weblate 4.11.2\n" | ||||||
|  |  | ||||||
|  | ## This file is a PO Template file. | ||||||
|  | ## | ||||||
|  | ## "msgid"s here are often extracted from source code. | ||||||
|  | ## Add new translations manually only if they're dynamic | ||||||
|  | ## translations that can't be statically extracted. | ||||||
|  | ## | ||||||
|  | ## Run "mix gettext.extract" to bring this file up to | ||||||
|  | ## date. Leave "msgstr"s empty as changing them here has no | ||||||
|  | ## effect: edit them in PO (.po) files instead. | ||||||
|  | #: lib/cannery/containers.ex:122 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Container must be empty before deleting" | ||||||
|  | msgstr "Behälter muss vor dem Löschen leer sein" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:71 | ||||||
|  | #: lib/cannery_web/live/container_live/show.ex:73 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Could not delete %{name}: %{error}" | ||||||
|  | msgstr "Konnte %{name} nicht löschen: %{error}" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:59 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Could not find that container" | ||||||
|  | msgstr "Konnte Behälter nicht finden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:67 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Email change link is invalid or it has expired." | ||||||
|  | msgstr "Mailadressenänderungs-Link ist ungültig oder abgelaufen." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/error/error.html.heex:8 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Error" | ||||||
|  | msgstr "Fehler" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/error/error.html.heex:29 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Go back home" | ||||||
|  | msgstr "Zur Hauptseite zurückkehren" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/views/error_view.ex:11 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Internal Server Error" | ||||||
|  | msgstr "Interner Serverfehler" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_session_controller.ex:17 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Invalid email or password" | ||||||
|  | msgstr "Ungültige Mailadresse oder Passwort" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/views/error_view.ex:9 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Not found" | ||||||
|  | msgstr "Nicht gefunden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_registration/new.html.heex:16 | ||||||
|  | #: lib/cannery_web/templates/user_reset_password/edit.html.heex:16 | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:22 | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:66 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Oops, something went wrong! Please check the errors below." | ||||||
|  | msgstr "Oops, etwas ist schiefgegangen. Bitte beachten Sie den Fehler unten." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:63 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Reset password link is invalid or it has expired." | ||||||
|  | msgstr "Link zum Passwort zurücksetzen ist ungültig oder abgelaufen." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:25 | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:56 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Sorry, public registration is disabled" | ||||||
|  | msgstr "Entschuldigung, aber öffentliche Registrierung ist deaktiviert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:15 | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:46 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Sorry, this invite was not found or expired" | ||||||
|  | msgstr "" | ||||||
|  | "Entschuldigung, aber diese Einladung wurde nicht gefunden oder ist abgelaufen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:82 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Unable to delete user" | ||||||
|  | msgstr "Dieser Nutzer konnte nicht gelöscht werden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/views/error_view.ex:10 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Unauthorized" | ||||||
|  | msgstr "Unbefugt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_confirmation_controller.ex:54 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "User confirmation link is invalid or it has expired." | ||||||
|  | msgstr "Nutzerkonto Bestätigungslink ist ungültig oder abgelaufen." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:20 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You are not authorized to view this page" | ||||||
|  | msgstr "Sie sind nicht berechtigt, diese Seite aufzurufen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_auth.ex:177 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You are not authorized to view this page." | ||||||
|  | msgstr "Sie sind nicht berechtigt, diese Seite aufzurufen." | ||||||
|  |  | ||||||
|  | #: lib/cannery/accounts/user.ex:128 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "did not change" | ||||||
|  | msgstr "hat sich nicht geändert" | ||||||
|  |  | ||||||
|  | #: lib/cannery/accounts/user.ex:149 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "does not match password" | ||||||
|  | msgstr "Passwort stimmt nicht überein" | ||||||
|  |  | ||||||
|  | #: lib/cannery/accounts/user.ex:186 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "is not valid" | ||||||
|  | msgstr "ist nicht gültig" | ||||||
|  |  | ||||||
|  | #: lib/cannery/accounts/user.ex:82 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "must have the @ sign and no spaces" | ||||||
|  | msgstr "Muss ein @ Zeichen und keine Leerzeichen haben" | ||||||
|  |  | ||||||
|  | #: lib/cannery/tags.ex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Tag not found" | ||||||
|  | msgstr "Tag nicht gefunden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.ex:30 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Tag could not be added" | ||||||
|  | msgstr "Tag konnte nicht hinzugefügt werden" | ||||||
|  |  | ||||||
|  | #: lib/cannery/activity_log.ex:125 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Count must be at least 1" | ||||||
|  | msgstr "Anzahl muss mindestens 1 sein" | ||||||
|  |  | ||||||
|  | #: lib/cannery/activity_log.ex:73 | ||||||
|  | #: lib/cannery/activity_log.ex:120 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Count must be less than %{count}" | ||||||
|  | msgstr "Anzahl muss weniger als %{count} betragen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_auth.ex:39 | ||||||
|  | #: lib/cannery_web/controllers/user_auth.ex:161 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You must confirm your account and log in to access this page." | ||||||
|  | msgstr "" | ||||||
|  | "Sie müssen ihr Nutzerkonto bestätigen und einloggen, um diese Seite " | ||||||
|  | "anzuzeigen." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.ex:52 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Tag could not be removed" | ||||||
|  | msgstr "Tag konnte nicht gelöscht werden" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:126 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Could not parse number of copies" | ||||||
|  | msgstr "Konnte die Anzahl der Kopien nicht verstehen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:111 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Invalid number of copies, must be between 1 and %{max}. Was %{multiplier}" | ||||||
|  | msgstr "" | ||||||
|  | "Ungültige Nummer an Kopien. Muss zwischen 1 and %{max} liegen. War " | ||||||
|  | "%{multiplier}" | ||||||
							
								
								
									
										281
									
								
								priv/gettext/de/LC_MESSAGES/prompts.po
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										281
									
								
								priv/gettext/de/LC_MESSAGES/prompts.po
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,281 @@ | |||||||
|  | msgid "" | ||||||
|  | msgstr "" | ||||||
|  | "Project-Id-Version: PACKAGE VERSION\n" | ||||||
|  | "Report-Msgid-Bugs-To: \n" | ||||||
|  | "POT-Creation-Date: 2022-04-19 19:32+0000\n" | ||||||
|  | "PO-Revision-Date: 2022-04-19 21:32+0000\n" | ||||||
|  | "Last-Translator: Kaia Estra <kaia@fedora.email>\n" | ||||||
|  | "Language-Team: German <https://weblate.bubbletea.dev/projects/cannery/" | ||||||
|  | "prompts/de/>\n" | ||||||
|  | "Language: de\n" | ||||||
|  | "MIME-Version: 1.0\n" | ||||||
|  | "Content-Type: text/plain; charset=UTF-8\n" | ||||||
|  | "Content-Transfer-Encoding: 8bit\n" | ||||||
|  | "Plural-Forms: nplurals=2; plural=n != 1;\n" | ||||||
|  | "X-Generator: Weblate 4.11.2\n" | ||||||
|  |  | ||||||
|  | ## This file is a PO Template file. | ||||||
|  | ## | ||||||
|  | ## "msgid"s here are often extracted from source code. | ||||||
|  | ## Add new translations manually only if they're dynamic | ||||||
|  | ## translations that can't be statically extracted. | ||||||
|  | ## | ||||||
|  | ## Run "mix gettext.extract" to bring this file up to | ||||||
|  | ## date. Leave "msgstr"s empty as changing them here has no | ||||||
|  | ## effect: edit them in PO (.po) files instead. | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.ex:64 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.ex:65 | ||||||
|  | #: lib/cannery_web/live/invite_live/form_component.ex:59 | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:101 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} created successfully" | ||||||
|  | msgstr "%{name} erfolgreich erstellt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:41 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.ex:40 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:55 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:135 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.ex:40 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} deleted succesfully" | ||||||
|  | msgstr "%{name} erfolgreich gelöscht" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:111 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} disabled succesfully" | ||||||
|  | msgstr "%{name} erfolgreich deaktiviert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:89 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} enabled succesfully" | ||||||
|  | msgstr "%{name} erfolgreich aktiviert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:64 | ||||||
|  | #: lib/cannery_web/live/container_live/show.ex:63 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} has been deleted" | ||||||
|  | msgstr "%{name} wurde gelöscht" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:69 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} updated succesfully" | ||||||
|  | msgstr "%{name} erfolgreich aktualisiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.ex:46 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.ex:47 | ||||||
|  | #: lib/cannery_web/live/invite_live/form_component.ex:41 | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:83 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} updated successfully" | ||||||
|  | msgstr "%{name} erfolgreich aktualisiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:28 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "A link to confirm your email change has been sent to the new address." | ||||||
|  | msgstr "Eine Mail zum Bestätigen ihre Mailadresse wurde Ihnen zugesandt." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:56 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:52 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo group deleted succesfully" | ||||||
|  | msgstr "Munitionsgruppe erfolgreich gelöscht" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:88 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo group updated successfully" | ||||||
|  | msgstr "Munitionsgruppe erfolgreich aktualisiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:102 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:131 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to delete %{email}? This action is permanent!" | ||||||
|  | msgstr "" | ||||||
|  | "Sind Sie sicher, dass sie %{email} löschen möchten? Dies kann nicht " | ||||||
|  | "zurückgenommen werden!" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/show.html.heex:29 | ||||||
|  | #: lib/cannery_web/live/container_live/index.html.heex:46 | ||||||
|  | #: lib/cannery_web/live/container_live/show.html.heex:37 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.html.heex:38 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to delete %{name}?" | ||||||
|  | msgstr "Sind Sie sicher, dass sie %{name} löschen möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:49 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to delete the invite for %{name}?" | ||||||
|  | msgstr "Sind Sie sicher, dass sie die Einladung für %{name} löschen möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.ex:167 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.html.heex:66 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:130 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to delete this ammo?" | ||||||
|  | msgstr "Sind Sie sicher, dass sie diese Munition löschen möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/templates/user_settings/edit.html.heex:112 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to delete your account?" | ||||||
|  | msgstr "Sind Sie sicher, dass sie Ihren Account löschen möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/topbar.ex:86 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to log out?" | ||||||
|  | msgstr "Wirklich ausloggen?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.html.heex:74 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to make %{name} unlimited?" | ||||||
|  | msgstr "Sind Sie sicher, dass sie %{name} auf unbegrenzt setzen möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:60 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Email changed successfully." | ||||||
|  | msgstr "Mailadresse erfolgreich geändert." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_confirmation_controller.ex:23 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly." | ||||||
|  | msgstr "" | ||||||
|  | "Falls Ihre Mailadresse bereits in unserer Datenbank ist und noch nicht " | ||||||
|  | "bestätigt wurde, erhalten Sie gleich eine Mail mit Anweisungen." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:24 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "If your email is in our system, you will receive instructions to reset your password shortly." | ||||||
|  | msgstr "" | ||||||
|  | "Falls Ihre Mailadresse bereits in unserer Datenbank ist, erhalten Sie gleich " | ||||||
|  | "eine Mail mit Anweisungen zum Ändern ihres Passworts." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_session_controller.ex:23 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Logged out successfully." | ||||||
|  | msgstr "Erfolgreich ausgeloggt." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:46 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Password reset successfully." | ||||||
|  | msgstr "Passwort erfolgreich zurückgesetzt." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:47 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Password updated successfully." | ||||||
|  | msgstr "Passwort erfolgreich geändert." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:74 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Please check your email to verify your account" | ||||||
|  | msgstr "Bitte überprüfen Sie ihre Mailbox und bestätigen Sie das Nutzerkonto" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/home_live.ex:103 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Register to setup %{name}" | ||||||
|  | msgstr "Registrieren Sie sich, um %{name} zu bearbeiten" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.html.heex:48 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:74 | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:157 | ||||||
|  | #: lib/cannery_web/live/container_live/form_component.html.heex:52 | ||||||
|  | #: lib/cannery_web/live/invite_live/form_component.html.heex:30 | ||||||
|  | #: lib/cannery_web/live/range_live/form_component.html.heex:42 | ||||||
|  | #: lib/cannery_web/live/tag_live/form_component.ex:68 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Saving..." | ||||||
|  | msgstr "Speichere..." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:78 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Your account has been deleted" | ||||||
|  | msgstr "Ihr Nutzerkonto wurde gelöscht" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.html.heex:16 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to remove the %{tag_name} tag from %{container_name}?" | ||||||
|  | msgstr "" | ||||||
|  | "Sind Sie sicher, dass sie %{tag_name} Tag von %{container_name} entfernen " | ||||||
|  | "wollen?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.ex:36 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} added successfully" | ||||||
|  | msgstr "%{name} erfolgreich hinzugefügt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/show.ex:39 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{tag_name} has been removed from %{container_name}" | ||||||
|  | msgstr "%{tag_name} wurde von %{container_name} entfernt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.html.heex:54 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Adding..." | ||||||
|  | msgstr "Füge hinzu..." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/add_shot_group_component.ex:68 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Shots recorded successfully" | ||||||
|  | msgstr "Schüsse erfolgreich dokumentiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.html.heex:29 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to unstage this ammo?" | ||||||
|  | msgstr "Sind sie sicher, dass Sie diese Munition demarkieren möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:70 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo group unstaged succesfully" | ||||||
|  | msgstr "Munition erfolgreich demarkiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:132 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:130 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Are you sure you want to delete this shot record?" | ||||||
|  | msgstr "Sind sie sicher, dass sie die Schießkladde löschen möchten?" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:80 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:56 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Shot records deleted succesfully" | ||||||
|  | msgstr "Schießkladde erfolgreich gelöscht" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/range_live/form_component.ex:55 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Shot records updated successfully" | ||||||
|  | msgstr "Schießkladde erfolgreich aktualisiert" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/controllers/user_confirmation_controller.ex:38 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{email} confirmed successfully." | ||||||
|  | msgstr "%{email} erfolgreich bestätigt." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/components/move_ammo_group_component.ex:53 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo moved to %{name} successfully" | ||||||
|  | msgstr "Munition erfolgreich zu %{name} verschoben" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:123 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Copied to clipboard" | ||||||
|  | msgstr "Der Zwischenablage hinzugefügt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/container_live/edit_tags_component.ex:58 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "%{name} removed successfully" | ||||||
|  | msgstr "%{name} erfolgreich entfernt" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:15 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:33 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "You'll need to" | ||||||
|  | msgstr "Sie müssen" | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:67 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Creating..." | ||||||
|  | msgstr "Erstellen..." | ||||||
|  |  | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:147 | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | msgid "Ammo group created successfully" | ||||||
|  | msgid_plural "Ammo groups created successfully" | ||||||
|  | msgstr[0] "Munitionsgruppe erfolgreich erstellt" | ||||||
|  | msgstr[1] "Munitionsgruppen erfolgreich erstellt" | ||||||
| @@ -33,13 +33,13 @@ msgstr "" | |||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:52 | #: lib/cannery_web/components/topbar.ex:52 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:3 | #: lib/cannery_web/live/ammo_group_live/index.html.heex:3 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:61 | #: lib/cannery_web/live/range_live/index.ex:82 | ||||||
| msgid "Ammo" | msgid "Ammo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:21 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:21 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:27 | #: lib/cannery_web/live/ammo_group_live/index.ex:80 | ||||||
| msgid "Ammo type" | msgid "Ammo type" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -54,19 +54,19 @@ msgid "Background color" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:154 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:140 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:67 | #: lib/cannery_web/live/ammo_type_live/index.ex:71 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:55 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:55 | ||||||
| msgid "Blank" | msgid "Blank" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:74 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:68 | ||||||
| msgid "Brass" | msgid "Brass" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:46 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:44 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:53 | #: lib/cannery_web/live/ammo_type_live/index.ex:53 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:41 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:41 | ||||||
| msgid "Bullet core" | msgid "Bullet core" | ||||||
| @@ -80,49 +80,50 @@ msgid "Bullet type" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:62 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:58 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:55 | #: lib/cannery_web/live/ammo_type_live/index.ex:55 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:43 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:43 | ||||||
| msgid "Caliber" | msgid "Caliber" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:55 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:51 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:54 | #: lib/cannery_web/live/ammo_type_live/index.ex:54 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:42 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:42 | ||||||
| msgid "Cartridge" | msgid "Cartridge" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:69 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:65 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:56 | #: lib/cannery_web/live/ammo_type_live/index.ex:56 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:44 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:44 | ||||||
| msgid "Case material" | msgid "Case material" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:22 | #: lib/cannery_web/components/move_ammo_group_component.ex:67 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:48 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:48 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:42 | #: lib/cannery_web/live/ammo_group_live/index.ex:85 | ||||||
| msgid "Container" | msgid "Container" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:46 | #: lib/cannery_web/components/topbar.ex:46 | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:38 | ||||||
| #: lib/cannery_web/live/container_live/index.html.heex:3 | #: lib/cannery_web/live/container_live/index.html.heex:3 | ||||||
| msgid "Containers" | msgid "Containers" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:158 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:144 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:68 | #: lib/cannery_web/live/ammo_type_live/index.ex:72 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:56 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:56 | ||||||
| msgid "Corrosive" | msgid "Corrosive" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:27 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:27 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:30 | #: lib/cannery_web/live/ammo_group_live/index.ex:81 | ||||||
| msgid "Count" | msgid "Count" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -155,8 +156,8 @@ msgid "Easy to Use:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:36 | #: lib/cannery_web/live/ammo_group_live/index.ex:38 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:55 | #: lib/cannery_web/live/ammo_group_live/show.ex:42 | ||||||
| msgid "Edit Ammo group" | msgid "Edit Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -187,20 +188,20 @@ msgid "Example bullet type abbreviations" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:42 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:40 | ||||||
| msgid "FMJ" | msgid "FMJ" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:113 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:103 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:61 | #: lib/cannery_web/live/ammo_type_live/index.ex:65 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:49 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:49 | ||||||
| msgid "Grains" | msgid "Grains" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:150 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:136 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:66 | #: lib/cannery_web/live/ammo_type_live/index.ex:70 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:54 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:54 | ||||||
| msgid "Incendiary" | msgid "Incendiary" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -222,6 +223,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:71 | #: lib/cannery_web/components/topbar.ex:71 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:43 | ||||||
| #: lib/cannery_web/live/invite_live/index.html.heex:3 | #: lib/cannery_web/live/invite_live/index.html.heex:3 | ||||||
| msgid "Invites" | msgid "Invites" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -232,27 +234,7 @@ msgid "Keep me logged in for 60 days" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:34 | #: lib/cannery_web/components/move_ammo_group_component.ex:69 | ||||||
| msgid "Listing Ammo types" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/container_live/index.ex:38 |  | ||||||
| msgid "Listing Containers" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/invite_live/index.ex:43 |  | ||||||
| msgid "Listing Invites" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/tag_live/index.ex:34 |  | ||||||
| msgid "Listing Tags" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:30 |  | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:42 | #: lib/cannery_web/live/container_live/form_component.html.heex:42 | ||||||
| msgid "Location" | msgid "Location" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -274,8 +256,8 @@ msgid "Manage" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:162 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:148 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:69 | #: lib/cannery_web/live/ammo_type_live/index.ex:73 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:57 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:57 | ||||||
| msgid "Manufacturer" | msgid "Manufacturer" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -335,7 +317,7 @@ msgid "No ammo for this type" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/container_live/show.html.heex:77 | #: lib/cannery_web/live/container_live/show.html.heex:78 | ||||||
| msgid "No ammo groups in this container" | msgid "No ammo groups in this container" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -358,9 +340,9 @@ msgstr "" | |||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:30 | #: lib/cannery_web/components/add_shot_group_component.html.heex:30 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:41 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:41 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:122 | #: lib/cannery_web/live/ammo_group_live/show.ex:90 | ||||||
| #: lib/cannery_web/live/range_live/form_component.html.heex:29 | #: lib/cannery_web/live/range_live/form_component.html.heex:29 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:67 | #: lib/cannery_web/live/range_live/index.ex:84 | ||||||
| msgid "Notes" | msgid "Notes" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -376,15 +358,15 @@ msgid "On the bookshelf" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:121 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:111 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:62 | #: lib/cannery_web/live/ammo_type_live/index.ex:66 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:50 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:50 | ||||||
| msgid "Pressure" | msgid "Pressure" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:34 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:34 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:33 | #: lib/cannery_web/live/ammo_group_live/index.ex:82 | ||||||
| msgid "Price paid" | msgid "Price paid" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -394,8 +376,8 @@ msgid "Price paid:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:128 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:118 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:63 | #: lib/cannery_web/live/ammo_type_live/index.ex:67 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:51 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:51 | ||||||
| msgid "Primer type" | msgid "Primer type" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -421,12 +403,13 @@ msgid "Set Unlimited" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:10 | ||||||
| #: lib/cannery_web/templates/user_settings/edit.html.heex:3 | #: lib/cannery_web/templates/user_settings/edit.html.heex:3 | ||||||
| msgid "Settings" | msgid "Settings" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:54 | #: lib/cannery_web/live/ammo_group_live/show.ex:41 | ||||||
| msgid "Show Ammo group" | msgid "Show Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -441,7 +424,7 @@ msgid "Simple:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:51 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:47 | ||||||
| msgid "Steel" | msgid "Steel" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -452,6 +435,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:40 | #: lib/cannery_web/components/topbar.ex:40 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.ex:34 | ||||||
| #: lib/cannery_web/live/tag_live/index.html.heex:3 | #: lib/cannery_web/live/tag_live/index.html.heex:3 | ||||||
| msgid "Tags" | msgid "Tags" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -477,14 +461,14 @@ msgid "This ammo group is not in a container" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:146 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:132 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:65 | #: lib/cannery_web/live/ammo_type_live/index.ex:69 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:53 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:53 | ||||||
| msgid "Tracer" | msgid "Tracer" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:26 | #: lib/cannery_web/components/move_ammo_group_component.ex:68 | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:35 | #: lib/cannery_web/live/container_live/form_component.html.heex:35 | ||||||
| msgid "Type" | msgid "Type" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -527,7 +511,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:64 | #: lib/cannery_web/components/topbar.ex:64 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:39 | #: lib/cannery_web/live/ammo_group_live/index.ex:84 | ||||||
| msgid "Range" | msgid "Range" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -537,8 +521,8 @@ msgid "Range day" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:125 | #: lib/cannery_web/live/ammo_group_live/show.ex:91 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:70 | #: lib/cannery_web/live/range_live/index.ex:85 | ||||||
| msgid "Date" | msgid "Date" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -564,15 +548,9 @@ msgstr "" | |||||||
| msgid "Unstage from range" | msgid "Unstage from range" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:52 |  | ||||||
| msgid "Add Shot group" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:3 | #: lib/cannery_web/components/add_shot_group_component.html.heex:3 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:24 | #: lib/cannery_web/live/ammo_group_live/index.ex:26 | ||||||
| #: lib/cannery_web/live/range_live/index.ex:28 |  | ||||||
| msgid "Record shots" | msgid "Record shots" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -582,7 +560,7 @@ msgid "Ammo Types" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:47 | #: lib/cannery_web/live/ammo_group_live/index.ex:49 | ||||||
| msgid "Ammo groups" | msgid "Ammo groups" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -593,6 +571,7 @@ msgid "Date (UTC)" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:39 | ||||||
| #: lib/cannery_web/live/range_live/index.ex:34 | #: lib/cannery_web/live/range_live/index.ex:34 | ||||||
| msgid "Edit Shot Records" | msgid "Edit Shot Records" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -613,8 +592,8 @@ msgid "Rounds left" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:119 | #: lib/cannery_web/live/ammo_group_live/show.ex:89 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:64 | #: lib/cannery_web/live/range_live/index.ex:83 | ||||||
| msgid "Rounds shot" | msgid "Rounds shot" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -624,18 +603,18 @@ msgid "Shot Records" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:30 | #: lib/cannery_web/live/ammo_group_live/index.ex:32 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:53 | #: lib/cannery_web/live/ammo_group_live/show.ex:40 | ||||||
| msgid "Move Ammo group" | msgid "Move Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:3 | #: lib/cannery_web/components/move_ammo_group_component.ex:80 | ||||||
| msgid "Move ammo" | msgid "Move ammo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:8 | #: lib/cannery_web/components/move_ammo_group_component.ex:85 | ||||||
| msgid "No other containers" | msgid "No other containers" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -646,7 +625,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/ammo_group_card.ex:43 | #: lib/cannery_web/components/ammo_group_card.ex:43 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:64 | #: lib/cannery_web/live/ammo_group_live/index.ex:117 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:32 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:32 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:39 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:39 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:98 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:98 | ||||||
| @@ -654,41 +633,41 @@ msgid "$%{amount}" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:83 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:75 | ||||||
| msgid "Bimetal" | msgid "Bimetal" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:78 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:72 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:57 | #: lib/cannery_web/live/ammo_type_live/index.ex:57 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:45 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:45 | ||||||
| msgid "Jacket type" | msgid "Jacket type" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:87 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:79 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:58 | #: lib/cannery_web/live/ammo_type_live/index.ex:58 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:46 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:46 | ||||||
| msgid "Muzzle velocity" | msgid "Muzzle velocity" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:103 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:93 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:60 | #: lib/cannery_web/live/ammo_type_live/index.ex:61 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:48 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:48 | ||||||
| msgid "Powder grains per charge" | msgid "Powder grains per charge" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:97 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:89 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:59 | #: lib/cannery_web/live/ammo_type_live/index.ex:59 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:47 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:47 | ||||||
| msgid "Powder type" | msgid "Powder type" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:168 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:152 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:70 | #: lib/cannery_web/live/ammo_type_live/index.ex:74 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:58 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:58 | ||||||
| msgid "UPC" | msgid "UPC" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -710,18 +689,18 @@ msgid "New password" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:82 | #: lib/cannery_web/live/ammo_group_live/index.ex:131 | ||||||
| msgid "Stage" | msgid "Stage" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:82 | #: lib/cannery_web/live/ammo_group_live/index.ex:131 | ||||||
| msgid "Unstage" | msgid "Unstage" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:137 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:125 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:64 | #: lib/cannery_web/live/ammo_type_live/index.ex:68 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:52 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:52 | ||||||
| msgid "Firing type" | msgid "Firing type" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -764,7 +743,7 @@ msgid "No cost information" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:36 | #: lib/cannery_web/live/ammo_group_live/index.ex:83 | ||||||
| msgid "% left" | msgid "% left" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -799,7 +778,7 @@ msgid "Current # of rounds:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.html.heex:32 | #: lib/cannery_web/live/ammo_type_live/index.ex:86 | ||||||
| msgid "Total # of rounds" | msgid "Total # of rounds" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -807,3 +786,44 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:85 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:85 | ||||||
| msgid "Total rounds shot:" | msgid "Total rounds shot:" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_confirmation_controller.ex:8 | ||||||
|  | msgid "Confirm your account" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:9 | ||||||
|  | msgid "Forgot your password?" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_session_controller.ex:8 | ||||||
|  | msgid "Log in" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:35 | ||||||
|  | msgid "Register" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:36 | ||||||
|  | msgid "Reset your password" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:38 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:28 | ||||||
|  | msgid "Record Shots" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:58 | ||||||
|  | msgid "Copies" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:34 | ||||||
|  | msgid "Ammo types" | ||||||
|  | msgstr "" | ||||||
|   | |||||||
| @@ -12,12 +12,12 @@ msgstr "" | |||||||
| "Plural-Forms: nplurals=2\n" | "Plural-Forms: nplurals=2\n" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:42 | #: lib/cannery_web/live/ammo_group_live/index.ex:44 | ||||||
| msgid "Add Ammo" | msgid "Add Ammo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:12 | #: lib/cannery_web/live/ammo_group_live/index.html.heex:24 | ||||||
| msgid "Add your first box!" | msgid "Add your first box!" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -82,7 +82,7 @@ msgid "Make your first tag!" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:17 | #: lib/cannery_web/live/ammo_group_live/index.html.heex:42 | ||||||
| msgid "New Ammo group" | msgid "New Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -125,9 +125,9 @@ msgid "Reset password" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:42 | #: lib/cannery_web/components/add_shot_group_component.html.heex:46 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:54 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:73 | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:172 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:156 | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:50 | #: lib/cannery_web/live/container_live/form_component.html.heex:50 | ||||||
| #: lib/cannery_web/live/invite_live/form_component.html.heex:28 | #: lib/cannery_web/live/invite_live/form_component.html.heex:28 | ||||||
| #: lib/cannery_web/live/range_live/form_component.html.heex:40 | #: lib/cannery_web/live/range_live/form_component.html.heex:40 | ||||||
| @@ -161,7 +161,7 @@ msgid "Why not get some ready to shoot?" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:85 | #: lib/cannery_web/live/ammo_group_live/index.ex:133 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:86 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:86 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:36 | #: lib/cannery_web/live/range_live/index.html.heex:36 | ||||||
| msgid "Record shots" | msgid "Record shots" | ||||||
| @@ -173,7 +173,7 @@ msgid "Ammo Details" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:12 | #: lib/cannery_web/components/move_ammo_group_component.ex:89 | ||||||
| msgid "Add another container!" | msgid "Add another container!" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -183,7 +183,7 @@ msgid "Move containers" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:60 | #: lib/cannery_web/components/move_ammo_group_component.ex:127 | ||||||
| msgid "Select" | msgid "Select" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -191,3 +191,14 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/invite_live/index.html.heex:33 | #: lib/cannery_web/live/invite_live/index.html.heex:33 | ||||||
| msgid "Copy to clipboard" | msgid "Copy to clipboard" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:18 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:36 | ||||||
|  | msgid "add a container first" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format, fuzzy | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:66 | ||||||
|  | msgid "Create" | ||||||
|  | msgstr "" | ||||||
|   | |||||||
| @@ -34,13 +34,13 @@ msgstr "" | |||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:52 | #: lib/cannery_web/components/topbar.ex:52 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:3 | #: lib/cannery_web/live/ammo_group_live/index.html.heex:3 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:61 | #: lib/cannery_web/live/range_live/index.ex:82 | ||||||
| msgid "Ammo" | msgid "Ammo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:21 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:21 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:27 | #: lib/cannery_web/live/ammo_group_live/index.ex:80 | ||||||
| msgid "Ammo type" | msgid "Ammo type" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -55,19 +55,19 @@ msgid "Background color" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:154 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:140 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:67 | #: lib/cannery_web/live/ammo_type_live/index.ex:71 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:55 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:55 | ||||||
| msgid "Blank" | msgid "Blank" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:74 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:68 | ||||||
| msgid "Brass" | msgid "Brass" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:46 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:44 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:53 | #: lib/cannery_web/live/ammo_type_live/index.ex:53 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:41 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:41 | ||||||
| msgid "Bullet core" | msgid "Bullet core" | ||||||
| @@ -81,49 +81,50 @@ msgid "Bullet type" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:62 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:58 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:55 | #: lib/cannery_web/live/ammo_type_live/index.ex:55 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:43 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:43 | ||||||
| msgid "Caliber" | msgid "Caliber" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:55 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:51 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:54 | #: lib/cannery_web/live/ammo_type_live/index.ex:54 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:42 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:42 | ||||||
| msgid "Cartridge" | msgid "Cartridge" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:69 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:65 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:56 | #: lib/cannery_web/live/ammo_type_live/index.ex:56 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:44 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:44 | ||||||
| msgid "Case material" | msgid "Case material" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:22 | #: lib/cannery_web/components/move_ammo_group_component.ex:67 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:48 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:48 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:42 | #: lib/cannery_web/live/ammo_group_live/index.ex:85 | ||||||
| msgid "Container" | msgid "Container" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:46 | #: lib/cannery_web/components/topbar.ex:46 | ||||||
|  | #: lib/cannery_web/live/container_live/index.ex:38 | ||||||
| #: lib/cannery_web/live/container_live/index.html.heex:3 | #: lib/cannery_web/live/container_live/index.html.heex:3 | ||||||
| msgid "Containers" | msgid "Containers" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:158 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:144 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:68 | #: lib/cannery_web/live/ammo_type_live/index.ex:72 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:56 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:56 | ||||||
| msgid "Corrosive" | msgid "Corrosive" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:27 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:27 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:30 | #: lib/cannery_web/live/ammo_group_live/index.ex:81 | ||||||
| msgid "Count" | msgid "Count" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -156,8 +157,8 @@ msgid "Easy to Use:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:36 | #: lib/cannery_web/live/ammo_group_live/index.ex:38 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:55 | #: lib/cannery_web/live/ammo_group_live/show.ex:42 | ||||||
| msgid "Edit Ammo group" | msgid "Edit Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -188,20 +189,20 @@ msgid "Example bullet type abbreviations" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:42 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:40 | ||||||
| msgid "FMJ" | msgid "FMJ" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:113 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:103 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:61 | #: lib/cannery_web/live/ammo_type_live/index.ex:65 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:49 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:49 | ||||||
| msgid "Grains" | msgid "Grains" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:150 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:136 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:66 | #: lib/cannery_web/live/ammo_type_live/index.ex:70 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:54 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:54 | ||||||
| msgid "Incendiary" | msgid "Incendiary" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -223,6 +224,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:71 | #: lib/cannery_web/components/topbar.ex:71 | ||||||
|  | #: lib/cannery_web/live/invite_live/index.ex:43 | ||||||
| #: lib/cannery_web/live/invite_live/index.html.heex:3 | #: lib/cannery_web/live/invite_live/index.html.heex:3 | ||||||
| msgid "Invites" | msgid "Invites" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -233,27 +235,7 @@ msgid "Keep me logged in for 60 days" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:34 | #: lib/cannery_web/components/move_ammo_group_component.ex:69 | ||||||
| msgid "Listing Ammo types" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/container_live/index.ex:38 |  | ||||||
| msgid "Listing Containers" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/invite_live/index.ex:43 |  | ||||||
| msgid "Listing Invites" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/tag_live/index.ex:34 |  | ||||||
| msgid "Listing Tags" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:30 |  | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:42 | #: lib/cannery_web/live/container_live/form_component.html.heex:42 | ||||||
| msgid "Location" | msgid "Location" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -275,8 +257,8 @@ msgid "Manage" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:162 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:148 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:69 | #: lib/cannery_web/live/ammo_type_live/index.ex:73 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:57 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:57 | ||||||
| msgid "Manufacturer" | msgid "Manufacturer" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -336,7 +318,7 @@ msgid "No ammo for this type" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/container_live/show.html.heex:77 | #: lib/cannery_web/live/container_live/show.html.heex:78 | ||||||
| msgid "No ammo groups in this container" | msgid "No ammo groups in this container" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -359,9 +341,9 @@ msgstr "" | |||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:30 | #: lib/cannery_web/components/add_shot_group_component.html.heex:30 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:41 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:41 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:122 | #: lib/cannery_web/live/ammo_group_live/show.ex:90 | ||||||
| #: lib/cannery_web/live/range_live/form_component.html.heex:29 | #: lib/cannery_web/live/range_live/form_component.html.heex:29 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:67 | #: lib/cannery_web/live/range_live/index.ex:84 | ||||||
| msgid "Notes" | msgid "Notes" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -377,15 +359,15 @@ msgid "On the bookshelf" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:121 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:111 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:62 | #: lib/cannery_web/live/ammo_type_live/index.ex:66 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:50 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:50 | ||||||
| msgid "Pressure" | msgid "Pressure" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:34 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:34 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:33 | #: lib/cannery_web/live/ammo_group_live/index.ex:82 | ||||||
| msgid "Price paid" | msgid "Price paid" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -395,8 +377,8 @@ msgid "Price paid:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:128 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:118 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:63 | #: lib/cannery_web/live/ammo_type_live/index.ex:67 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:51 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:51 | ||||||
| msgid "Primer type" | msgid "Primer type" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -422,12 +404,13 @@ msgid "Set Unlimited" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_settings_controller.ex:10 | ||||||
| #: lib/cannery_web/templates/user_settings/edit.html.heex:3 | #: lib/cannery_web/templates/user_settings/edit.html.heex:3 | ||||||
| msgid "Settings" | msgid "Settings" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:54 | #: lib/cannery_web/live/ammo_group_live/show.ex:41 | ||||||
| msgid "Show Ammo group" | msgid "Show Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -442,7 +425,7 @@ msgid "Simple:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:51 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:47 | ||||||
| msgid "Steel" | msgid "Steel" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -453,6 +436,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:40 | #: lib/cannery_web/components/topbar.ex:40 | ||||||
|  | #: lib/cannery_web/live/tag_live/index.ex:34 | ||||||
| #: lib/cannery_web/live/tag_live/index.html.heex:3 | #: lib/cannery_web/live/tag_live/index.html.heex:3 | ||||||
| msgid "Tags" | msgid "Tags" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -478,14 +462,14 @@ msgid "This ammo group is not in a container" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:146 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:132 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:65 | #: lib/cannery_web/live/ammo_type_live/index.ex:69 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:53 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:53 | ||||||
| msgid "Tracer" | msgid "Tracer" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:26 | #: lib/cannery_web/components/move_ammo_group_component.ex:68 | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:35 | #: lib/cannery_web/live/container_live/form_component.html.heex:35 | ||||||
| msgid "Type" | msgid "Type" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -528,7 +512,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/topbar.ex:64 | #: lib/cannery_web/components/topbar.ex:64 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:39 | #: lib/cannery_web/live/ammo_group_live/index.ex:84 | ||||||
| msgid "Range" | msgid "Range" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -538,8 +522,8 @@ msgid "Range day" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:125 | #: lib/cannery_web/live/ammo_group_live/show.ex:91 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:70 | #: lib/cannery_web/live/range_live/index.ex:85 | ||||||
| msgid "Date" | msgid "Date" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -565,15 +549,9 @@ msgstr "" | |||||||
| msgid "Unstage from range" | msgid "Unstage from range" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:52 |  | ||||||
| msgid "Add Shot group" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:3 | #: lib/cannery_web/components/add_shot_group_component.html.heex:3 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:24 | #: lib/cannery_web/live/ammo_group_live/index.ex:26 | ||||||
| #: lib/cannery_web/live/range_live/index.ex:28 |  | ||||||
| msgid "Record shots" | msgid "Record shots" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -583,7 +561,7 @@ msgid "Ammo Types" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:47 | #: lib/cannery_web/live/ammo_group_live/index.ex:49 | ||||||
| msgid "Ammo groups" | msgid "Ammo groups" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -594,6 +572,7 @@ msgid "Date (UTC)" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:39 | ||||||
| #: lib/cannery_web/live/range_live/index.ex:34 | #: lib/cannery_web/live/range_live/index.ex:34 | ||||||
| msgid "Edit Shot Records" | msgid "Edit Shot Records" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -614,8 +593,8 @@ msgid "Rounds left" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:119 | #: lib/cannery_web/live/ammo_group_live/show.ex:89 | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:64 | #: lib/cannery_web/live/range_live/index.ex:83 | ||||||
| msgid "Rounds shot" | msgid "Rounds shot" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -625,18 +604,18 @@ msgid "Shot Records" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:30 | #: lib/cannery_web/live/ammo_group_live/index.ex:32 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:53 | #: lib/cannery_web/live/ammo_group_live/show.ex:40 | ||||||
| msgid "Move Ammo group" | msgid "Move Ammo group" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:3 | #: lib/cannery_web/components/move_ammo_group_component.ex:80 | ||||||
| msgid "Move ammo" | msgid "Move ammo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.html.heex:8 | #: lib/cannery_web/components/move_ammo_group_component.ex:85 | ||||||
| msgid "No other containers" | msgid "No other containers" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -647,7 +626,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/ammo_group_card.ex:43 | #: lib/cannery_web/components/ammo_group_card.ex:43 | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:64 | #: lib/cannery_web/live/ammo_group_live/index.ex:117 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:32 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:32 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:39 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:39 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:98 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:98 | ||||||
| @@ -655,41 +634,41 @@ msgid "$%{amount}" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:83 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:75 | ||||||
| msgid "Bimetal" | msgid "Bimetal" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:78 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:72 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:57 | #: lib/cannery_web/live/ammo_type_live/index.ex:57 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:45 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:45 | ||||||
| msgid "Jacket type" | msgid "Jacket type" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:87 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:79 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:58 | #: lib/cannery_web/live/ammo_type_live/index.ex:58 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:46 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:46 | ||||||
| msgid "Muzzle velocity" | msgid "Muzzle velocity" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:103 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:93 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:60 | #: lib/cannery_web/live/ammo_type_live/index.ex:61 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:48 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:48 | ||||||
| msgid "Powder grains per charge" | msgid "Powder grains per charge" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:97 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:89 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:59 | #: lib/cannery_web/live/ammo_type_live/index.ex:59 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:47 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:47 | ||||||
| msgid "Powder type" | msgid "Powder type" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:168 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:152 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:70 | #: lib/cannery_web/live/ammo_type_live/index.ex:74 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:58 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:58 | ||||||
| msgid "UPC" | msgid "UPC" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -711,18 +690,18 @@ msgid "New password" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:82 | #: lib/cannery_web/live/ammo_group_live/index.ex:130 | ||||||
| msgid "Stage" | msgid "Stage" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:82 | #: lib/cannery_web/live/ammo_group_live/index.ex:130 | ||||||
| msgid "Unstage" | msgid "Unstage" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:137 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:125 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.ex:64 | #: lib/cannery_web/live/ammo_type_live/index.ex:68 | ||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:52 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:52 | ||||||
| msgid "Firing type" | msgid "Firing type" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -765,7 +744,7 @@ msgid "No cost information" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:36 | #: lib/cannery_web/live/ammo_group_live/index.ex:83 | ||||||
| msgid "% left" | msgid "% left" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -800,7 +779,7 @@ msgid "Current # of rounds:" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.html.heex:32 | #: lib/cannery_web/live/ammo_type_live/index.ex:86 | ||||||
| msgid "Total # of rounds" | msgid "Total # of rounds" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -808,3 +787,44 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/ammo_type_live/show.html.heex:85 | #: lib/cannery_web/live/ammo_type_live/show.html.heex:85 | ||||||
| msgid "Total rounds shot:" | msgid "Total rounds shot:" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_confirmation_controller.ex:8 | ||||||
|  | msgid "Confirm your account" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:9 | ||||||
|  | msgid "Forgot your password?" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_session_controller.ex:8 | ||||||
|  | msgid "Log in" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_registration_controller.ex:35 | ||||||
|  | msgid "Register" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/controllers/user_reset_password_controller.ex:36 | ||||||
|  | msgid "Reset your password" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format, fuzzy | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:38 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:28 | ||||||
|  | msgid "Record Shots" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:58 | ||||||
|  | msgid "Copies" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format, fuzzy | ||||||
|  | #: lib/cannery_web/live/ammo_type_live/index.ex:34 | ||||||
|  | msgid "Ammo types" | ||||||
|  | msgstr "" | ||||||
|   | |||||||
| @@ -65,19 +65,19 @@ msgid "Oops, something went wrong! Please check the errors below." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_reset_password_controller.ex:60 | #: lib/cannery_web/controllers/user_reset_password_controller.ex:63 | ||||||
| msgid "Reset password link is invalid or it has expired." | msgid "Reset password link is invalid or it has expired." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:25 | #: lib/cannery_web/controllers/user_registration_controller.ex:25 | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:53 | #: lib/cannery_web/controllers/user_registration_controller.ex:56 | ||||||
| msgid "Sorry, public registration is disabled" | msgid "Sorry, public registration is disabled" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:15 | #: lib/cannery_web/controllers/user_registration_controller.ex:15 | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:43 | #: lib/cannery_web/controllers/user_registration_controller.ex:46 | ||||||
| msgid "Sorry, this invite was not found or expired" | msgid "Sorry, this invite was not found or expired" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -92,7 +92,7 @@ msgid "Unauthorized" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_confirmation_controller.ex:53 | #: lib/cannery_web/controllers/user_confirmation_controller.ex:54 | ||||||
| msgid "User confirmation link is invalid or it has expired." | msgid "User confirmation link is invalid or it has expired." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -158,3 +158,13 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/container_live/edit_tags_component.ex:52 | #: lib/cannery_web/live/container_live/edit_tags_component.ex:52 | ||||||
| msgid "Tag could not be removed" | msgid "Tag could not be removed" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:113 | ||||||
|  | msgid "Could not parse number of copies" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:98 | ||||||
|  | msgid "Invalid number of copies, must be between 1 and %{max}. Was %{multiplier}" | ||||||
|  | msgstr "" | ||||||
|   | |||||||
| @@ -63,18 +63,13 @@ msgid "A link to confirm your email change has been sent to the new address." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.ex:87 | #: lib/cannery_web/live/ammo_group_live/index.ex:56 | ||||||
| msgid "Ammo group created successfully" | #: lib/cannery_web/live/ammo_group_live/show.ex:52 | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:54 |  | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:34 |  | ||||||
| msgid "Ammo group deleted succesfully" | msgid "Ammo group deleted succesfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.ex:69 | #: lib/cannery_web/live/ammo_group_live/form_component.ex:75 | ||||||
| msgid "Ammo group updated successfully" | msgid "Ammo group updated successfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -98,9 +93,9 @@ msgid "Are you sure you want to delete the invite for %{name}?" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:120 | #: lib/cannery_web/live/ammo_group_live/index.ex:165 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:66 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:66 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.html.heex:75 | #: lib/cannery_web/live/ammo_type_live/index.ex:130 | ||||||
| msgid "Are you sure you want to delete this ammo?" | msgid "Are you sure you want to delete this ammo?" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -125,7 +120,7 @@ msgid "Email changed successfully." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_confirmation_controller.ex:22 | #: lib/cannery_web/controllers/user_confirmation_controller.ex:23 | ||||||
| msgid "If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly." | msgid "If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -140,7 +135,7 @@ msgid "Logged out successfully." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_reset_password_controller.ex:43 | #: lib/cannery_web/controllers/user_reset_password_controller.ex:46 | ||||||
| msgid "Password reset successfully." | msgid "Password reset successfully." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -150,7 +145,7 @@ msgid "Password updated successfully." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:71 | #: lib/cannery_web/controllers/user_registration_controller.ex:74 | ||||||
| msgid "Please check your email to verify your account" | msgid "Please check your email to verify your account" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -160,9 +155,9 @@ msgid "Register to setup %{name}" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:44 | #: lib/cannery_web/components/add_shot_group_component.html.heex:48 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:55 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:74 | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:173 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:157 | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:52 | #: lib/cannery_web/live/container_live/form_component.html.heex:52 | ||||||
| #: lib/cannery_web/live/invite_live/form_component.html.heex:30 | #: lib/cannery_web/live/invite_live/form_component.html.heex:30 | ||||||
| #: lib/cannery_web/live/range_live/form_component.html.heex:42 | #: lib/cannery_web/live/range_live/form_component.html.heex:42 | ||||||
| @@ -211,11 +206,13 @@ msgid "Ammo group unstaged succesfully" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:108 | #: lib/cannery_web/live/ammo_group_live/show.ex:132 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:130 | ||||||
| msgid "Are you sure you want to delete this shot record?" | msgid "Are you sure you want to delete this shot record?" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:80 | ||||||
| #: lib/cannery_web/live/range_live/index.ex:56 | #: lib/cannery_web/live/range_live/index.ex:56 | ||||||
| msgid "Shot records deleted succesfully" | msgid "Shot records deleted succesfully" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -226,12 +223,12 @@ msgid "Shot records updated successfully" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_confirmation_controller.ex:37 | #: lib/cannery_web/controllers/user_confirmation_controller.ex:38 | ||||||
| msgid "%{email} confirmed successfully." | msgid "%{email} confirmed successfully." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.ex:47 | #: lib/cannery_web/components/move_ammo_group_component.ex:53 | ||||||
| msgid "Ammo moved to %{name} successfully" | msgid "Ammo moved to %{name} successfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -244,3 +241,21 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/container_live/edit_tags_component.ex:58 | #: lib/cannery_web/live/container_live/edit_tags_component.ex:58 | ||||||
| msgid "%{name} removed successfully" | msgid "%{name} removed successfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:15 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:33 | ||||||
|  | msgid "You'll need to" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format, fuzzy | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:67 | ||||||
|  | msgid "Creating..." | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format, fuzzy | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:134 | ||||||
|  | msgid "Ammo group created successfully" | ||||||
|  | msgid_plural "Ammo groups created successfully" | ||||||
|  | msgstr[0] "" | ||||||
|  | msgstr[1] "" | ||||||
|   | |||||||
| @@ -65,19 +65,19 @@ msgid "Oops, something went wrong! Please check the errors below." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_reset_password_controller.ex:60 | #: lib/cannery_web/controllers/user_reset_password_controller.ex:63 | ||||||
| msgid "Reset password link is invalid or it has expired." | msgid "Reset password link is invalid or it has expired." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:25 | #: lib/cannery_web/controllers/user_registration_controller.ex:25 | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:53 | #: lib/cannery_web/controllers/user_registration_controller.ex:56 | ||||||
| msgid "Sorry, public registration is disabled" | msgid "Sorry, public registration is disabled" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:15 | #: lib/cannery_web/controllers/user_registration_controller.ex:15 | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:43 | #: lib/cannery_web/controllers/user_registration_controller.ex:46 | ||||||
| msgid "Sorry, this invite was not found or expired" | msgid "Sorry, this invite was not found or expired" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -92,7 +92,7 @@ msgid "Unauthorized" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_confirmation_controller.ex:53 | #: lib/cannery_web/controllers/user_confirmation_controller.ex:54 | ||||||
| msgid "User confirmation link is invalid or it has expired." | msgid "User confirmation link is invalid or it has expired." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -157,3 +157,13 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/container_live/edit_tags_component.ex:52 | #: lib/cannery_web/live/container_live/edit_tags_component.ex:52 | ||||||
| msgid "Tag could not be removed" | msgid "Tag could not be removed" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:126 | ||||||
|  | msgid "Could not parse number of copies" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:111 | ||||||
|  | msgid "Invalid number of copies, must be between 1 and %{max}. Was %{multiplier}" | ||||||
|  | msgstr "" | ||||||
|   | |||||||
| @@ -62,18 +62,13 @@ msgid "A link to confirm your email change has been sent to the new address." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.ex:87 | #: lib/cannery_web/live/ammo_group_live/index.ex:56 | ||||||
| msgid "Ammo group created successfully" | #: lib/cannery_web/live/ammo_group_live/show.ex:52 | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format |  | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.ex:54 |  | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.ex:34 |  | ||||||
| msgid "Ammo group deleted succesfully" | msgid "Ammo group deleted succesfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.ex:69 | #: lib/cannery_web/live/ammo_group_live/form_component.ex:88 | ||||||
| msgid "Ammo group updated successfully" | msgid "Ammo group updated successfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -97,9 +92,9 @@ msgid "Are you sure you want to delete the invite for %{name}?" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/ammo_group_live/index.html.heex:120 | #: lib/cannery_web/live/ammo_group_live/index.ex:167 | ||||||
| #: lib/cannery_web/live/ammo_group_live/show.html.heex:66 | #: lib/cannery_web/live/ammo_group_live/show.html.heex:66 | ||||||
| #: lib/cannery_web/live/ammo_type_live/index.html.heex:75 | #: lib/cannery_web/live/ammo_type_live/index.ex:130 | ||||||
| msgid "Are you sure you want to delete this ammo?" | msgid "Are you sure you want to delete this ammo?" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -124,7 +119,7 @@ msgid "Email changed successfully." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_confirmation_controller.ex:22 | #: lib/cannery_web/controllers/user_confirmation_controller.ex:23 | ||||||
| msgid "If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly." | msgid "If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -139,7 +134,7 @@ msgid "Logged out successfully." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_reset_password_controller.ex:43 | #: lib/cannery_web/controllers/user_reset_password_controller.ex:46 | ||||||
| msgid "Password reset successfully." | msgid "Password reset successfully." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -149,7 +144,7 @@ msgid "Password updated successfully." | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_registration_controller.ex:71 | #: lib/cannery_web/controllers/user_registration_controller.ex:74 | ||||||
| msgid "Please check your email to verify your account" | msgid "Please check your email to verify your account" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -159,9 +154,9 @@ msgid "Register to setup %{name}" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/add_shot_group_component.html.heex:44 | #: lib/cannery_web/components/add_shot_group_component.html.heex:48 | ||||||
| #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:55 | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:74 | ||||||
| #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:173 | #: lib/cannery_web/live/ammo_type_live/form_component.html.heex:157 | ||||||
| #: lib/cannery_web/live/container_live/form_component.html.heex:52 | #: lib/cannery_web/live/container_live/form_component.html.heex:52 | ||||||
| #: lib/cannery_web/live/invite_live/form_component.html.heex:30 | #: lib/cannery_web/live/invite_live/form_component.html.heex:30 | ||||||
| #: lib/cannery_web/live/range_live/form_component.html.heex:42 | #: lib/cannery_web/live/range_live/form_component.html.heex:42 | ||||||
| @@ -210,11 +205,13 @@ msgid "Ammo group unstaged succesfully" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/live/range_live/index.html.heex:108 | #: lib/cannery_web/live/ammo_group_live/show.ex:132 | ||||||
|  | #: lib/cannery_web/live/range_live/index.ex:130 | ||||||
| msgid "Are you sure you want to delete this shot record?" | msgid "Are you sure you want to delete this shot record?" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/show.ex:80 | ||||||
| #: lib/cannery_web/live/range_live/index.ex:56 | #: lib/cannery_web/live/range_live/index.ex:56 | ||||||
| msgid "Shot records deleted succesfully" | msgid "Shot records deleted succesfully" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -225,12 +222,12 @@ msgid "Shot records updated successfully" | |||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/controllers/user_confirmation_controller.ex:37 | #: lib/cannery_web/controllers/user_confirmation_controller.ex:38 | ||||||
| msgid "%{email} confirmed successfully." | msgid "%{email} confirmed successfully." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #, elixir-autogen, elixir-format | #, elixir-autogen, elixir-format | ||||||
| #: lib/cannery_web/components/move_ammo_group_component.ex:47 | #: lib/cannery_web/components/move_ammo_group_component.ex:53 | ||||||
| msgid "Ammo moved to %{name} successfully" | msgid "Ammo moved to %{name} successfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -243,3 +240,21 @@ msgstr "" | |||||||
| #: lib/cannery_web/live/container_live/edit_tags_component.ex:58 | #: lib/cannery_web/live/container_live/edit_tags_component.ex:58 | ||||||
| msgid "%{name} removed successfully" | msgid "%{name} removed successfully" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:15 | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/index.html.heex:33 | ||||||
|  | msgid "You'll need to" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.html.heex:67 | ||||||
|  | msgid "Creating..." | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #, elixir-autogen, elixir-format | ||||||
|  | #: lib/cannery_web/live/ammo_group_live/form_component.ex:147 | ||||||
|  | msgid "Ammo group created successfully" | ||||||
|  | msgid_plural "Ammo groups created successfully" | ||||||
|  | msgstr[0] "" | ||||||
|  | msgstr[1] "" | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								priv/i18n/de.tbx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								priv/i18n/de.tbx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | <?xml version="1.0"?> | ||||||
|  | <!DOCTYPE martif PUBLIC "ISO 12200:1999A//DTD MARTIF core (DXFcdV04)//EN" "TBXcdv04.dtd"> | ||||||
|  | <martif type="TBX"> | ||||||
|  | <martifHeader> | ||||||
|  | <fileDesc> | ||||||
|  | <sourceDesc><p>Translate Toolkit</p></sourceDesc> | ||||||
|  | </fileDesc> | ||||||
|  | </martifHeader> | ||||||
|  | <text><body></body></text> | ||||||
|  | </martif> | ||||||
| @@ -304,9 +304,9 @@ defmodule Cannery.AccountsTest do | |||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "deletes all tokens for the given user", %{user: user} do |     test "deletes all tokens for the given user", %{user: user} do | ||||||
|       _ = Accounts.generate_user_session_token(user) |       _token = Accounts.generate_user_session_token(user) | ||||||
|  |  | ||||||
|       {:ok, _} = |       {:ok, _user} = | ||||||
|         Accounts.update_user_password(user, valid_user_password(), %{ |         Accounts.update_user_password(user, valid_user_password(), %{ | ||||||
|           "password" => "new valid password" |           "password" => "new valid password" | ||||||
|         }) |         }) | ||||||
| @@ -501,8 +501,8 @@ defmodule Cannery.AccountsTest do | |||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "deletes all tokens for the given user", %{user: user} do |     test "deletes all tokens for the given user", %{user: user} do | ||||||
|       _ = Accounts.generate_user_session_token(user) |       _token = Accounts.generate_user_session_token(user) | ||||||
|       {:ok, _} = Accounts.reset_user_password(user, %{"password" => "new valid password"}) |       {:ok, _user} = Accounts.reset_user_password(user, %{"password" => "new valid password"}) | ||||||
|       refute Repo.get_by(UserToken, user_id: user.id) |       refute Repo.get_by(UserToken, user_id: user.id) | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
|   | |||||||
| @@ -20,8 +20,8 @@ defmodule Cannery.ActivityLogTest do | |||||||
|       container = container_fixture(current_user) |       container = container_fixture(current_user) | ||||||
|       ammo_type = ammo_type_fixture(current_user) |       ammo_type = ammo_type_fixture(current_user) | ||||||
|  |  | ||||||
|       %{id: ammo_group_id} = |       {1, [%{id: ammo_group_id} = ammo_group]} = | ||||||
|         ammo_group = ammo_group_fixture(%{"count" => 25}, ammo_type, container, current_user) |         ammo_group_fixture(%{"count" => 25}, ammo_type, container, current_user) | ||||||
|  |  | ||||||
|       shot_group = |       shot_group = | ||||||
|         %{"count" => 5, "date" => ~N[2022-02-13 03:17:00], "notes" => "some notes"} |         %{"count" => 5, "date" => ~N[2022-02-13 03:17:00], "notes" => "some notes"} | ||||||
|   | |||||||
| @@ -108,7 +108,7 @@ defmodule Cannery.AmmoTest do | |||||||
|       current_user = user_fixture() |       current_user = user_fixture() | ||||||
|       ammo_type = ammo_type_fixture(current_user) |       ammo_type = ammo_type_fixture(current_user) | ||||||
|       container = container_fixture(current_user) |       container = container_fixture(current_user) | ||||||
|       ammo_group = ammo_group_fixture(ammo_type, container, current_user) |       {1, [ammo_group]} = ammo_group_fixture(ammo_type, container, current_user) | ||||||
|  |  | ||||||
|       [ |       [ | ||||||
|         ammo_type: ammo_type, |         ammo_type: ammo_type, | ||||||
| @@ -129,28 +129,28 @@ defmodule Cannery.AmmoTest do | |||||||
|                ammo_group |> Repo.preload(:shot_groups) |                ammo_group |> Repo.preload(:shot_groups) | ||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "create_ammo_group/1 with valid data creates a ammo_group", |     test "create_ammo_groups/3 with valid data creates a ammo_group", | ||||||
|          %{ |          %{ | ||||||
|            ammo_type: ammo_type, |            ammo_type: ammo_type, | ||||||
|            container: container, |            container: container, | ||||||
|            current_user: current_user |            current_user: current_user | ||||||
|          } do |          } do | ||||||
|       assert {:ok, %AmmoGroup{} = ammo_group} = |       assert {:ok, {1, [%AmmoGroup{} = ammo_group]}} = | ||||||
|                @valid_attrs |                @valid_attrs | ||||||
|                |> Map.merge(%{"ammo_type_id" => ammo_type.id, "container_id" => container.id}) |                |> Map.merge(%{"ammo_type_id" => ammo_type.id, "container_id" => container.id}) | ||||||
|                |> Ammo.create_ammo_group(current_user) |                |> Ammo.create_ammo_groups(1, current_user) | ||||||
|  |  | ||||||
|       assert ammo_group.count == 42 |       assert ammo_group.count == 42 | ||||||
|       assert ammo_group.notes == "some notes" |       assert ammo_group.notes == "some notes" | ||||||
|       assert ammo_group.price_paid == 120.5 |       assert ammo_group.price_paid == 120.5 | ||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "create_ammo_group/1 with invalid data returns error changeset", |     test "create_ammo_groups/3 with invalid data returns error changeset", | ||||||
|          %{ammo_type: ammo_type, container: container, current_user: current_user} do |          %{ammo_type: ammo_type, container: container, current_user: current_user} do | ||||||
|       assert {:error, %Changeset{}} = |       assert {:error, %Changeset{}} = | ||||||
|                @invalid_attrs |                @invalid_attrs | ||||||
|                |> Map.merge(%{"ammo_type_id" => ammo_type.id, "container_id" => container.id}) |                |> Map.merge(%{"ammo_type_id" => ammo_type.id, "container_id" => container.id}) | ||||||
|                |> Ammo.create_ammo_group(current_user) |                |> Ammo.create_ammo_groups(1, current_user) | ||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "update_ammo_group/2 with valid data updates the ammo_group", |     test "update_ammo_group/2 with valid data updates the ammo_group", | ||||||
|   | |||||||
| @@ -116,7 +116,7 @@ defmodule CanneryWeb.UserAuthTest do | |||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "does not authenticate if data is missing", %{conn: conn, current_user: current_user} do |     test "does not authenticate if data is missing", %{conn: conn, current_user: current_user} do | ||||||
|       _ = Accounts.generate_user_session_token(current_user) |       _token = Accounts.generate_user_session_token(current_user) | ||||||
|       conn = UserAuth.fetch_current_user(conn, []) |       conn = UserAuth.fetch_current_user(conn, []) | ||||||
|       refute get_session(conn, :user_token) |       refute get_session(conn, :user_token) | ||||||
|       refute conn.assigns.current_user |       refute conn.assigns.current_user | ||||||
|   | |||||||
| @@ -5,8 +5,7 @@ defmodule CanneryWeb.UserConfirmationControllerTest do | |||||||
|  |  | ||||||
|   use CanneryWeb.ConnCase, async: true |   use CanneryWeb.ConnCase, async: true | ||||||
|   import CanneryWeb.Gettext |   import CanneryWeb.Gettext | ||||||
|   alias Cannery.Accounts |   alias Cannery.{Accounts, Repo} | ||||||
|   alias Cannery.Repo |  | ||||||
|  |  | ||||||
|   @moduletag :user_confirmation_controller_test |   @moduletag :user_confirmation_controller_test | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,18 +6,26 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|   use CanneryWeb.ConnCase |   use CanneryWeb.ConnCase | ||||||
|   import Phoenix.LiveViewTest |   import Phoenix.LiveViewTest | ||||||
|   import CanneryWeb.Gettext |   import CanneryWeb.Gettext | ||||||
|   alias Cannery.Repo |   alias Cannery.{Ammo, Repo} | ||||||
|  |  | ||||||
|   @moduletag :ammo_group_live_test |   @moduletag :ammo_group_live_test | ||||||
|   @shot_group_create_attrs %{"ammo_left" => 5, "notes" => "some notes"} |   @shot_group_create_attrs %{"ammo_left" => 5, "notes" => "some notes"} | ||||||
|   @create_attrs %{count: 42, notes: "some notes", price_paid: 120.5} |   @shot_group_update_attrs %{"count" => 5, "notes" => "some updated notes"} | ||||||
|   @update_attrs %{count: 43, notes: "some updated notes", price_paid: 456.7} |   @create_attrs %{"count" => 42, "notes" => "some notes", "price_paid" => 120.5} | ||||||
|  |   @update_attrs %{"count" => 43, "notes" => "some updated notes", "price_paid" => 456.7} | ||||||
|  |   @ammo_group_create_limit 10_000 | ||||||
|   # @invalid_attrs %{count: -1, notes: nil, price_paid: nil} |   # @invalid_attrs %{count: -1, notes: nil, price_paid: nil} | ||||||
|  |  | ||||||
|   defp create_ammo_group(%{current_user: current_user}) do |   defp create_ammo_group(%{current_user: current_user}) do | ||||||
|     ammo_type = ammo_type_fixture(current_user) |     ammo_type = ammo_type_fixture(current_user) | ||||||
|     container = container_fixture(current_user) |     container = container_fixture(current_user) | ||||||
|     %{ammo_group: ammo_group_fixture(ammo_type, container, current_user)} |     {1, [ammo_group]} = ammo_group_fixture(ammo_type, container, current_user) | ||||||
|  |  | ||||||
|  |     shot_group = | ||||||
|  |       %{"count" => 5, "date" => ~N[2022-02-13 03:17:00], "notes" => "some notes"} | ||||||
|  |       |> shot_group_fixture(current_user, ammo_group) | ||||||
|  |  | ||||||
|  |     %{ammo_group: ammo_group, shot_group: shot_group} | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   describe "Index" do |   describe "Index" do | ||||||
| @@ -31,7 +39,7 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       assert html =~ ammo_group.ammo_type.name |       assert html =~ ammo_group.ammo_type.name | ||||||
|     end |     end | ||||||
|  |  | ||||||
|     test "saves new ammo_group", %{conn: conn} do |     test "saves a single new ammo_group", %{conn: conn} do | ||||||
|       {:ok, index_live, _html} = live(conn, Routes.ammo_group_index_path(conn, :index)) |       {:ok, index_live, _html} = live(conn, Routes.ammo_group_index_path(conn, :index)) | ||||||
|  |  | ||||||
|       assert index_live |> element("a", dgettext("actions", "New Ammo group")) |> render_click() =~ |       assert index_live |> element("a", dgettext("actions", "New Ammo group")) |> render_click() =~ | ||||||
| @@ -43,7 +51,7 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) |       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#ammo_group-form", ammo_group: @create_attrs) |         |> form("#ammo_group-form", ammo_group: @create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -53,6 +61,68 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       assert html =~ "42" |       assert html =~ "42" | ||||||
|     end |     end | ||||||
|  |  | ||||||
|  |     test "saves multiple new ammo_groups", %{conn: conn, current_user: current_user} do | ||||||
|  |       multiplier = 25 | ||||||
|  |  | ||||||
|  |       {:ok, index_live, _html} = live(conn, Routes.ammo_group_index_path(conn, :index)) | ||||||
|  |  | ||||||
|  |       assert index_live |> element("a", dgettext("actions", "New Ammo group")) |> render_click() =~ | ||||||
|  |                gettext("New Ammo group") | ||||||
|  |  | ||||||
|  |       assert_patch(index_live, Routes.ammo_group_index_path(conn, :new)) | ||||||
|  |  | ||||||
|  |       # assert index_live | ||||||
|  |       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) | ||||||
|  |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|  |       {:ok, _view, html} = | ||||||
|  |         index_live | ||||||
|  |         |> form("#ammo_group-form", | ||||||
|  |           ammo_group: @create_attrs |> Map.put("multiplier", to_string(multiplier)) | ||||||
|  |         ) | ||||||
|  |         |> render_submit() | ||||||
|  |         |> follow_redirect(conn, Routes.ammo_group_index_path(conn, :index)) | ||||||
|  |  | ||||||
|  |       assert html =~ dgettext("prompts", "Ammo groups created successfully") | ||||||
|  |       assert Ammo.list_ammo_groups(current_user) |> Enum.count() == multiplier + 1 | ||||||
|  |     end | ||||||
|  |  | ||||||
|  |     test "does not save invalid number of new ammo_groups", %{conn: conn} do | ||||||
|  |       {:ok, index_live, _html} = live(conn, Routes.ammo_group_index_path(conn, :index)) | ||||||
|  |  | ||||||
|  |       assert index_live |> element("a", dgettext("actions", "New Ammo group")) |> render_click() =~ | ||||||
|  |                gettext("New Ammo group") | ||||||
|  |  | ||||||
|  |       assert_patch(index_live, Routes.ammo_group_index_path(conn, :new)) | ||||||
|  |  | ||||||
|  |       # assert index_live | ||||||
|  |       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) | ||||||
|  |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|  |       assert index_live | ||||||
|  |              |> form("#ammo_group-form", ammo_group: @create_attrs |> Map.put("multiplier", "0")) | ||||||
|  |              |> render_submit() =~ | ||||||
|  |                dgettext( | ||||||
|  |                  "errors", | ||||||
|  |                  "Invalid number of copies, must be between 1 and %{max}. Was %{multiplier}", | ||||||
|  |                  multiplier: 0, | ||||||
|  |                  max: @ammo_group_create_limit | ||||||
|  |                ) | ||||||
|  |  | ||||||
|  |       assert index_live | ||||||
|  |              |> form("#ammo_group-form", | ||||||
|  |                ammo_group: | ||||||
|  |                  @create_attrs |> Map.put("multiplier", to_string(@ammo_group_create_limit + 1)) | ||||||
|  |              ) | ||||||
|  |              |> render_submit() =~ | ||||||
|  |                dgettext( | ||||||
|  |                  "errors", | ||||||
|  |                  "Invalid number of copies, must be between 1 and %{max}. Was %{multiplier}", | ||||||
|  |                  multiplier: @ammo_group_create_limit + 1, | ||||||
|  |                  max: @ammo_group_create_limit | ||||||
|  |                ) | ||||||
|  |     end | ||||||
|  |  | ||||||
|     test "saves new shot_group", %{conn: conn, ammo_group: ammo_group} do |     test "saves new shot_group", %{conn: conn, ammo_group: ammo_group} do | ||||||
|       {:ok, index_live, _html} = live(conn, Routes.ammo_group_index_path(conn, :index)) |       {:ok, index_live, _html} = live(conn, Routes.ammo_group_index_path(conn, :index)) | ||||||
|  |  | ||||||
| @@ -65,7 +135,7 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       #        |> form("#shot_group-form", shot_group: @invalid_attrs) |       #        |> form("#shot_group-form", shot_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "is invalid") |       #        |> render_change() =~ dgettext("errors", "is invalid") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#shot-group-form", shot_group: @shot_group_create_attrs) |         |> form("#shot-group-form", shot_group: @shot_group_create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -88,7 +158,7 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) |       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#ammo_group-form", ammo_group: @update_attrs) |         |> form("#ammo_group-form", ammo_group: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -134,7 +204,7 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) |       #        |> form("#ammo_group-form", ammo_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         show_live |         show_live | ||||||
|         |> form("#ammo_group-form", ammo_group: @update_attrs) |         |> form("#ammo_group-form", ammo_group: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -156,7 +226,7 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|       #        |> form("#shot_group-form", shot_group: @invalid_attrs) |       #        |> form("#shot_group-form", shot_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "is invalid") |       #        |> render_change() =~ dgettext("errors", "is invalid") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#shot-group-form", shot_group: @shot_group_create_attrs) |         |> form("#shot-group-form", shot_group: @shot_group_create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -164,5 +234,40 @@ defmodule CanneryWeb.AmmoGroupLiveTest do | |||||||
|  |  | ||||||
|       assert html =~ dgettext("prompts", "Shots recorded successfully") |       assert html =~ dgettext("prompts", "Shots recorded successfully") | ||||||
|     end |     end | ||||||
|  |  | ||||||
|  |     test "updates shot_group in listing", | ||||||
|  |          %{conn: conn, ammo_group: ammo_group, shot_group: shot_group} do | ||||||
|  |       {:ok, index_live, _html} = live(conn, Routes.ammo_group_show_path(conn, :edit, ammo_group)) | ||||||
|  |  | ||||||
|  |       assert index_live |> element("[data-qa=\"edit-#{shot_group.id}\"]") |> render_click() =~ | ||||||
|  |                gettext("Edit Shot Records") | ||||||
|  |  | ||||||
|  |       assert_patch( | ||||||
|  |         index_live, | ||||||
|  |         Routes.ammo_group_show_path(conn, :edit_shot_group, ammo_group, shot_group) | ||||||
|  |       ) | ||||||
|  |  | ||||||
|  |       # assert index_live | ||||||
|  |       #        |> form("#shot_group-form", shot_group: @invalid_attrs) | ||||||
|  |       #        |> render_change() =~ dgettext("errors", "is invalid") | ||||||
|  |  | ||||||
|  |       {:ok, _view, html} = | ||||||
|  |         index_live | ||||||
|  |         |> form("#shot-group-form", shot_group: @shot_group_update_attrs) | ||||||
|  |         |> render_submit() | ||||||
|  |         |> follow_redirect(conn, Routes.ammo_group_show_path(conn, :show, ammo_group)) | ||||||
|  |  | ||||||
|  |       assert html =~ dgettext("actions", "Shot records updated successfully") | ||||||
|  |       assert html =~ "some updated notes" | ||||||
|  |     end | ||||||
|  |  | ||||||
|  |     test "deletes shot_group in listing", | ||||||
|  |          %{conn: conn, ammo_group: ammo_group, shot_group: shot_group} do | ||||||
|  |       {:ok, index_live, _html} = | ||||||
|  |         live(conn, Routes.ammo_group_show_path(conn, :edit_shot_group, ammo_group, shot_group)) | ||||||
|  |  | ||||||
|  |       assert index_live |> element("[data-qa=\"delete-#{shot_group.id}\"]") |> render_click() | ||||||
|  |       refute has_element?(index_live, "#shot_group-#{shot_group.id}") | ||||||
|  |     end | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| @@ -62,7 +62,7 @@ defmodule CanneryWeb.AmmoTypeLiveTest do | |||||||
|       #        |> form("#ammo_type-form", ammo_type: @invalid_attrs) |       #        |> form("#ammo_type-form", ammo_type: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#ammo_type-form", ammo_type: @create_attrs) |         |> form("#ammo_type-form", ammo_type: @create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -86,7 +86,7 @@ defmodule CanneryWeb.AmmoTypeLiveTest do | |||||||
|       #        |> form("#ammo_type-form", ammo_type: @invalid_attrs) |       #        |> form("#ammo_type-form", ammo_type: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#ammo_type-form", ammo_type: @update_attrs) |         |> form("#ammo_type-form", ammo_type: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -128,7 +128,7 @@ defmodule CanneryWeb.AmmoTypeLiveTest do | |||||||
|       #        |> form("#ammo_type-form", ammo_type: @invalid_attrs) |       #        |> form("#ammo_type-form", ammo_type: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         show_live |         show_live | ||||||
|         |> form("#ammo_type-form", ammo_type: @update_attrs) |         |> form("#ammo_type-form", ammo_type: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ defmodule CanneryWeb.ContainerLiveTest do | |||||||
|       #        |> form("#container-form", container: @invalid_attrs) |       #        |> form("#container-form", container: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#container-form", container: @create_attrs) |         |> form("#container-form", container: @create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -78,7 +78,7 @@ defmodule CanneryWeb.ContainerLiveTest do | |||||||
|       #        |> form("#container-form", container: @invalid_attrs) |       #        |> form("#container-form", container: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#container-form", container: @update_attrs) |         |> form("#container-form", container: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -123,7 +123,7 @@ defmodule CanneryWeb.ContainerLiveTest do | |||||||
|       #        |> form("#container-form", container: @invalid_attrs) |       #        |> form("#container-form", container: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         show_live |         show_live | ||||||
|         |> form("#container-form", container: @update_attrs) |         |> form("#container-form", container: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ defmodule CanneryWeb.InviteLiveTest do | |||||||
|       #        |> form("#invite-form", invite: @invalid_attrs) |       #        |> form("#invite-form", invite: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#invite-form", invite: @create_attrs) |         |> form("#invite-form", invite: @create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -63,7 +63,7 @@ defmodule CanneryWeb.InviteLiveTest do | |||||||
|       #        |> form("#invite-form", invite: @invalid_attrs) |       #        |> form("#invite-form", invite: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#invite-form", invite: @update_attrs) |         |> form("#invite-form", invite: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
|   | |||||||
| @@ -16,7 +16,9 @@ defmodule CanneryWeb.RangeLiveTest do | |||||||
|   defp create_shot_group(%{current_user: current_user}) do |   defp create_shot_group(%{current_user: current_user}) do | ||||||
|     container = container_fixture(%{"staged" => true}, current_user) |     container = container_fixture(%{"staged" => true}, current_user) | ||||||
|     ammo_type = ammo_type_fixture(current_user) |     ammo_type = ammo_type_fixture(current_user) | ||||||
|     ammo_group = ammo_group_fixture(%{"staged" => true}, ammo_type, container, current_user) |  | ||||||
|  |     {1, [ammo_group]} = | ||||||
|  |       ammo_group_fixture(%{"staged" => true}, ammo_type, container, current_user) | ||||||
|  |  | ||||||
|     shot_group = |     shot_group = | ||||||
|       %{"count" => 5, "date" => ~N[2022-02-13 03:17:00], "notes" => "some notes"} |       %{"count" => 5, "date" => ~N[2022-02-13 03:17:00], "notes" => "some notes"} | ||||||
| @@ -47,7 +49,7 @@ defmodule CanneryWeb.RangeLiveTest do | |||||||
|       #        |> form("#shot_group-form", shot_group: @invalid_attrs) |       #        |> form("#shot_group-form", shot_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "is invalid") |       #        |> render_change() =~ dgettext("errors", "is invalid") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#shot-group-form", shot_group: @create_attrs) |         |> form("#shot-group-form", shot_group: @create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -69,7 +71,7 @@ defmodule CanneryWeb.RangeLiveTest do | |||||||
|       #        |> form("#shot_group-form", shot_group: @invalid_attrs) |       #        |> form("#shot_group-form", shot_group: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "is invalid") |       #        |> render_change() =~ dgettext("errors", "is invalid") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#shot-group-form", shot_group: @update_attrs) |         |> form("#shot-group-form", shot_group: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
|   | |||||||
| @@ -53,7 +53,7 @@ defmodule CanneryWeb.TagLiveTest do | |||||||
|       #        |> form("#tag-form", tag: @invalid_attrs) |       #        |> form("#tag-form", tag: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#tag-form", tag: @create_attrs) |         |> form("#tag-form", tag: @create_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
| @@ -75,7 +75,7 @@ defmodule CanneryWeb.TagLiveTest do | |||||||
|       #        |> form("#tag-form", tag: @invalid_attrs) |       #        |> form("#tag-form", tag: @invalid_attrs) | ||||||
|       #        |> render_change() =~ dgettext("errors", "can't be blank") |       #        |> render_change() =~ dgettext("errors", "can't be blank") | ||||||
|  |  | ||||||
|       {:ok, _, html} = |       {:ok, _view, html} = | ||||||
|         index_live |         index_live | ||||||
|         |> form("#tag-form", tag: @update_attrs) |         |> form("#tag-form", tag: @update_attrs) | ||||||
|         |> render_submit() |         |> render_submit() | ||||||
|   | |||||||
| @@ -5,12 +5,11 @@ defmodule CanneryWeb.ErrorViewTest do | |||||||
|  |  | ||||||
|   use CanneryWeb.ConnCase, async: true |   use CanneryWeb.ConnCase, async: true | ||||||
|   import CanneryWeb.Gettext |   import CanneryWeb.Gettext | ||||||
|  |  | ||||||
|   @moduletag :error_view_test |  | ||||||
|  |  | ||||||
|   # Bring render/3 and render_to_string/3 for testing custom views |   # Bring render/3 and render_to_string/3 for testing custom views | ||||||
|   import Phoenix.View |   import Phoenix.View | ||||||
|  |  | ||||||
|  |   @moduletag :error_view_test | ||||||
|  |  | ||||||
|   test "renders 404.html" do |   test "renders 404.html" do | ||||||
|     assert render_to_string(CanneryWeb.ErrorView, "404.html", []) =~ |     assert render_to_string(CanneryWeb.ErrorView, "404.html", []) =~ | ||||||
|              dgettext("errors", "Not found") |              dgettext("errors", "Not found") | ||||||
|   | |||||||
| @@ -25,6 +25,7 @@ defmodule CanneryWeb.ConnCase do | |||||||
|       # Import conveniences for testing with connections |       # Import conveniences for testing with connections | ||||||
|       import Plug.Conn |       import Plug.Conn | ||||||
|       import Phoenix.ConnTest |       import Phoenix.ConnTest | ||||||
|  |       # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse | ||||||
|       import Cannery.Fixtures |       import Cannery.Fixtures | ||||||
|       import CanneryWeb.ConnCase |       import CanneryWeb.ConnCase | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,10 +22,8 @@ defmodule Cannery.DataCase do | |||||||
|       alias Cannery.Repo |       alias Cannery.Repo | ||||||
|  |  | ||||||
|       import Ecto |       import Ecto | ||||||
|       import Ecto.Changeset |       import Ecto.{Changeset, Query} | ||||||
|       import Ecto.Query |       import Cannery.{DataCase, Fixtures} | ||||||
|       import Cannery.DataCase |  | ||||||
|       import Cannery.Fixtures |  | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
| @@ -45,7 +43,7 @@ defmodule Cannery.DataCase do | |||||||
|   """ |   """ | ||||||
|   def errors_on(changeset) do |   def errors_on(changeset) do | ||||||
|     Ecto.Changeset.traverse_errors(changeset, fn {message, opts} -> |     Ecto.Changeset.traverse_errors(changeset, fn {message, opts} -> | ||||||
|       Regex.replace(~r"%{(\w+)}", message, fn _, key -> |       Regex.replace(~r"%{(\w+)}", message, fn _content, key -> | ||||||
|         opts |> Keyword.get(String.to_existing_atom(key), key) |> to_string() |         opts |> Keyword.get(String.to_existing_atom(key), key) |> to_string() | ||||||
|       end) |       end) | ||||||
|     end) |     end) | ||||||
|   | |||||||
| @@ -111,10 +111,20 @@ defmodule Cannery.Fixtures do | |||||||
|   @doc """ |   @doc """ | ||||||
|   Generate a AmmoGroup |   Generate a AmmoGroup | ||||||
|   """ |   """ | ||||||
|   @spec ammo_group_fixture(AmmoType.t(), Container.t(), User.t()) :: AmmoGroup.t() |   @spec ammo_group_fixture(AmmoType.t(), Container.t(), User.t()) :: | ||||||
|   @spec ammo_group_fixture(attrs :: map(), AmmoType.t(), Container.t(), User.t()) :: AmmoGroup.t() |           {count :: non_neg_integer(), [AmmoGroup.t()]} | ||||||
|  |   @spec ammo_group_fixture(attrs :: map(), AmmoType.t(), Container.t(), User.t()) :: | ||||||
|  |           {count :: non_neg_integer(), [AmmoGroup.t()]} | ||||||
|  |   @spec ammo_group_fixture( | ||||||
|  |           attrs :: map(), | ||||||
|  |           multiplier :: non_neg_integer(), | ||||||
|  |           AmmoType.t(), | ||||||
|  |           Container.t(), | ||||||
|  |           User.t() | ||||||
|  |         ) :: {count :: non_neg_integer(), [AmmoGroup.t()]} | ||||||
|   def ammo_group_fixture( |   def ammo_group_fixture( | ||||||
|         attrs \\ %{}, |         attrs \\ %{}, | ||||||
|  |         multiplier \\ 1, | ||||||
|         %AmmoType{id: ammo_type_id}, |         %AmmoType{id: ammo_type_id}, | ||||||
|         %Container{id: container_id}, |         %Container{id: container_id}, | ||||||
|         %User{} = user |         %User{} = user | ||||||
| @@ -125,7 +135,7 @@ defmodule Cannery.Fixtures do | |||||||
|       "container_id" => container_id, |       "container_id" => container_id, | ||||||
|       "count" => 20 |       "count" => 20 | ||||||
|     }) |     }) | ||||||
|     |> Ammo.create_ammo_group(user) |     |> Ammo.create_ammo_groups(multiplier, user) | ||||||
|     |> unwrap_ok_tuple() |     |> unwrap_ok_tuple() | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user