-
-
Notifications
You must be signed in to change notification settings - Fork 19
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Backported Case2. Left 5 tests failing; I believe they are very opini…
…onated.
- Loading branch information
Aleksei Matiushkin
committed
Feb 14, 2019
1 parent
d28dbc1
commit c880007
Showing
12 changed files
with
267 additions
and
134 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,127 @@ | ||
defmodule Recase.Generic do | ||
@moduledoc """ | ||
Generic module to split and join strings back. | ||
This module should not be used directly. | ||
""" | ||
|
||
@splitters Application.get_env(:recase, :delimiters, :symbol) | ||
|
||
@delimiters (case @splitters do | ||
list when is_list(list) -> | ||
list | ||
|
||
:symbol -> | ||
[all, down, up] = Enum.map([32..127, ?a..?z, ?A..?Z], &Enum.to_list/1) | ||
all -- (down ++ up) | ||
end) | ||
|
||
@doc """ | ||
Splits the input into **`list`**. Utility function. | ||
## Examples | ||
iex> Recase.Generic.split "foo_barBaz-λambdaΛambda-привет-Мир" | ||
["foo", "bar", "Baz", "λambda", "Λambda", "привет", "Мир"] | ||
""" | ||
@spec split(input :: String.t()) :: [String.t()] | ||
def split(input) when is_binary(input), do: do_split(input) | ||
|
||
@doc """ | ||
Splits the input and **`rejoins`** it with a separator given. Optionally | ||
converts parts to `downcase`, `upcase` or `titlecase`. | ||
- `opts[:case] :: [:down | :up | :title | :none]` | ||
- `opts[:separator] :: binary() | integer()` | ||
Default separator is `?_`, default conversion is `:downcase` so that | ||
it behaves the same way as `to_snake/1`. | ||
## Examples | ||
iex> Recase.Generic.rejoin "foo_barBaz-λambdaΛambda-привет-Мир", separator: "__" | ||
"foo__bar__baz__λambda__λambda__привет__мир" | ||
""" | ||
@spec rejoin(input :: String.t(), opts :: Keyword.t()) :: String.t() | ||
def rejoin(input, opts \\ []) when is_binary(input) do | ||
mapper = | ||
case Keyword.get(opts, :case, :down) do | ||
:down -> | ||
&String.downcase/1 | ||
|
||
:title -> | ||
fn <<char::utf8, rest::binary>> -> | ||
String.upcase(<<char::utf8>>) <> String.downcase(rest) | ||
end | ||
|
||
:up -> | ||
&String.upcase/1 | ||
|
||
_ -> | ||
& &1 | ||
end | ||
|
||
input | ||
|> do_split() | ||
|> Enum.map(mapper) | ||
|> Enum.join(Keyword.get(opts, :separator, ?_)) | ||
end | ||
|
||
############################################################################## | ||
|
||
@spec do_split(input :: String.t(), acc :: [String.t()]) :: [String.t()] | ||
defp do_split(string, acc \\ {"", []}) | ||
|
||
defp do_split("", {"", acc}), do: Enum.reverse(acc) | ||
|
||
defp do_split("", {curr, acc}), | ||
do: do_split("", {"", [curr | acc]}) | ||
|
||
Enum.each(@delimiters, fn delim -> | ||
defp do_split(<<unquote(delim)::utf8, rest::binary>>, {"", acc}), | ||
do: do_split(rest, {"", acc}) | ||
|
||
defp do_split(<<unquote(delim), rest::binary>>, {curr, acc}), | ||
do: do_split(rest, {"", [curr | acc]}) | ||
end) | ||
|
||
Enum.each(?A..?Z, fn char -> | ||
defp do_split(<<unquote(char), rest::binary>>, {"", acc}), | ||
do: do_split(rest, {<<unquote(char)::utf8>>, acc}) | ||
|
||
defp do_split(<<unquote(char), rest::binary>>, {curr, acc}) do | ||
<<c::utf8, _::binary>> = String.reverse(curr) | ||
|
||
if c in ?A..?Z do | ||
do_split(rest, {curr <> <<unquote(char)::utf8>>, acc}) | ||
else | ||
do_split(rest, {<<unquote(char)::utf8>>, [curr | acc]}) | ||
end | ||
end | ||
end) | ||
|
||
[32..64, 91..127] | ||
|> Enum.map(&Enum.to_list/1) | ||
|> Enum.reduce(&Kernel.++/2) | ||
|> Kernel.--(@delimiters) | ||
|> Enum.each(fn char -> | ||
defp do_split(<<unquote(char)::utf8, rest::binary>>, {"", acc}), | ||
do: do_split(rest, {<<unquote(char)::utf8>>, acc}) | ||
|
||
defp do_split(<<unquote(char), rest::binary>>, {curr, acc}), | ||
do: do_split(rest, {curr <> <<unquote(char)::utf8>>, acc}) | ||
end) | ||
|
||
defp do_split(<<char::utf8, rest::binary>>, {"", acc}), | ||
do: do_split(rest, {<<char::utf8>>, acc}) | ||
|
||
@upcase ~r/(?<!\p{Lu})\p{Lu}/u | ||
|
||
defp do_split(<<char::utf8, rest::binary>>, {curr, acc}) do | ||
if Regex.match?(@upcase, <<char::utf8>>) do | ||
do_split(rest, {<<char::utf8>>, [curr | acc]}) | ||
else | ||
do_split(rest, {curr <> <<char::utf8>>, acc}) | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.