Preoccupy the user's tab
Artsiom Mezin
@iketari
data:image/s3,"s3://crabby-images/30ef8/30ef877aa542533ca4811b43a67403073f5ce1e3" alt=""
About me
A Senior front-end dev at Behavox.
Have been doing web for over 10 years.
Always curious.
data:image/s3,"s3://crabby-images/dbdb2/dbdb2fdbd8241fcabb55bc539814171758b5abad" alt=""
What is it about?
It is about a way of implementing a remote access
data:image/s3,"s3://crabby-images/1cfe5/1cfe55bdac65f3408f7ba52755ae8dc7f0c9d48f" alt=""
Why do we need it?
-
To support
-
To collaborate
-
To report bugs and issues
-
To teach and to learn (theoretically)
-
… could be anything else what you can do toghter
Nothing new. It exists.
data:image/s3,"s3://crabby-images/687a4/687a48c1a184f28526777eae7a89a26a00c6d9d4" alt=""
Yes, but...
-
Access is granted not just to an app but the entire system.
-
Users have to install a client part of the solution.
-
An operator has to install a host program on his side.
data:image/s3,"s3://crabby-images/ac972/ac972a5747bad52253cd6ce8a5c58b4e4a1fcd96" alt=""
The Web can help
-
Browsers sandbox is isolated.
-
Web apps do not need to be installed.
-
The host also could be implemented as a web app.
data:image/s3,"s3://crabby-images/708bb/708bb92cc3ae66680069cc3eb7d7336cfe27a898" alt=""
Technological questions
-
How to share the user’s screen?
-
How to perform an event on the user's side and how to transfer it to there?
-
How to install the client part so that it would work within any web application?
data:image/s3,"s3://crabby-images/fbb2f/fbb2f8cf384b78520ccf06fa808aa791e89e0e18" alt=""
How to share the user’s screen?
-
HTML2Canvas -> canvas.captureStream(frameRate)
-
Do not share a screen, share the DOM
-
navigator.getDisplayMedia() + WebRTC
HTML to Canvas
console.time('HTMLToCanvas');
html2canvas(document.querySelector('body')).then(canvas => {
let img = canvas.toDataURL("image/png");
document.body.append('<img id="result" src="'+img+'"/>');
console.timeEnd('HTMLToCanvas');
});
// HTMLToCanvas: 26687.60693359375ms / 0.0374 FPS
data:image/s3,"s3://crabby-images/94be4/94be49edf13405eee7fe1252cef801d202732f7c" alt=""
TOGETHERJS
data:image/s3,"s3://crabby-images/d469f/d469f92914bdd86504f708d9be8f69781cf74dbf" alt=""
data:image/s3,"s3://crabby-images/01310/01310c731f5d97d898e4bf70c2c01dfb989c1dea" alt=""
Title Text
MediaDevices.getDisplayMedia()
let promise = navigator.mediaDevices.getDisplayMedia(constraints);
data:image/s3,"s3://crabby-images/df632/df6328533b97d4a368120da0c03fd5cd3125771b" alt=""
Title Text
MediaDevices.getDisplayMedia()
data:image/s3,"s3://crabby-images/1fbfb/1fbfb4edfb97ba50bb93fa89891e1faaf121b164" alt=""
Title Text
WebRTC
data:image/s3,"s3://crabby-images/ed46b/ed46bbfd48f7174c35b77ca62f57725ec5d3ccf8" alt=""
data:image/s3,"s3://crabby-images/76511/765117eaf5444032ae0aac84d18cc8624c4b4b65" alt=""
1. No plugins
2. P2P
3. Audio, Video and Data all encrypted in transit
Title Text
WebRTC. Signalling.
data:image/s3,"s3://crabby-images/ed46b/ed46bbfd48f7174c35b77ca62f57725ec5d3ccf8" alt=""
data:image/s3,"s3://crabby-images/f7e41/f7e41f4cae4d1f3ee1ee8984d8c9c98ca8fe645d" alt=""
Title Text
Signalling. AWS Lambda
data:image/s3,"s3://crabby-images/f15d8/f15d85b635772994e48b592c9a4d3700c792b841" alt=""
data:image/s3,"s3://crabby-images/4b7e3/4b7e384e98ffce66a4b5a681cac79b95a17742bc" alt=""
Title Text
Lambda. OnConnect
data:image/s3,"s3://crabby-images/f15d8/f15d85b635772994e48b592c9a4d3700c792b841" alt=""
// Copyright 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved.
// SPDX-License-Identifier: MIT-0
var AWS = require("aws-sdk");
AWS.config.update({ region: process.env.AWS_REGION });
var DDB = new AWS.DynamoDB({ apiVersion: "2012-10-08" });
exports.handler = function (event, context, callback) {
var putParams = {
TableName: process.env.TABLE_NAME,
Item: {
connectionId: { S: event.requestContext.connectionId }
}
};
DDB.putItem(putParams, function (err) {
callback(null, {
statusCode: err ? 500 : 200,
body: err ? "Failed to connect: " + JSON.stringify(err) : "Connected."
});
});
};
Title Text
Lambda. OnMessage
data:image/s3,"s3://crabby-images/f15d8/f15d85b635772994e48b592c9a4d3700c792b841" alt=""
exports.handler = async (event, context) => {
const apigwManagementApi = new AWS.ApiGatewayManagementApi({
apiVersion: '2018-11-29',
endpoint: event.requestContext.domainName + '/' + event.requestContext.stage
});
const { data } = JSON.parse(event.body);
switch (data.action) {
case 'register':
await registerController(data, event, ddb, apigwManagementApi);
break;
case 'communicate':
await communicateController(data, event, ddb, apigwManagementApi);
break;
default:
break;
}
}
Title Text
All together
data:image/s3,"s3://crabby-images/f15d8/f15d85b635772994e48b592c9a4d3700c792b841" alt=""
data:image/s3,"s3://crabby-images/ed46b/ed46bbfd48f7174c35b77ca62f57725ec5d3ccf8" alt=""
Technological questions
-
How to share the user’s screen? -
How to perform an event on the user's side and how to transfer it to there?
-
How to install the client part so that it would work within any web application?
data:image/s3,"s3://crabby-images/82ac9/82ac96002e1ccc53761ad46d46964a2cb96ddce1" alt=""
How to perform a desirable event on the user's side and how to transfer it there?
data:image/s3,"s3://crabby-images/6e390/6e39029fa0224c1ac0f2c3813b0bef8cffbdf318" alt=""
data:image/s3,"s3://crabby-images/6d711/6d7119b664129eaf309ed52b58f9a227a2795a0a" alt=""
Preoccupy
What happens when a user clicks on some element on the page?
-
The element is being located.
-
A JavaScript event is being fired.
-
A standard action is happening.
PreoccupyJS is trying to reproduce browser's behaviour
-
Find an element
-
Fire an event
-
Make changes in DOM if necessary
let el = document.elementFromPoint(x, y);
// https://developer.mozilla.org/en-US/docs/Web/API/DocumentOrShadowRoot/elementFromPoint
el.dispatchEvent(new MouseEvent('click', opts));
if (document.activeElement !== null) {
this.fireEvent('blur', document.activeElement as HTMLElement);
}
if (el.focus) {
el.focus();
}
PreoccupyJS contains a number of replaceable Actions each one for a dedicated browser event
export class ClickToAction implements PreoccupyAction {
static type: string = ActionsName.CLICK_TO;
static eventName: string = 'click';
constructor(
public payload: {
x: number;
y: number;
}
) {
this.type = ClickToAction.type;
}
performEvent(dom: DomController, stack: PreoccupyAction[]) {
dom.clickTo(this.payload);
}
static handleEvent(host: Host, event: Event): PreoccupyAction {
const payload = host.getRelativeCoordinate(event as MouseEvent);
return new ClickToAction(payload);
}
}
PreoccupyJS handles events on the host side, serialize and passes them to a transport
import { Host, RxjsTransport } from 'preoccupyjs';
// fill free to style and modify this element as you wish, but keep it in the DOM!
const hostEl = document.createElement('div');
hostEl.tabIndex = 0; // make it focusable
const transport = new RxjsTransport(options as {
subject: new Subject<object>; //e.g. WebSocketSubject
});
const host = new Host(transport, hostEl);
host.start(); // run the communication whenever your app is ready!
Transport is responsible for delivering a serialized action to the client's side, where PreoccupyJS will perform it
import { Host, RxjsTransport, DomController } from 'preoccupyjs';
//Set up the transport
const transport = new RxjsTransport(options as {
subject: new Subject() // e.g. WebSocketSubject
});
// Set up and run the client
// It's possible to specify the controlled scope of the page by passing any DOM element
const client = new Client(transport, new DomController(document.body));
// run the communication whenever your app is ready!
client.start();
Technological questions
-
How to share the user’s screen? -
How to perform an event on the user's side and how to transfer it to there? -
How to install the client part so that it would work within any web application?
data:image/s3,"s3://crabby-images/3e8b5/3e8b5e788118ba77caf27df5fdf13cf632dc545f" alt=""
How to install the client part so that it would work within any web application?
data:image/s3,"s3://crabby-images/dd22a/dd22a25bc109992a815289931bcc9084162083e8" alt=""
Bookmarklets
(in theory)
javascript:(function(){
//Statements returning a non-undefined type, e.g. assignments
})();
<a href="{bookmarkletText}"> Bookmark me </a>
import bookmarklet from './assets/bookmarklet.bundle';
// via JSX
return (
<a href={bookmarklet}>
Click me, I am a bookmarklet.
</a>
);
// via vanilla JS
let link = document.createElement('a');
link.href = bookmarklet;
link.innerHTML = 'Click me, I am a bookmaklet';
return link;
Bookmarklets
(in practice)
Demo
data:image/s3,"s3://crabby-images/b74fb/b74fb2835e04ca9f3eca7af836574fb7886ab515" alt=""
Thank you! Any questions?
Please tell me what to improve or what to dig into next time!
Twitter: @iketari
data:image/s3,"s3://crabby-images/6f012/6f01275e10f2e5c5a1cce82999795d9572a83827" alt=""
Title Text
PreoccupyJS
By iketari
PreoccupyJS
- 228