1gShowBounds = false
2gUseBlurInTransitions = false
3
4gPath = "resources/"
5
6function load_file(file)
7    local prev_path = package.path
8    package.path = package.path .. ";" .. gPath .. file .. ".lua"
9    require(file)
10    package.path = prev_path
11end
12
13load_file("slides_utils")
14
15gSlides = parse_file(io.open("resources/slides_content2.lua", "r"))
16
17function make_rect(l, t, r, b)
18    return { left = l, top = t, right = r, bottom = b }
19end
20
21function make_paint(typefacename, style, size, color)
22    local paint = Sk.newPaint();
23    paint:setAntiAlias(true)
24    paint:setSubpixelText(true)
25    paint:setTypeface(Sk.newTypeface(typefacename, style))
26    paint:setTextSize(size)
27    paint:setColor(color)
28    return paint
29end
30
31function draw_bullet(canvas, x, y, paint, indent)
32    if 0 == indent then
33        return
34    end
35    local ps = paint:getTextSize()
36    local cx = x - ps * .8
37    local cy = y - ps * .4
38    local radius = ps * .2
39    canvas:drawCircle(cx, cy, radius, paint)
40end
41
42function stroke_rect(canvas, rect, color)
43    local paint = Sk.newPaint()
44    paint:setStroke(true);
45    paint:setColor(color)
46    canvas:drawRect(rect, paint)
47end
48
49function drawSlide(canvas, slide, master_template)
50
51    if #slide == 1 then
52        template = master_template.title
53        canvas:drawText(slide[1].text, 320, 240, template[1])
54        return
55    end
56
57    template = master_template.slide
58
59    local x = template.margin_x
60    local y = template.margin_y
61    local scale = 1.25
62
63    if slide.blockstyle == "code" then
64        local paint = master_template.codePaint
65        local fm = paint:getFontMetrics()
66        local height = #slide * (fm.descent - fm.ascent)
67        y = (480 - height) / 2
68        for i = 1, #slide do
69            local node = slide[i]
70            y = y - fm.ascent * scale
71            canvas:drawText(node.text, x, y, paint)
72            y = y + fm.descent * scale
73        end
74        return
75    end
76
77    for i = 1, #slide do
78        local node = slide[i]
79        local paint = template[node.indent + 1].paint
80        local extra_dy = template[node.indent + 1].extra_dy
81        local fm = paint:getFontMetrics()
82        local x_offset = -fm.ascent * node.indent * 1.25
83
84        local bounds = make_rect(x + x_offset, y, 620, 640)
85        local blob, newBottom = Sk.newTextBlob(node.text, bounds, paint)
86        draw_bullet(canvas, x + x_offset, y - fm.ascent, paint, node.indent)
87        canvas:drawTextBlob(blob, 0, 0, paint)
88        y = newBottom + paint:getTextSize() * .5 + extra_dy
89
90        if gShowBounds then
91            bounds.bottom = newBottom
92            stroke_rect(canvas, bounds, {a=1,r=0,g=1,b=0})
93            stroke_rect(canvas, blob:bounds(), {a=1,r=1,g=0,b=0})
94        end
95
96    end
97end
98
99--------------------------------------------------------------------------------------
100function make_tmpl(paint, extra_dy)
101    return { paint = paint, extra_dy = extra_dy }
102end
103
104function SkiaPoint_make_template()
105    normal = Sk.newFontStyle()
106    bold = Sk.newFontStyle(700)
107    local title = {
108        margin_x = 30,
109        margin_y = 100,
110    }
111    title[1] = make_paint("Arial", bold, 45, { a=1, r=1, g=1, b=1 })
112    title[1]:setTextAlign("center")
113    title[2] = make_paint("Arial", bold, 25, { a=1, r=.75, g=.75, b=.75 })
114    title[2]:setTextAlign("center")
115
116    local slide = {
117        margin_x = 20,
118        margin_y = 25,
119    }
120    slide[1] = make_tmpl(make_paint("Arial", bold, 35, { a=1, r=1, g=1, b=1 }), 18)
121    slide[2] = make_tmpl(make_paint("Arial", normal, 25, { a=1, r=1, g=1, b=1 }), 10)
122    slide[3] = make_tmpl(make_paint("Arial", normal, 20, { a=1, r=.9, g=.9, b=.9 }), 5)
123
124    return {
125        title = title,
126        slide = slide,
127        codePaint = make_paint("Courier", normal, 20, { a=1, r=.9, g=.9, b=.9 }),
128    }
129end
130
131gTemplate = SkiaPoint_make_template()
132
133gRedPaint = Sk.newPaint()
134gRedPaint:setAntiAlias(true)
135gRedPaint:setColor{a=1, r=1, g=0, b=0 }
136
137-- animation.proc is passed the canvas before drawing.
138-- The animation.proc returns itself or another animation (which means keep animating)
139-- or it returns nil, which stops the animation.
140--
141local gCurrAnimation
142
143gSlideIndex = 1
144
145-----------------------------------------------------------------------------
146
147function new_drawable_picture(pic)
148    return {
149        picture = pic,
150        width = pic:width(),
151        height = pic:height(),
152        draw = function (self, canvas, x, y, paint)
153            canvas:drawPicture(self.picture, x, y, paint)
154        end
155    }
156end
157
158function new_drawable_image(img)
159    return {
160        image = img,
161        width = img:width(),
162        height = img:height(),
163        draw = function (self, canvas, x, y, paint)
164            canvas:drawImage(self.image, x, y, paint)
165        end
166    }
167end
168
169function convert_to_picture_drawable(slide)
170    local rec = Sk.newPictureRecorder()
171    drawSlide(rec:beginRecording(640, 480), slide, gTemplate)
172    return new_drawable_picture(rec:endRecording())
173end
174
175function convert_to_image_drawable(slide)
176    local surf = Sk.newRasterSurface(640, 480)
177    drawSlide(surf:getCanvas(), slide, gTemplate)
178    return new_drawable_image(surf:newImageSnapshot())
179end
180
181function new_drawable_slide(slide)
182    return {
183        slide = slide,
184        draw = function (self, canvas, x, y, paint)
185            if (nil == paint or ("number" == type(paint) and (1 == paint))) then
186                canvas:save()
187            else
188                canvas:saveLayer(paint)
189            end
190            canvas:translate(x, y)
191            drawSlide(canvas, self.slide, gTemplate)
192            canvas:restore()
193        end
194    }
195end
196
197gNewDrawableFactory = {
198    default = new_drawable_slide,
199    picture = convert_to_picture_drawable,
200    image = convert_to_image_drawable,
201}
202
203-----------------------------------------------------------------------------
204
205function next_slide()
206    local prev = gSlides[gSlideIndex]
207
208    if gSlideIndex < #gSlides then
209        gSlideIndex = gSlideIndex + 1
210        spawn_transition(prev, gSlides[gSlideIndex], true)
211    end
212end
213
214function prev_slide()
215    local prev = gSlides[gSlideIndex]
216
217    if gSlideIndex > 1 then
218        gSlideIndex = gSlideIndex - 1
219        spawn_transition(prev, gSlides[gSlideIndex], false)
220    end
221end
222
223gDrawableType = "default"
224
225load_file("slides_transitions")
226
227function spawn_transition(prevSlide, nextSlide, is_forward)
228    local transition
229    if is_forward then
230        transition = gTransitionTable[nextSlide.transition]
231    else
232        transition = gTransitionTable[prevSlide.transition]
233    end
234
235    if not transition then
236        transition = fade_slide_transition
237    end
238
239    local prevDrawable = gNewDrawableFactory[gDrawableType](prevSlide)
240    local nextDrawable = gNewDrawableFactory[gDrawableType](nextSlide)
241    gCurrAnimation = transition(prevDrawable, nextDrawable, is_forward)
242end
243
244--------------------------------------------------------------------------------------
245
246function spawn_rotate_animation()
247    gCurrAnimation = {
248        angle = 0,
249        angle_delta = 5,
250        pivot_x = 320,
251        pivot_y = 240,
252        proc = function (self, canvas, drawSlideProc)
253            if self.angle >= 360 then
254                drawSlideProc(canvas)
255                return nil
256            end
257            canvas:translate(self.pivot_x, self.pivot_y)
258            canvas:rotate(self.angle)
259            canvas:translate(-self.pivot_x, -self.pivot_y)
260            drawSlideProc(canvas)
261
262            self.angle = self.angle + self.angle_delta
263            return self
264        end
265    }
266end
267
268function spawn_scale_animation()
269    gCurrAnimation = {
270        scale = 1,
271        scale_delta = .95,
272        scale_limit = 0.2,
273        pivot_x = 320,
274        pivot_y = 240,
275        proc = function (self, canvas, drawSlideProc)
276            if self.scale < self.scale_limit then
277                self.scale = self.scale_limit
278                self.scale_delta = 1 / self.scale_delta
279            end
280            if self.scale > 1 then
281                drawSlideProc(canvas)
282                return nil
283            end
284            canvas:translate(self.pivot_x, self.pivot_y)
285            canvas:scale(self.scale, self.scale)
286            canvas:translate(-self.pivot_x, -self.pivot_y)
287            drawSlideProc(canvas)
288
289            self.scale = self.scale * self.scale_delta
290            return self
291        end
292    }
293end
294
295local bgPaint = nil
296
297function draw_bg(canvas)
298    if not bgPaint then
299        bgPaint = Sk.newPaint()
300        local grad = Sk.newLinearGradient(  0,   0, { a=1, r=0, g=0, b=.3 },
301                                          640, 480, { a=1, r=0, g=0, b=.8 })
302        bgPaint:setShader(grad)
303        bgPaint:setDither(true)
304    end
305
306    canvas:drawPaint(bgPaint)
307end
308
309function onDrawContent(canvas, width, height)
310    local matrix = Sk.newMatrix()
311    matrix:setRectToRect(make_rect(0, 0, 640, 480), make_rect(0, 0, width, height), "center")
312    canvas:concat(matrix)
313
314    draw_bg(canvas)
315
316    local drawSlideProc = function(canvas)
317        drawSlide(canvas, gSlides[gSlideIndex], gTemplate)
318    end
319
320    if gCurrAnimation then
321        gCurrAnimation = gCurrAnimation:proc(canvas, drawSlideProc)
322        return true
323    else
324        drawSlideProc(canvas)
325        return false
326    end
327end
328
329function onClickHandler(x, y)
330    return false
331end
332
333local keyProcs = {
334    n = next_slide,
335    p = prev_slide,
336    r = spawn_rotate_animation,
337    s = spawn_scale_animation,
338    ["="] = function () scale_text_delta(gTemplate, 1) end,
339    ["-"] = function () scale_text_delta(gTemplate, -1) end,
340
341    b = function () gShowBounds = not gShowBounds end,
342    B = function () gUseBlurInTransitions = not gUseBlurInTransitions end,
343
344    ["1"] = function () gDrawableType = "default" end,
345    ["2"] = function () gDrawableType = "picture" end,
346    ["3"] = function () gDrawableType = "image" end,
347}
348
349function onCharHandler(uni)
350    local proc = keyProcs[uni]
351    if proc then
352        proc()
353        return true
354    end
355    return false
356end
357