<section style="align-items: center; display: grid; grid-gap: 1em; grid-template-columns: 1fr 3fr; margin-top: 1.83em;">
<img src="assets/logo.svg" style="margin-right: 2em;">
<div>
<h1 style="margin-top: 0;">DevJoy</h1>
<p>Easier creation of the Visual Novel-like games or presentations.</p>
<div style="display: flex; gap: 0.5em; flex-wrap: wrap;">
<a href="https://hex.pm/packages/dev_joy">
<img src="https://img.shields.io/hexpm/v/dev_joy?label=&color=midnightblue&logo=&style=for-the-badge" />
</a>
<a href="https://gitlab.com/ramurix-software/dev-joy">
<img src="https://img.shields.io/badge/source-%23333333.svg?logo=gitlab&style=for-the-badge" />
</a>
<a href="https://hexdocs.pm/dev_joy">
<img src="https://img.shields.io/badge/📖 docs-darkgreen?style=for-the-badge" />
</a>
<a href="https://www.apache.org/licenses/LICENSE-2.0">
<img src="https://img.shields.io/badge/Apache 2.0-darkmagenta?logo=apache&style=for-the-badge" />
</a>
<a href="https://elixir-lang.org">
<img src="https://img.shields.io/badge/elixir-%234B275F.svg?logo=elixir&style=for-the-badge" />
</a>
<a href="https://phoenixframework.org">
<img src="https://img.shields.io/badge/phoenix-%23f05423.svg?logo=phoenixframework&logoColor=white&style=for-the-badge" />
</a>
</div>
</div>
</section>
## Setup
Except drawing you only have to provide a content to the game and fetch it. To "draw" your game all you need to do is to convert structs into `hologram` or `phoenix` templates or `scenic` graphs and style them respectively.
<!-- tabs-open -->
### Dependency
Mix dependency:
```elixir
def deps do
[
{:dev_joy, "~> 2.0"}
]
end
```
Rebar dependency:
```erlang
{deps,[
{dev_joy, "~> 2.0"}
]}.
```
### DSL
```elixir
defmodule MyApp.Scenes.Main do
use DevJoy.Scene
part :main, page_title: "Hello!" do
dialog :john_doe, "Hello world"
end
end
```
### Getting data
```elixir
defmodule MyApp.PageLive do
use MyAppWeb, :live_view
alias MyApp.Scenes.Main
def handle_event("btn-click", _params, socket) do
Main.get_part_item(:main, 1)
{:noreply, socket}
end
end
```
### LiveView
```elixir
defmodule MyApp.PageLive do
use MyAppWeb, :live_view
use DevJoy.Session.LiveView
use DevJoy.Session.Notifier
alias DevJoy.Session
alias MyApp.Scenes.Main
# …
def handle_event("btn-click", _params, socket) do
Session.get({Main, :main, 1})
{:noreply, socket}
end
def handle_info({:dev_joy, :item_changed, item}, socket) do
{:noreply, assign(socket, item: item)}
end
end
```
### Render
```elixir
defmodule MyApp.PageLive do
use MyAppWeb, :live_view
# …
alias DevJoy.Scene.Dialog
# …
def render(%{item: %Dialog{}} = assigns) do
~H"""
{@item.content}
"""
end
end
```
### I18n
By default all template files and their locale translations
are stored in `i18n` directory inside project's priv directory.
```bash
PROJECT_PATH="/path/to/project/"
I18N_DIR="i18n"
SCENE_MODULE="MyApp.Scenes.Main"
mkdir -p "$PROJECT_PATH/priv/$I18N_DIR/fr/LC_MESSAGES"
touch "$PROJECT_PATH/priv/$I18N_DIR/fr/LC_MESSAGES/$SCENE_MODULE.po"
xdg-open "$PROJECT_PATH/priv/$I18N_DIR/fr/LC_MESSAGES/$SCENE_MODULE.po"
```
If translation is not found or is empty the msgid is used instead.
```elixir
scene = Main
default_locale = "en"
# get locale from cookie, db, headers, params or session
available_locales = scene.get_locales()
verified_locale = if locale in available_locales, do: locale, else: default_locale
DevJoy.API.Storage.set_locale(verified_locale)
assign(conn_or_socket, available_locales: available_locales, locale: locale)
# …
push_navigate(socket, to: some_path_with_locale(socket.assigns.locale))
```
### JS
See [JavaScript documentation](js/index.html) for more information.
<!-- tabs-close -->
## Example: Importing data from external sources
```elixir
Mix.install [:dev_joy, :nimble_csv]
priv =
__DIR__
|> Path.absname()
|> Path.join("script_priv")
|> tap(&File.mkdir_p!/1)
defmodule MainScene do
use DevJoy.Scene, priv: priv
alias NimbleCSV.RFC4180, as: CSV
data = CSV.parse_string("""
string id,dialog content
john_doe,Hello world!
""")
part :main do
for [name, content] <- data do
# Note: prefer String.to_existing_atom/1 instead
dialog String.to_atom(name), content
end
end
end
dialog = MainScene.get_part_item(:main, 1)
# => %DevJoy.Scene.Dialog{…}
dialog.character.id
# => :john_doe
dialog.content
# => "Hello world!"
```
## Example: Using [Item API](https://hexdocs.pm/dev_joy/DevJoy.API.Item.html) to create the DSL for code samples
<!-- tabs-open -->
### Struct
```elixir
defmodule MyApp.CodeSample do
@moduledoc """
The code sample is visually represented as a snipped containing the highlighted text
> ### Usage {: .info}
>
> The [`code sample`](`t:t/0`) always requires its [`content`](`t:content/0`) and [`language`](`t:lang/0`).
> You can also optionally specify additional [`data`](`t:data/0`).
>
> defmodule MyApp.Scenes.Main do
> use DevJoy.Scene
>
> import MyApp.DSL
>
> part :main do
> code :elixir, ~S[IO.puts("Hello world!")]
>
> code :elixir, [some: :data], ~S[IO.puts("Hello world!")]
>
> code :elixir, [some: :data] do
> ~S[IO.puts("Hello world!")]
> end
> end
> end
"""
@enforce_keys ~w[content lang]a
@doc """
The [`code sample`](`t:t/0`) structure contains the following keys:
- [`content`](`t:content/0`) - a content of the code sample
- [`data`](`t:data/0`) - a keyword list of additional code sample data
- [`lang`](`t:lang/0`) - a programming language of the code sample
"""
defstruct [:content, :lang, data: []]
@typedoc """
The type representing the [`code sample`](`__struct__/0`) structure.
"""
@typedoc section: :main
@type t :: %__MODULE__{
content: content(),
data: data(),
lang: lang()
}
@typedoc """
The type representing the content of the code sample.
"""
@typedoc section: :field
@type content :: String.t()
@typedoc """
The type representing the additional data of the code sample.
"""
@typedoc section: :field
@type data :: Keyword.t()
@typedoc """
The type representing the lang of the code sample.
"""
@typedoc section: :field
@type lang :: atom
end
```
### DSL
```elixir
defmodule MyApp.DSL do
alias MyApp.CodeSample
@doc """
A callback implementations generator for the `MyApp.CodeSample` struct.
> ### Usage {: .info}
> defmodule MyApp.Scenes.Main do
> use DevJoy.Scene
>
> import MyApp.DSL
>
> part :main do
> code :elixir, ~S[IO.puts("Hello world!")]
>
> code :elixir, [some: :data], ~S[IO.puts("Hello world!")]
>
> code :elixir, [some: :data] do
> ~S[IO.puts("Hello world!")]
> end
> end
> end
"""
defmacro code(lang, data \\ [], content)
@spec code(
Scene.macro_input(CodeSample.lang()),
Scene.macro_input(CodeSample.data()),
do: Scene.macro_input(CodeSample.content())
) :: Macro.output()
defmacro code(lang, data, do: block) do
fields = [content: block, data: data, lang: lang]
Item.generate(CodeSample, fields, __CALLER__, [:content])
end
@spec code(
Scene.macro_input(CodeSample.lang()),
Scene.macro_input(CodeSample.data()),
Scene.macro_input(CodeSample.content())
) :: Macro.output()
defmacro code(lang, data, content) do
fields = [content: content, data: data, lang: lang]
Item.generate(CodeSample, fields, __CALLER__, [:content])
end
end
```
<!-- tabs-close -->
## Example: Implementing the [Character](https://hexdocs.pm/dev_joy/DevJoy.Character.html) base module for a Discourse forum API
<!-- tabs-open -->
### Forum module
```elixir
defmodule MyApp.Forum do
@moduledoc "Fetches forum data using username and Discourse API."
@behaviour DevJoy.Character
alias DevJoy.API.Item
alias DevJoy.Character
@avatar_size "48"
@impl DevJoy.Character
def get_character_fields(id) do
{:ok, _app_list} = Application.ensure_all_started(:req)
fetch_forum_data(id)
end
@spec fetch_forum_data(Character.id()) :: Item.fields()
defp fetch_forum_data(id) do
:dev_joy
|> Application.get_env(__MODULE__, endpoint: "https://elixirforum.com")
|> Keyword.fetch!(:endpoint)
|> then(fn endpoint ->
endpoint
|> Path.join("u/#{id}.json")
|> Req.get!()
|> then(& &1.body["user"])
|> then(&[avatar: avatar_from_template(&1["avatar_template"], endpoint), full_name: &1["name"]])
end)
end
@spec avatar_from_template(String.t(), String.t()) :: Character.portrait()
defp avatar_from_template(template, endpoint) do
template
|> String.replace("{size}", @avatar_size)
|> then(&Path.join(endpoint, &1))
end
end
```
### Cache module
```elixir
defmodule MyApp.Forum.Cache do
@moduledoc "Caches character base fields using `Agent`."
alias DevJoy.API.Item
alias DevJoy.Character
@spec start_link :: Agent.on_start()
def start_link, do: Agent.start_link(fn -> %{} end, name: __MODULE__)
@spec get(Character.id()) :: Item.fields() | nil
def get(id), do: Agent.get(__MODULE__, & &1[id])
@spec put(Character.id(), Item.fields()) :: :ok
def put(id, data), do: Agent.update(&Map.put(&1, id, data))
end
```
### Forum module with ForumCache support
```elixir
defmodule MyApp.Forum do
@moduledoc "Fetches forum data using username and Discourse API."
@behaviour DevJoy.Character
alias DevJoy.API.Item
alias DevJoy.Character
alias MyApp.Forum.Cache
@avatar_size "48"
@impl DevJoy.Character
def get_character_fields(id) do
# The cache has to be started at compile-time when defined in same app
# and this function is called by scene DSL at compile-time
Process.whereis(Cache) || Cache.start_link()
id
|> Cache.get()
|> then(&(&1 || fetch_forum_data(id)))
end
@spec fetch_forum_data(Character.id()) :: Item.fields()
defp fetch_forum_data(id) do
{:ok, _app_list} = Application.ensure_all_started(:req)
:dev_joy
|> Application.get_env(__MODULE__, endpoint: "https://elixirforum.com")
|> Keyword.fetch!(:endpoint)
|> then(fn endpoint ->
endpoint
|> Path.join("u/#{id}.json")
|> Req.get!()
|> then(& &1.body["user"])
|> then(&[avatar: avatar_from_template(&1["avatar_template"], endpoint), full_name: &1["name"]])
end)
|> tap(&Cache.put(id, &1))
end
@spec avatar_from_template(String.t(), String.t()) :: Character.avatar()
defp avatar_from_template(template, endpoint) do
template
|> String.replace("{size}", @avatar_size)
|> then(&Path.join(endpoint, &1))
end
end
```
<!-- tabs-close -->