|
|
|
|
@@ -1,207 +0,0 @@
|
|
|
|
|
// Copyright 2015 The Chromium Authors. All rights reserved.
|
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
|
|
import 'dart:ui' as ui;
|
|
|
|
|
|
|
|
|
|
import 'package:mojo/application.dart';
|
|
|
|
|
import 'package:mojo/bindings.dart' as bindings;
|
|
|
|
|
import 'package:mojo/core.dart' as core;
|
|
|
|
|
import 'package:mojo/mojo/service_provider.mojom.dart' as mojom;
|
|
|
|
|
import 'package:mojo/mojo/shell.mojom.dart' as mojom;
|
|
|
|
|
|
|
|
|
|
/// Signature for connecting to services. The generated mojom.dart code has
|
|
|
|
|
/// static functions that match this signature on the interface objects. For
|
|
|
|
|
/// example, the `mojom.Foo` interface has a `mojom.Foo.connectToService`
|
|
|
|
|
/// function that matches this signature.
|
|
|
|
|
typedef bindings.Proxy<dynamic> ServiceConnectionCallback(
|
|
|
|
|
bindings.ServiceConnector s, String url, [String serviceName]);
|
|
|
|
|
|
|
|
|
|
/// Signature for replacements for [shell.connectToApplicationService]. If the
|
|
|
|
|
/// function returns `null`, then [shell.connectToApplicationService] will fall
|
|
|
|
|
/// back to its default behavior.
|
|
|
|
|
typedef bindings.Proxy<dynamic> OverrideConnectToService(String url, ServiceConnectionCallback callback);
|
|
|
|
|
|
|
|
|
|
ApplicationConnection _initEmbedderConnection() {
|
|
|
|
|
core.MojoHandle incomingServicesHandle = new core.MojoHandle(ui.MojoServices.takeIncomingServices());
|
|
|
|
|
core.MojoHandle outgoingServicesHandle = new core.MojoHandle(ui.MojoServices.takeOutgoingServices());
|
|
|
|
|
if (!incomingServicesHandle.isValid || !outgoingServicesHandle.isValid)
|
|
|
|
|
return null;
|
|
|
|
|
mojom.ServiceProviderProxy incomingServices = new mojom.ServiceProviderProxy.fromHandle(incomingServicesHandle);
|
|
|
|
|
mojom.ServiceProviderStub outgoingServices = new mojom.ServiceProviderStub.fromHandle(outgoingServicesHandle);
|
|
|
|
|
return new ApplicationConnection(outgoingServices, incomingServices);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mojom.ShellProxy _takeShell() {
|
|
|
|
|
core.MojoHandle shellHandle = new core.MojoHandle(ui.MojoServices.takeShell());
|
|
|
|
|
if (!shellHandle.isValid)
|
|
|
|
|
return null;
|
|
|
|
|
return new mojom.ShellProxy.fromHandle(shellHandle);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mojom.ServiceProviderProxy _takeViewServices() {
|
|
|
|
|
core.MojoHandle services = new core.MojoHandle(ui.MojoServices.takeViewServices());
|
|
|
|
|
if (!services.isValid)
|
|
|
|
|
return null;
|
|
|
|
|
return new mojom.ServiceProviderProxy.fromHandle(services);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _ShellServiceConnector extends bindings.ServiceConnector {
|
|
|
|
|
@override
|
|
|
|
|
void connectToService(String url, bindings.Proxy<dynamic> proxy, [String serviceName]) {
|
|
|
|
|
final MojoShell instance = MojoShell.instance;
|
|
|
|
|
if (url == null || instance._shell == null) {
|
|
|
|
|
// If the application URL is null, it means the service to connect
|
|
|
|
|
// to is one provided by the embedder.
|
|
|
|
|
// If the applircation URL isn't null but there's no shell, then
|
|
|
|
|
// ask the embedder in case it provides it. (For example, if you're
|
|
|
|
|
// running on Android without the Mojo shell, then you can obtain
|
|
|
|
|
// the media service from the embedder directly, instead of having
|
|
|
|
|
// to ask the media application for it.)
|
|
|
|
|
// This makes it easier to write an application that works both
|
|
|
|
|
// with and without a Mojo environment.
|
|
|
|
|
instance._embedderConnection?.requestService(proxy);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mojom.ServiceProviderProxy services = new mojom.ServiceProviderProxy.unbound();
|
|
|
|
|
instance._shell.connectToApplication(url, services);
|
|
|
|
|
core.MojoMessagePipe pipe = new core.MojoMessagePipe();
|
|
|
|
|
proxy.ctrl.bind(pipe.endpoints[0]);
|
|
|
|
|
services.connectToService_(serviceName, pipe.endpoints[1]);
|
|
|
|
|
services.close();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _ViewAssociatedServiceConnector extends bindings.ServiceConnector {
|
|
|
|
|
final mojom.ServiceProviderProxy _viewServices = _takeViewServices();
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void connectToService(String url, bindings.Proxy<dynamic> proxy, [String serviceName]) {
|
|
|
|
|
if (_viewServices != null) {
|
|
|
|
|
core.MojoMessagePipe pipe = new core.MojoMessagePipe();
|
|
|
|
|
proxy.ctrl.bind(pipe.endpoints[0]);
|
|
|
|
|
_viewServices.connectToService_(serviceName, pipe.endpoints[1]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Manages connections with embedder-provided services.
|
|
|
|
|
class MojoShell {
|
|
|
|
|
/// Creates the MojoShell singleton. This constructor can only be called once.
|
|
|
|
|
/// If your application uses bindings, it is called by the [ServicesBinding] binding.
|
|
|
|
|
/// (See [BindingBase] for more details on bindings. Any application using
|
|
|
|
|
/// the Flutter 'rendering' or 'widgets' libraries uses a binding.)
|
|
|
|
|
MojoShell() {
|
|
|
|
|
assert(_instance == null);
|
|
|
|
|
_instance = this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The unique instance of this class.
|
|
|
|
|
static MojoShell get instance => _instance;
|
|
|
|
|
static MojoShell _instance;
|
|
|
|
|
|
|
|
|
|
final ApplicationConnection _embedderConnection = _initEmbedderConnection();
|
|
|
|
|
final mojom.Shell _shell = _takeShell();
|
|
|
|
|
|
|
|
|
|
final _ShellServiceConnector _shellServiceConnector = new _ShellServiceConnector();
|
|
|
|
|
final _ViewAssociatedServiceConnector _viewServiceConnector = new _ViewAssociatedServiceConnector();
|
|
|
|
|
|
|
|
|
|
/// Whether [connectToApplication] is able to connect to other applications.
|
|
|
|
|
bool get canConnectToOtherApplications => _shell != null;
|
|
|
|
|
|
|
|
|
|
/// Attempts to connect to an application via the Mojo shell.
|
|
|
|
|
///
|
|
|
|
|
/// Returns `null` if [canConnectToOtherApplications] is false.
|
|
|
|
|
ApplicationConnection connectToApplication(String url) {
|
|
|
|
|
if (_shell == null)
|
|
|
|
|
return null;
|
|
|
|
|
mojom.ServiceProviderProxy services = new mojom.ServiceProviderProxy.unbound();
|
|
|
|
|
_shell.connectToApplication(url, services);
|
|
|
|
|
return new ApplicationConnection(null, services);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Interceptor for calls to [connectToService] and
|
|
|
|
|
/// [connectToViewAssociatedService] so that tests can supply alternative
|
|
|
|
|
/// implementations of services (for example, a mock for testing).
|
|
|
|
|
OverrideConnectToService overrideConnectToService;
|
|
|
|
|
|
|
|
|
|
/// Attempts to connect to a service implementing the interface for the given
|
|
|
|
|
/// proxy. If an application URL is specified and
|
|
|
|
|
/// [canConnectToOtherApplications] is true, the service will be requested
|
|
|
|
|
/// from that application. Otherwise, it will be requested from the embedder
|
|
|
|
|
/// (the Flutter engine).
|
|
|
|
|
///
|
|
|
|
|
/// For example, suppose there was a service of type `Foo` that was normally
|
|
|
|
|
/// hosted with the URL "mojo:foo" and that was also provided by the Flutter
|
|
|
|
|
/// embedder when there is no shell (i.e. when [canConnectToOtherApplications]
|
|
|
|
|
/// returns false). The following code (assuming the relevant mojom file
|
|
|
|
|
/// declaring `Foo` was imported with the prefix `mojom`) would connect to it,
|
|
|
|
|
/// and then invoke the method `bar()` on it:
|
|
|
|
|
///
|
|
|
|
|
/// ```dart
|
|
|
|
|
/// mojom.FooProxy foo = shell.connectToApplicationService(
|
|
|
|
|
/// "mojo:foo", mojom.Foo.connectToService);
|
|
|
|
|
/// foo.bar();
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// For examples of mojom files, see the `sky_services` package.
|
|
|
|
|
///
|
|
|
|
|
/// See also [connectToViewAssociatedService].
|
|
|
|
|
bindings.Proxy<dynamic> connectToApplicationService(String url, ServiceConnectionCallback callback) {
|
|
|
|
|
bindings.Proxy<dynamic> proxy;
|
|
|
|
|
if (overrideConnectToService != null)
|
|
|
|
|
proxy = overrideConnectToService(url, callback);
|
|
|
|
|
return proxy ?? callback(_shellServiceConnector, url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Attempts to connect to a service provided specifically for the current
|
|
|
|
|
/// view by the embedder or host platform.
|
|
|
|
|
///
|
|
|
|
|
/// For example, keyboard services are specific to a view; you can only
|
|
|
|
|
/// receive keyboard input when the application's view is the one with focus.
|
|
|
|
|
///
|
|
|
|
|
/// For example, suppose there was a service of type `Foo` that was provided
|
|
|
|
|
/// on a view-by-view basis by the embedder or host platform. The following
|
|
|
|
|
/// code (assuming the relevant mojom file declaring `Foo` was imported with
|
|
|
|
|
/// the prefix `mojom`) would connect to it, and then invoke the method
|
|
|
|
|
/// `bar()` on it:
|
|
|
|
|
///
|
|
|
|
|
/// ```dart
|
|
|
|
|
/// mojom.FooProxy foo = shell.connectToViewAssociatedService(
|
|
|
|
|
/// mojom.Foo.connectToService);
|
|
|
|
|
/// foo.bar();
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// For examples of mojom files, see the `sky_services` package.
|
|
|
|
|
///
|
|
|
|
|
/// See also [connectToService].
|
|
|
|
|
bindings.Proxy<dynamic> connectToViewAssociatedService(ServiceConnectionCallback callback) {
|
|
|
|
|
bindings.Proxy<dynamic> proxy;
|
|
|
|
|
if (overrideConnectToService != null)
|
|
|
|
|
proxy = overrideConnectToService(null, callback);
|
|
|
|
|
return proxy ?? callback(_viewServiceConnector, null);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Registers a service to expose to the embedder.
|
|
|
|
|
///
|
|
|
|
|
/// For example, suppose a Flutter application wanted to provide a service
|
|
|
|
|
/// `Foo` to the embedder, that a mojom file declaring `Foo` was imported with
|
|
|
|
|
/// the prefix `mojom`, that `package:mojo/core.dart` was imported with the
|
|
|
|
|
/// prefix `core`, and that an implementation of the `Foo` service existed in
|
|
|
|
|
/// the class `MyFooImplementation`. The following code, run during the
|
|
|
|
|
/// binding initialization (i.e. during the same call stack as the call to the
|
|
|
|
|
/// [new MojoShell] constructor) would achieve this:
|
|
|
|
|
///
|
|
|
|
|
/// ```dart
|
|
|
|
|
/// shell.provideService(mojom.Foo.serviceName, (core.MojoMessagePipeEndpoint endpoint) {
|
|
|
|
|
/// mojom.FooStub foo = new mojom.FooStub.fromEndpoint(endpoint);
|
|
|
|
|
/// foo.impl = new MyFooImplementation();
|
|
|
|
|
/// });
|
|
|
|
|
/// ```
|
|
|
|
|
void provideService(String interfaceName, ServiceFactory factory) {
|
|
|
|
|
_embedderConnection?.provideService(interfaceName, factory);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The singleton object that manages connections with embedder-provided services.
|
|
|
|
|
MojoShell get shell => MojoShell.instance;
|