aboutsummaryrefslogtreecommitdiffstats
path: root/src/servo/engine.rs
blob: dc693e7e2a087e1614d9581a93922d8ad5fe67c0 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

use compositing::CompositorImpl;
use content::content_task::{ContentTask, ExecuteMsg, ParseMsg};
use content::content_task;
use dom::event::Event;
use layout::layout_task;
use layout::layout_task::LayoutTask;
use resource::image_cache_task::{ImageCacheTask, ImageCacheTaskClient};
use resource::resource_task::ResourceTask;
use resource::resource_task;
use util::task::spawn_listener;

use core::cell::Cell;
use core::comm::{Chan, Port, SharedChan};
use gfx::compositor::Compositor;
use gfx::opts::Opts;
use gfx::render_task::RenderTask;
use gfx::render_task;
use std::net::url::Url;

pub type EngineTask = Chan<Msg>;

pub enum Msg {
    LoadUrlMsg(Url),
    ExitMsg(Chan<()>)
}

pub struct Engine {
    request_port: Port<Msg>,
    compositor: CompositorImpl,
    render_task: RenderTask,
    resource_task: ResourceTask,
    image_cache_task: ImageCacheTask,
    layout_task: LayoutTask,
    content_task: ContentTask
}

impl Engine {
    pub fn start(compositor: CompositorImpl,
                 opts: &Opts,
                 dom_event_port: Port<Event>,
                 dom_event_chan: SharedChan<Event>,
                 resource_task: ResourceTask,
                 image_cache_task: ImageCacheTask)
                 -> EngineTask {
        let dom_event_port = Cell(dom_event_port);
        let dom_event_chan = Cell(dom_event_chan);

        let opts = Cell(copy *opts);
        do spawn_listener::<Msg> |request| {
            let render_task = RenderTask(compositor.clone(), opts.with_ref(|o| copy *o));

            let opts = opts.take();
            let layout_task = LayoutTask(render_task.clone(), image_cache_task.clone(), opts);

            let content_task = ContentTask(layout_task.clone(),
                                           dom_event_port.take(),
                                           dom_event_chan.take(),
                                           resource_task.clone(),
                                           image_cache_task.clone());

            Engine {
                request_port: request,
                compositor: compositor.clone(),
                render_task: render_task,
                resource_task: resource_task.clone(),
                image_cache_task: image_cache_task.clone(),
                layout_task: layout_task,
                content_task: content_task,
            }.run()
        }
    }

    fn run(&self) {
        while self.handle_request(self.request_port.recv()) {
            // Go on...
        }
    }

    fn handle_request(&self, request: Msg) -> bool {
        match request {
            LoadUrlMsg(url) => {
                if url.path.ends_with(".js") {
                    self.content_task.send(ExecuteMsg(url))
                } else {
                    self.content_task.send(ParseMsg(url))
                }
                return true
            }

            ExitMsg(sender) => {
                self.content_task.send(content_task::ExitMsg);
                self.layout_task.send(layout_task::ExitMsg);

                let (response_port, response_chan) = comm::stream();

                self.render_task.send(render_task::ExitMsg(response_chan));
                response_port.recv();

                self.image_cache_task.exit();
                self.resource_task.send(resource_task::Exit);

                sender.send(());
                return false
            }
        }
    }
}