mirror of
https://github.com/matrix-construct/construct
synced 2025-01-07 13:25:22 +01:00
209 lines
4.8 KiB
JavaScript
209 lines
4.8 KiB
JavaScript
/*
|
|
// Matrix Construct
|
|
//
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
|
|
//
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
// full license for this software is available in the LICENSE file.
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
/**************************************
|
|
* Sync
|
|
*
|
|
* Called with a block of data specific to this room obtained from
|
|
* the /sync resource. Called by the rooms.sync suite which is called
|
|
* by the main sync loop.
|
|
*
|
|
* The action specifies which category the data was found under i.e "join"
|
|
* "leave" or "invite". Rather than having separate function handlers for
|
|
* each action, we simply consider it as advice through a single stack
|
|
* of handlers.
|
|
*/
|
|
room.sync = function(data, action)
|
|
{
|
|
// `data` can be a single event which is reprocessed into the standard
|
|
// sync structure; see: `room.sync.one()`
|
|
if((data instanceof mc.event))
|
|
return this.sync.one(data, action);
|
|
|
|
Object.each(data, (name, data) =>
|
|
{
|
|
if(!(name in this.sync.handle))
|
|
{
|
|
console.warn("Unhandled room.sync type '" + name + "'");
|
|
return;
|
|
}
|
|
|
|
this.sync.handle[name](data, action);
|
|
});
|
|
|
|
if(this.focused())
|
|
{
|
|
this.sync.activity();
|
|
if(this.control.mode == "LIVE") try
|
|
{
|
|
this.scroll.to.bottom("fast");
|
|
//TODO: only call this after some user activity.
|
|
//this.receipt.send.current();
|
|
}
|
|
catch(e)
|
|
{
|
|
console.error(e);
|
|
}
|
|
}
|
|
};
|
|
|
|
room.sync.one = function(event, action = undefined)
|
|
{
|
|
if(!(event instanceof mc.event))
|
|
event = new mc.event(event);
|
|
|
|
let data = {};
|
|
let key = defined(event.state_key)? "state" : "timeline";
|
|
data[key] = {};
|
|
data[key].events = [event];
|
|
this.sync(data, action);
|
|
}
|
|
|
|
room.sync.activity = function()
|
|
{
|
|
if(this.control.activity === true)
|
|
return;
|
|
|
|
let timeout = 100;
|
|
this.control.activity = true;
|
|
mc.timeout(timeout, () => this.control.activity = false);
|
|
};
|
|
|
|
/** This function is applied to events before they are handed off
|
|
* to the room's timeline. The timeline may have other alterations.
|
|
*/
|
|
room.sync.fix_event = function(event, action)
|
|
{
|
|
if(event.type == "m.room.member")
|
|
if(event.state_key == mc.my.mxid)
|
|
if(!event.membership && !maybe(() => event.content.membership) && action !== undefined)
|
|
event.membership = action;
|
|
|
|
if(!event.origin_server_ts)
|
|
event.origin_server_ts = mc.now();
|
|
};
|
|
|
|
/** Handlers for members of the sync data for this room. Each member of this
|
|
* object is a handler for a corresponding named key in the sync data parsed
|
|
* by the room's sync function.
|
|
*/
|
|
room.sync.handle = {};
|
|
|
|
/**
|
|
* @param data - The member value in the sync data for the key this handler is handling.
|
|
* @param action - The action from sync should be passed along with the data
|
|
*/
|
|
room.sync.handle["state"] = function(data, action)
|
|
{
|
|
let events = data.events;
|
|
if(events === undefined)
|
|
return;
|
|
|
|
events.forEach((event) =>
|
|
{
|
|
this.sync.fix_event(event, action);
|
|
mc.users.learn(event);
|
|
});
|
|
|
|
this.timeline.insert(events);
|
|
events.forEach((event) =>
|
|
{
|
|
this.oracle.invalidate(event.type);
|
|
});
|
|
};
|
|
|
|
room.sync.handle["timeline"] = function(data, action)
|
|
{
|
|
let events = data.events;
|
|
if(events === undefined)
|
|
return;
|
|
|
|
events.forEach((event) =>
|
|
{
|
|
this.sync.fix_event(event, action);
|
|
mc.users.learn(event);
|
|
});
|
|
|
|
this.timeline.insert(events);
|
|
};
|
|
|
|
room.sync.handle["account_data"] = function(data, action)
|
|
{
|
|
let events = data.events;
|
|
if(events === undefined)
|
|
return;
|
|
|
|
let handlers = this.sync.handle["account_data"];
|
|
events.forEach((event) =>
|
|
{
|
|
let type = event.type;
|
|
if(!(type in handlers))
|
|
{
|
|
console.warn("Unhandled account_data [" + type + "] in " + this.id);
|
|
return;
|
|
}
|
|
|
|
handlers[type](event);
|
|
});
|
|
};
|
|
|
|
room.sync.handle["account_data"]["ircd_storage"] = function(event, action)
|
|
{
|
|
this.account.update(event.content);
|
|
};
|
|
|
|
room.sync.handle["unread_notifications"] = function(data, action)
|
|
{
|
|
Object.update(this.notifications, data);
|
|
};
|
|
|
|
room.sync.handle["ephemeral"] = function(data)
|
|
{
|
|
let events = data.events;
|
|
if(events === undefined)
|
|
return;
|
|
|
|
let handlers = this.sync.handle["ephemeral"];
|
|
events.forEach((event) =>
|
|
{
|
|
let type = event.type;
|
|
if(!(type in handlers))
|
|
{
|
|
console.warn("Unhandled ephemeral [" + type + "] in " + this.id);
|
|
return;
|
|
}
|
|
|
|
this.sync.fix_event(event);
|
|
mc.users.learn(event);
|
|
handlers[type](event);
|
|
});
|
|
};
|
|
|
|
room.sync.handle["ephemeral"]["m.typing"] = function(event)
|
|
{
|
|
if(!maybe(() => event.content.user_ids))
|
|
return;
|
|
|
|
this.typing.update(event.content.user_ids);
|
|
};
|
|
|
|
room.sync.handle["ephemeral"]["m.receipt"] = function(event)
|
|
{
|
|
this.receipt.update(event.content);
|
|
};
|
|
|
|
room.sync.handle["ephemeral"]["m.presence"] = function(event)
|
|
{
|
|
debug.object({presence: event});
|
|
};
|