Skip to content

Commit 0a853a5

Browse files
committed
first commit. Build a simple example app using Scenic v0.11
0 parents  commit 0a853a5

32 files changed

+1223
-0
lines changed

.formatter.exs

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
# Used by "mix format"
2+
[
3+
inputs: ["{mix,.formatter}.exs", "{config,lib,test}/**/*.{ex,exs}"]
4+
]

.gitignore

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
# The directory Mix will write compiled artifacts to.
2+
/_build/
3+
4+
# If you run "mix test --cover", coverage assets end up here.
5+
/cover/
6+
7+
# The directory Mix downloads your dependencies sources to.
8+
/deps/
9+
10+
# Where third-party dependencies like ExDoc output generated docs.
11+
/doc/
12+
13+
# Ignore .fetch files in case you like to edit your project deps locally.
14+
/.fetch
15+
16+
# If the VM crashes, it generates a dump, let's ignore it too.
17+
erl_crash.dump
18+
19+
# Also ignore archive artifacts (built via "mix archive.build").
20+
*.ez
21+
22+
# Ignore package tarball (built via "mix hex.build").
23+
example-*.tar
24+
25+
# Temporary files, for example, from tests.
26+
/tmp/
27+
28+
.DS_Store
29+
30+
/priv/assets/*

README.md

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
# Example
2+
3+
**TODO: Add description**
4+
5+
## Installation
6+
7+
If [available in Hex](https://hex.pm/docs/publish), the package can be installed
8+
by adding `example` to your list of dependencies in `mix.exs`:
9+
10+
```elixir
11+
def deps do
12+
[
13+
{:example, "~> 0.1.0"}
14+
]
15+
end
16+
```
17+
18+
Documentation can be generated with [ExDoc](https://github.com/elixir-lang/ex_doc)
19+
and published on [HexDocs](https://hexdocs.pm). Once published, the docs can
20+
be found at [https://hexdocs.pm/example](https://hexdocs.pm/example).
21+

assets/fonts/akaya_telivigala.ttf

75.5 KB
Binary file not shown.

assets/fonts/bitter.ttf

88 KB
Binary file not shown.

assets/fonts/dot_gothic_16.ttf

91.4 KB
Binary file not shown.

assets/fonts/fascinate.ttf

34.5 KB
Binary file not shown.

assets/fonts/oi.ttf

54.3 KB
Binary file not shown.

assets/fonts/roboto.ttf

35.3 KB
Binary file not shown.

assets/fonts/roboto_mono.ttf

21.7 KB
Binary file not shown.

assets/fonts/stick.ttf

67.6 KB
Binary file not shown.

assets/images/fairy_grove.jpg

144 KB
Loading

assets/images/parrot.jpg

11.6 KB
Loading

config/config.exs

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
# This file is responsible for configuring your umbrella
2+
# and **all applications** and their dependencies with the
3+
# help of Mix.Config.
4+
#
5+
# Note that all applications in your umbrella share the
6+
# same configuration and dependencies, which is why they
7+
# all use the same configuration file. If you want different
8+
# configurations or dependencies per app, it is best to
9+
# move said applications out of the umbrella.
10+
use Mix.Config
11+
12+
# Configures Elixir's Logger
13+
config :logger, :console,
14+
format: "$time $metadata[$level] $message\n",
15+
metadata: [:request_id]
16+
17+
18+
# should normally live in config.exs of the device app
19+
config :scenic, :assets,
20+
module: Example.Assets,
21+
alias: [
22+
parrot: "images/parrot.jpg"
23+
]
24+
25+
# Import environment specific config. This must remain at the bottom
26+
# of this file so it overrides the configuration defined above.
27+
import_config "#{Mix.env()}.exs"

config/dev.exs

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
use Mix.Config
2+
3+
# Do not include metadata nor timestamps in development logs
4+
config :logger, :console, format: "[$level] $message\n"
5+
6+
# Configure the main viewport for the Scenic application
7+
config :example, :viewport, [
8+
name: :main_viewport,
9+
size: {800, 600},
10+
theme: :dark,
11+
default_scene: Example.Scene.Sensor,
12+
drivers: [[
13+
module: Scenic.Driver.Glfw,
14+
name: :glfw_driver,
15+
title: "Super Test Window",
16+
resizeable: false
17+
# on_close: :stop_driver
18+
]]
19+
]
20+

lib/assets.ex

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
defmodule Example.Assets do
2+
use Scenic.Assets.Static, otp_app: :example
3+
end
4+

lib/components/nav.ex

+96
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
defmodule Example.Component.Nav do
2+
use Scenic.Component
3+
4+
alias Scenic.Graph
5+
alias Scenic.ViewPort
6+
7+
import Scenic.Primitives, only: [{:text, 3}, {:rect, 3}]
8+
import Scenic.Components, only: [{:dropdown, 3}, {:toggle, 3}]
9+
10+
@height 60
11+
12+
# --------------------------------------------------------
13+
def validate(scene) when is_atom(scene), do: {:ok, scene}
14+
def validate({scene, _} = data) when is_atom(scene), do: {:ok, data}
15+
def validate(data) do
16+
{
17+
:error,
18+
"""
19+
#{IO.ANSI.red()}Invalid #{inspect(__MODULE__)} specification
20+
Received: #{inspect(data)}
21+
#{IO.ANSI.yellow()}
22+
Nav component requires a valid scene module or {module, param}
23+
"""
24+
}
25+
end
26+
27+
# ----------------------------------------------------------------------------
28+
def init(scene, current_scene, opts) do
29+
{width, _} = scene.viewport.size
30+
31+
{background, text} = case opts[:theme] do
32+
:dark ->
33+
{{48, 48, 48}, :white}
34+
35+
:light ->
36+
{{220, 220, 220}, :black}
37+
38+
_ ->
39+
{{48, 48, 48}, :white}
40+
end
41+
42+
graph = Graph.build( font_size: 20 )
43+
|> rect( {width, @height}, fill: background )
44+
|> text( "Scene:", translate: {15, 38}, align: :right, fill: text )
45+
|> dropdown(
46+
{[
47+
{"Sensor", Example.Scene.Sensor},
48+
{"Primitives", Example.Scene.Primitives},
49+
{"Fills", Example.Scene.Fills},
50+
{"Strokes", Example.Scene.Strokes},
51+
{"Components", Example.Scene.Components},
52+
{"Transforms", Example.Scene.Transforms},
53+
{"Sprites", Example.Scene.Sprites},
54+
], current_scene},
55+
id: :nav,
56+
translate: {90, 10}
57+
)
58+
|> toggle(
59+
opts[:theme] == :light,
60+
id: :light_or_dark,
61+
theme: :secondary,
62+
translate: {width - 60, 16},
63+
)
64+
# |> digital_clock(text_align: :right, translate: {width - 20, 35})
65+
66+
67+
push_graph(scene, graph)
68+
69+
{ :ok, scene }
70+
end
71+
72+
73+
# ----------------------------------------------------------------------------
74+
def handle_event({:value_changed, :nav, {scene_mod, param}}, _, scene) do
75+
ViewPort.set_root(scene.viewport, scene_mod, param)
76+
{ :noreply, scene }
77+
end
78+
79+
# ----------------------------------------------------------------------------
80+
def handle_event({:value_changed, :nav, scene_mod}, _, scene) do
81+
ViewPort.set_root(scene.viewport, scene_mod)
82+
{ :noreply, scene }
83+
end
84+
85+
# ----------------------------------------------------------------------------
86+
def handle_event({:value_changed, :light_or_dark, light?}, _, scene) do
87+
case light? do
88+
true -> ViewPort.set_theme(scene.viewport, :light)
89+
false -> ViewPort.set_theme(scene.viewport, :dark)
90+
end
91+
{ :noreply, scene }
92+
end
93+
94+
end
95+
96+

lib/components/notes.ex

+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
defmodule Example.Component.Notes do
2+
use Scenic.Component, has_children: false
3+
4+
# alias Scenic.ViewPort
5+
alias Scenic.Scene
6+
alias Scenic.Graph
7+
8+
import Scenic.Primitives, only: [{:text, 3}, {:rect, 3}]
9+
10+
@height 110
11+
@font_size 20
12+
@indent 30
13+
14+
15+
# --------------------------------------------------------
16+
def validate(notes) when is_bitstring(notes), do: {:ok, notes}
17+
def validate(data) do
18+
{
19+
:error,
20+
"""
21+
#{IO.ANSI.red()}Invalid #{inspect(__MODULE__)} specification
22+
Received: #{inspect(data)}
23+
#{IO.ANSI.yellow()}
24+
Notes data should be a string
25+
"""
26+
}
27+
end
28+
29+
# ----------------------------------------------------------------------------
30+
def init(%Scene{viewport: viewport } = scene, notes, opts) do
31+
{vp_width, vp_height} = viewport.size
32+
33+
{background, text} = case opts[:theme] do
34+
:dark ->
35+
{{48, 48, 48}, :white}
36+
37+
:light ->
38+
{{220, 220, 220}, :black}
39+
40+
_ ->
41+
{{48, 48, 48}, :white}
42+
end
43+
44+
graph = Graph.build(
45+
font_size: @font_size, font: :roboto,
46+
t: {0, vp_height - @height}, theme: opts[:theme]
47+
)
48+
|> rect( {vp_width, @height}, fill: background )
49+
|> text( notes, translate: {@indent, @font_size * 2}, fill: text )
50+
51+
{ :ok, push_graph(scene, graph) }
52+
end
53+
end

lib/example.ex

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
defmodule Example do
2+
@moduledoc """
3+
Documentation for `Example`.
4+
"""
5+
6+
def start(_type, _args) do
7+
# load the viewport configuration from config
8+
main_viewport = Application.get_env(:example, :viewport)
9+
10+
# start the application with the default view_port
11+
children = [
12+
{Scenic, [main_viewport]},
13+
Example.Sensors.Supervisor,
14+
]
15+
Supervisor.start_link(children, strategy: :one_for_one)
16+
end
17+
end
18+
19+

0 commit comments

Comments
 (0)