2021-07-02 11:42:24 +02:00
|
|
|
/**
|
|
|
|
* EGroupware - Home - Javascript UI
|
|
|
|
*
|
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @package home
|
|
|
|
* @author Nathan Gray
|
|
|
|
* @copyright (c) 2013 Nathan Gray
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
*/
|
|
|
|
|
|
|
|
import {et2_createWidget} from "../../api/js/etemplate/et2_core_widget";
|
2023-03-02 21:27:35 +01:00
|
|
|
import {EgwApp} from "../../api/js/jsapi/egw_app";
|
|
|
|
import {etemplate2} from "../../api/js/etemplate/etemplate2";
|
|
|
|
import {Et2Portlet} from "../../api/js/etemplate/Et2Portlet/Et2Portlet";
|
|
|
|
import {Et2PortletFavorite} from "./Et2PortletFavorite";
|
2023-03-03 23:41:56 +01:00
|
|
|
import {loadWebComponent} from "../../api/js/etemplate/Et2Widget/Et2Widget";
|
|
|
|
import "./Et2PortletList";
|
|
|
|
import Sortable from "sortablejs/modular/sortable.complete.esm.js";
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* JS for home application
|
|
|
|
*
|
|
|
|
* Home is a collection of little bits of content (portlets) from the other applications.
|
|
|
|
*
|
|
|
|
* Uses Gridster for the grid layout
|
|
|
|
* @see http://gridster.net
|
|
|
|
* @augments AppJS
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
export class HomeApp extends EgwApp
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Grid resolution. Must match et2_portlet GRID
|
|
|
|
*/
|
2023-03-03 23:41:56 +01:00
|
|
|
public static GRID = 150;
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Default size for new portlets
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
public static DEFAULT = {
|
|
|
|
WIDTH: 4,
|
|
|
|
HEIGHT: 1
|
|
|
|
};
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// List of portlets
|
2023-03-02 21:27:35 +01:00
|
|
|
private portlets = {};
|
|
|
|
portlet_container : any;
|
2023-03-03 23:41:56 +01:00
|
|
|
private sortable : Sortable;
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
constructor()
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
// call parent
|
2023-03-02 21:27:35 +01:00
|
|
|
super("home");
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Destructor
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
destroy()
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
delete this.et2;
|
|
|
|
delete this.portlet_container;
|
|
|
|
|
|
|
|
this.portlets = {};
|
|
|
|
|
|
|
|
// call parent
|
2023-03-02 21:27:35 +01:00
|
|
|
super.destroy(this.appname);
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Make sure all other sub-etemplates in portlets are done
|
2023-03-03 23:41:56 +01:00
|
|
|
if(this == window.app.home)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
let others = etemplate2.getByApplication(this.appname);
|
|
|
|
for(let i = 0; i < others.length; i++)
|
|
|
|
{
|
|
|
|
others[i].clear();
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This function is called when the etemplate2 object is loaded
|
|
|
|
* and ready. If you must store a reference to the et2 object,
|
|
|
|
* make sure to clean it up in destroy().
|
|
|
|
*
|
|
|
|
* @param {etemplate2} et2 Newly ready object
|
|
|
|
* @param {string} Template name
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
et2_ready(et2, name)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
// Top level
|
|
|
|
if(name == 'home.index')
|
|
|
|
{
|
|
|
|
// call parent
|
2023-03-02 21:27:35 +01:00
|
|
|
super.et2_ready(et2, name);
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
this.et2.set_id('home.index');
|
|
|
|
this.et2.set_actions(this.et2.getArrayMgr('modifications').getEntry('home.index')['actions']);
|
|
|
|
|
|
|
|
this.portlet_container = this.et2.getWidgetById("portlets");
|
|
|
|
|
|
|
|
// Accept drops of favorites, which aren't part of action system
|
2023-03-03 23:41:56 +01:00
|
|
|
this.sortable = new Sortable(this.et2.getDOMNode().parentNode, {
|
|
|
|
chosenClass: 'drop-hover',
|
2023-03-02 21:27:35 +01:00
|
|
|
accept: function(draggable)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
// Check for direct support for that application
|
|
|
|
if(draggable[0].dataset && draggable[0].dataset.appname)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
return egw_getActionManager('home', false, 1).getActionById('drop_' + draggable[0].dataset.appname + '_favorite_portlet') != null;
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
2023-03-03 23:41:56 +01:00
|
|
|
onAdd: function(event, ui)
|
2023-03-02 21:27:35 +01:00
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
debugger;
|
2021-07-02 11:42:24 +02:00
|
|
|
// Favorite dropped on home - fake an action and divert to normal handler
|
2023-03-02 21:27:35 +01:00
|
|
|
let action = {
|
2021-07-02 11:42:24 +02:00
|
|
|
data: {
|
|
|
|
class: 'add_home_favorite_portlet'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for direct support for that application
|
|
|
|
if(ui.helper.context.dataset && ui.helper.context.dataset.appname)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
action = egw_getActionManager('home', false, 1).getActionById('drop_' + ui.helper.context.dataset.appname + '_favorite_portlet') || {}
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
action.ui = ui;
|
|
|
|
app.home.add_from_drop(action, [{data: ui.helper.context.dataset}])
|
|
|
|
}
|
2023-03-03 23:41:56 +01:00
|
|
|
});
|
2023-03-02 21:27:35 +01:00
|
|
|
// Bind to unload to remove it from our list
|
2023-03-03 23:41:56 +01:00
|
|
|
/*
|
2023-03-02 21:27:35 +01:00
|
|
|
.on('clear', '.et2_container[id]', jQuery.proxy(function(e)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
if(e.target && e.target.id && this.portlets[e.target.id])
|
|
|
|
{
|
|
|
|
this.portlets[e.target.id].destroy();
|
|
|
|
delete this.portlets[e.target.id];
|
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
}, this));
|
2023-03-03 23:41:56 +01:00
|
|
|
|
|
|
|
*/
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
else if(et2.uniqueId)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
let portlet_container = this.portlet_container || window.app.home?.portlet_container;
|
|
|
|
// Handle bad timing - a sub-template was finished first
|
|
|
|
if(!portlet_container)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
window.setTimeout(() => {this.et2_ready(et2, name);}, 200);
|
2021-07-02 11:42:24 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-03-03 23:41:56 +01:00
|
|
|
let portlet = portlet_container.getWidgetById(et2.uniqueId) || et2.DOMContainer;
|
2021-07-02 11:42:24 +02:00
|
|
|
// Check for existing etemplate, this one loaded over it
|
|
|
|
// NOTE: Moving them around like this can cause problems with event handlers
|
2023-03-02 21:27:35 +01:00
|
|
|
let existing = etemplate2.getById(et2.uniqueId);
|
2021-07-02 11:42:24 +02:00
|
|
|
if(portlet && existing)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
for(let i = 0; i < portlet._children.length; i++)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
if(typeof portlet._children[i]._init == 'undefined')
|
|
|
|
{
|
|
|
|
portlet.removeChild(portlet._children[i])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
// Set size & position
|
2023-03-03 23:41:56 +01:00
|
|
|
let et2_data = et2.widgetContainer.getArrayMgr("content").data;
|
|
|
|
let settings = et2_data && et2_data.id == portlet.id && et2_data || portlet_container.getArrayMgr("content").data.find(e => et2.uniqueId.endsWith(e.id)) || {settings: {}};
|
|
|
|
portlet.settings = settings.settings || {};
|
2023-03-02 21:27:35 +01:00
|
|
|
portlet.style.gridArea = settings.row + "/" + settings.col + "/ span " + (settings.height || 1) + "/ span " + (settings.width || 1);
|
2023-03-03 23:41:56 +01:00
|
|
|
|
|
|
|
|
2021-07-02 11:42:24 +02:00
|
|
|
// It's in the right place for original load, but move it into portlet
|
2023-03-03 23:41:56 +01:00
|
|
|
|
2023-03-02 21:27:35 +01:00
|
|
|
let misplaced = jQuery(etemplate2.getById('home-index').DOMContainer).siblings('#' + et2.DOMContainer.id);
|
2023-03-03 23:41:56 +01:00
|
|
|
|
|
|
|
if(portlet && et2.DOMContainer !== portlet)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
portlet.append(et2.DOMContainer);
|
2021-07-02 11:42:24 +02:00
|
|
|
et2.resize();
|
|
|
|
}
|
|
|
|
if(portlet && misplaced.length)
|
|
|
|
{
|
|
|
|
et2.DOMContainer.id = et2.uniqueId;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Instanciate custom code for this portlet
|
|
|
|
this._get_portlet_code(portlet);
|
2023-03-03 23:41:56 +01:00
|
|
|
|
|
|
|
// Ordering of portlets
|
|
|
|
// Only needs to be done once, but its hard to tell when everything is loaded
|
|
|
|
this._do_ordering();
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Observer method receives update notifications from all applications
|
|
|
|
*
|
|
|
|
* Home passes the notification off to specific code for each portlet, which
|
|
|
|
* decide if they should be updated or not.
|
|
|
|
*
|
|
|
|
* @param {string} _msg message (already translated) to show, eg. 'Entry deleted'
|
|
|
|
* @param {string} _app application name
|
|
|
|
* @param {(string|number)} _id id of entry to refresh or null
|
|
|
|
* @param {string} _type either 'update', 'edit', 'delete', 'add' or null
|
|
|
|
* - update: request just modified data from given rows. Sorting is not considered,
|
|
|
|
* so if the sort field is changed, the row will not be moved.
|
|
|
|
* - edit: rows changed, but sorting may be affected. Requires full reload.
|
|
|
|
* - delete: just delete the given rows clientside (no server interaction neccessary)
|
|
|
|
* - add: requires full reload for proper sorting
|
|
|
|
* @param {string} _msg_type 'error', 'warning' or 'success' (default)
|
|
|
|
* @param {string} _targetapp which app's window should be refreshed, default current
|
|
|
|
* @return {false|*} false to stop regular refresh, thought all observers are run
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
observer(_msg, _app, _id, _type, _msg_type, _targetapp)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
for(let id in this.portlets)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
// App is home, refresh all portlets
|
|
|
|
if(_app == 'home')
|
|
|
|
{
|
|
|
|
this.refresh(id);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ask the portlets if they're interested
|
|
|
|
try
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let code = this.portlets[id];
|
2021-07-02 11:42:24 +02:00
|
|
|
if(code)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
code.observer(_msg, _app, _id, _type, _msg_type, _targetapp);
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
this.egw.debug("error", "Error trying to update portlet " + id, e);
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a new portlet from the context menu
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
add(action, source)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
// Basic portlet attributes
|
2023-03-02 21:27:35 +01:00
|
|
|
let attrs = {
|
|
|
|
...HomeApp.DEFAULT, ...{
|
|
|
|
id: this._create_id(),
|
|
|
|
class: action.data.class
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// Try to put it about where the menu was opened
|
|
|
|
if(action.menu_context)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let $portlet_container = jQuery(this.portlet_container.getDOMNode());
|
|
|
|
attrs.row = Math.max(1, Math.round((action.menu_context.posy - $portlet_container.offset().top) / HomeApp.GRID) + 1);
|
2023-03-03 23:41:56 +01:00
|
|
|
// Use "auto" col to avoid any overlap or overflow
|
|
|
|
attrs.col = "auto";
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
|
2023-03-03 23:41:56 +01:00
|
|
|
let portlet = <Et2Portlet>loadWebComponent('et2-portlet', attrs, this.portlet_container);
|
2021-07-02 11:42:24 +02:00
|
|
|
portlet.loadingFinished();
|
|
|
|
|
|
|
|
// Get actual attributes & settings, since they're not available client side yet
|
2023-03-02 21:27:35 +01:00
|
|
|
portlet.update_settings(attrs);
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Instanciate custom code for this portlet
|
|
|
|
this._get_portlet_code(portlet);
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* User dropped something on home. Add a new portlet
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
add_from_drop(action, source)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Actions got confused drop vs popup
|
|
|
|
if(source[0].id == 'portlets')
|
|
|
|
{
|
|
|
|
return this.add(action);
|
|
|
|
}
|
|
|
|
|
2023-03-02 21:27:35 +01:00
|
|
|
let $portlet_container = jQuery(this.portlet_container.getDOMNode());
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Basic portlet attributes
|
2023-03-02 21:27:35 +01:00
|
|
|
let attrs = {
|
2023-03-03 23:41:56 +01:00
|
|
|
...HomeApp.DEFAULT,
|
2021-07-02 11:42:24 +02:00
|
|
|
id: this._create_id(),
|
|
|
|
class: action.data.class || action.id.substr(5),
|
2023-03-03 23:41:56 +01:00
|
|
|
dropped_data: []
|
2021-07-02 11:42:24 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// Try to find where the drop was
|
|
|
|
if(action != null && action.ui && action.ui.position)
|
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
attrs.row = Math.max(1, Math.round((action.ui.position.top - $portlet_container.offset().top) / HomeApp.GRID));
|
|
|
|
// Use "auto" col to avoid any overlap or overflow
|
|
|
|
attrs.col = "auto";
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get actual attributes & settings, since they're not available client side yet
|
2023-03-02 21:27:35 +01:00
|
|
|
for(let i = 0; i < source.length; i++)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
if(source[i].id)
|
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
attrs.dropped_data.push(source[i].id);
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
attrs.dropped_data.push(source[i].data);
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-03 23:41:56 +01:00
|
|
|
|
|
|
|
let portlet = <Et2Portlet>loadWebComponent('et2-portlet', attrs, this.portlet_container);
|
|
|
|
portlet.loadingFinished();
|
|
|
|
|
|
|
|
// Get actual attributes & settings, since they're not available client side yet
|
|
|
|
portlet.update_settings(attrs);
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Instanciate custom code for this portlet
|
|
|
|
this._get_portlet_code(portlet);
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the current selection as default for other users
|
|
|
|
*
|
|
|
|
* Only works (and available) for admins, this shows a dialog to select
|
|
|
|
* the group, and then sets the default for that group.
|
|
|
|
*
|
|
|
|
* @param {egwAction} action
|
|
|
|
* @param {egwActionObject[]} selected
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
set_default(action, selected)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
// Gather just IDs, server will handle the details
|
2023-03-02 21:27:35 +01:00
|
|
|
let portlet_ids = [];
|
|
|
|
let group = action.data.portlet_group || false;
|
2021-07-02 11:42:24 +02:00
|
|
|
if(selected[0].id == 'home.index')
|
|
|
|
{
|
|
|
|
// Set all
|
2023-03-02 21:27:35 +01:00
|
|
|
this.portlet_container.iterateOver(function(portlet)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
portlet_ids.push(portlet.id);
|
2023-03-02 21:27:35 +01:00
|
|
|
}, this, et2_portlet);
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
for(let i = 0; i < selected.length; i++)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
portlet_ids.push(selected[i].id);
|
|
|
|
|
|
|
|
// Read the associated group so we can properly remove it
|
2023-03-02 21:27:35 +01:00
|
|
|
let portlet = egw.preference(selected[i].id, 'home');
|
2021-07-02 11:42:24 +02:00
|
|
|
if(!group && portlet && portlet.group)
|
|
|
|
{
|
|
|
|
group = portlet.group;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(action.id.indexOf("remove_default") == 0)
|
|
|
|
{
|
|
|
|
// Disable action for feedback
|
|
|
|
action.set_enabled(false);
|
|
|
|
|
|
|
|
// Pass them to server
|
|
|
|
egw.json('home_ui::ajax_set_default', ['delete', portlet_ids, group]).sendRequest(true);
|
|
|
|
return;
|
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
let dialog = et2_createWidget("dialog", {
|
2021-07-02 11:42:24 +02:00
|
|
|
// If you use a template, the second parameter will be the value of the template, as if it were submitted.
|
2023-03-02 21:27:35 +01:00
|
|
|
callback: function(button_id, value)
|
|
|
|
{
|
|
|
|
if(button_id != et2_dialog.OK_BUTTON)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Pass them to server
|
2023-03-02 21:27:35 +01:00
|
|
|
egw.json('home_ui::ajax_set_default', ['add', portlet_ids, value.group || false]).sendRequest(true);
|
2021-07-02 11:42:24 +02:00
|
|
|
},
|
|
|
|
buttons: et2_dialog.BUTTONS_OK_CANCEL,
|
|
|
|
title: action.caption,
|
2023-03-02 21:27:35 +01:00
|
|
|
template: "home.set_default",
|
|
|
|
value: {content: {}, sel_options: {group: {default: egw.lang('All'), forced: egw.lang('Forced')}}}
|
2021-07-02 11:42:24 +02:00
|
|
|
});
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Allow a refresh from anywhere by triggering an update with no changes
|
|
|
|
*
|
|
|
|
* @param {string} id
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
refresh(id)
|
|
|
|
{
|
|
|
|
let p = this.portlet_container.getWidgetById(id);
|
2021-07-02 11:42:24 +02:00
|
|
|
if(p)
|
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
p.update_settings('~reload~');
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine the best fitting code to use for the given portlet, instanciate
|
|
|
|
* it and add it to the list.
|
|
|
|
*
|
|
|
|
* @param {et2_portlet} portlet
|
|
|
|
* @returns {home_portlet}
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
_get_portlet_code(portlet)
|
|
|
|
{
|
|
|
|
let classname = portlet.class;
|
2021-07-02 11:42:24 +02:00
|
|
|
// Freshly added portlets can have 'add_' prefix
|
|
|
|
if(classname.indexOf('add_') == 0)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
classname = classname.replace('add_', '');
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
// Prefer a specific match
|
2023-03-02 21:27:35 +01:00
|
|
|
let _class = app.classes.home[classname] ||
|
|
|
|
(typeof customElements.get(classname) != "undefined" ? customElements.get(classname).class : false) ||
|
2021-07-02 11:42:24 +02:00
|
|
|
// If it has a nextmatch, use favorite base class
|
2023-03-02 21:27:35 +01:00
|
|
|
(portlet.getWidgetById('nm') ? Et2PortletFavorite : false) ||
|
2021-07-02 11:42:24 +02:00
|
|
|
// Fall back to base class
|
2023-03-02 21:27:35 +01:00
|
|
|
Et2Portlet;
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
this.portlets[portlet.id] = new _class(portlet);
|
|
|
|
|
|
|
|
return this.portlets[portlet.id];
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* For link_portlet - opens the configured record when the user
|
|
|
|
* double-clicks or chooses view from the context menu
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
open_link(action)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Get widget
|
2023-03-02 21:27:35 +01:00
|
|
|
let widget = null;
|
2021-07-02 11:42:24 +02:00
|
|
|
while(action.parent != null)
|
|
|
|
{
|
|
|
|
if(action.data && action.data.widget)
|
|
|
|
{
|
|
|
|
widget = action.data.widget;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
action = action.parent;
|
|
|
|
}
|
|
|
|
if(widget == null)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
this.egw.log("warning", "Could not find widget");
|
2021-07-02 11:42:24 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
this.egw.open(widget.options.settings.entry, "", 'view', null, widget.options.settings.entry.app);
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Set up the drag / drop / re-order of portlets
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
_do_ordering()
|
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
if(!this.portlet_container)
|
2023-03-02 21:27:35 +01:00
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let col_map = {};
|
|
|
|
this.portlet_container.getDOMNode().querySelectorAll("[style*='grid-area']").forEach((n) =>
|
2023-03-02 21:27:35 +01:00
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
let [col, span] = (getComputedStyle(n).gridColumn || "").split(" / ");
|
|
|
|
if(typeof col_map[col] !== "undefined")
|
2023-03-02 21:27:35 +01:00
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
// Set column to auto to avoid overlap
|
|
|
|
n.style.gridColumn = "auto / " + span;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
col_map[col] = true;
|
|
|
|
}
|
|
|
|
});
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an ID that should be unique, at least amoung a single user's portlets
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
_create_id()
|
|
|
|
{
|
|
|
|
let id = '';
|
2021-07-02 11:42:24 +02:00
|
|
|
do
|
|
|
|
{
|
|
|
|
id = Math.floor((1 + Math.random()) * 0x10000)
|
2023-03-02 21:27:35 +01:00
|
|
|
.toString(16)
|
|
|
|
.substring(1);
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
while(this.portlet_container.getWidgetById('portlet_' + id));
|
|
|
|
return 'portlet_' + id;
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Functions for the list portlet
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Remove a link from the list
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
link_change(list, link_id, row)
|
|
|
|
{
|
2023-03-03 23:41:56 +01:00
|
|
|
list.link_change(link_id, row);
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Functions for the note portlet
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Set up for editing a note
|
|
|
|
* CKEditor has CSP issues, so we need a popup
|
|
|
|
*
|
|
|
|
* @param {egwAction} action
|
|
|
|
* @param {egwActionObject[]} Selected
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
note_edit(action, selected)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
if(!selected && typeof action == 'string')
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let id = action;
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let id = selected[0].id;
|
2021-07-02 11:42:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Aim to match the size
|
2023-03-02 21:27:35 +01:00
|
|
|
let portlet_dom = jQuery('[id$=' + id + '][data-sizex]', this.portlet_container.getDOMNode());
|
|
|
|
let width = portlet_dom.attr('data-sizex') * this.GRID;
|
|
|
|
let height = portlet_dom.attr('data-sizey') * this.GRID;
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// CKEditor is impossible to use below a certain size
|
|
|
|
// Add 35px for the toolbar, 35px for the buttons
|
2023-03-02 21:27:35 +01:00
|
|
|
let window_width = Math.max(580, width + 20);
|
|
|
|
let window_height = Math.max(350, height + 70);
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Open popup, but add 70 to the height for the toolbar
|
2023-03-02 21:27:35 +01:00
|
|
|
this.egw.open_link(this.egw.link('/index.php', {
|
2021-07-02 11:42:24 +02:00
|
|
|
menuaction: 'home.home_note_portlet.edit',
|
|
|
|
id: id,
|
|
|
|
height: window_height - 70
|
2023-03-02 21:27:35 +01:00
|
|
|
}), 'home_' + id, window_width + 'x' + window_height, 'home');
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Favorites / nextmatch
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Toggle the nextmatch header shown / hidden
|
|
|
|
*
|
|
|
|
* @param {Event} event
|
|
|
|
* @param {et2_button} widget
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
nextmatch_toggle_header(event, widget)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
widget.set_class(widget.class == 'opened' ? 'closed' : 'opened');
|
|
|
|
// We operate on the DOM here, nm should be unaware of our fiddling
|
2023-03-02 21:27:35 +01:00
|
|
|
let nm = widget.getParent().getWidgetById('nm');
|
|
|
|
if(!nm)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
// Hide header
|
|
|
|
nm.div.toggleClass('header_hidden');
|
|
|
|
nm.set_hide_header(nm.div.hasClass('header_hidden'));
|
|
|
|
nm.resize();
|
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
app.classes.home = HomeApp;
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/// Base class code
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base class for portlet specific javascript
|
|
|
|
*
|
|
|
|
* Should this maybe extend et2_portlet? It would complicate instantiation.
|
|
|
|
*
|
|
|
|
* @type @exp;Class@call;extend
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
export class HomePortlet
|
|
|
|
{
|
|
|
|
protected portlet = null;
|
2021-07-02 11:42:24 +02:00
|
|
|
|
2023-03-02 21:27:35 +01:00
|
|
|
init(portlet)
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
this.portlet = portlet;
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
destroy()
|
|
|
|
{
|
2021-07-02 11:42:24 +02:00
|
|
|
this.portlet = null;
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle framework refresh messages to determine if the portlet needs to
|
|
|
|
* refresh too.
|
|
|
|
*
|
|
|
|
* App is responsible for only reacting to "messages" it is interested in!
|
|
|
|
*
|
|
|
|
*/
|
2023-03-02 21:27:35 +01:00
|
|
|
observer(_msg, _app, _id, _type, _msg_type, _targetapp)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
// Not interested
|
|
|
|
}
|
2023-03-02 21:27:35 +01:00
|
|
|
}
|
2021-07-02 11:42:24 +02:00
|
|
|
|
2023-03-02 21:27:35 +01:00
|
|
|
/*
|
2021-07-02 11:42:24 +02:00
|
|
|
app.classes.home.home_link_portlet = app.classes.home.home_portlet.extend({
|
|
|
|
init: function(portlet) {
|
|
|
|
// call parent
|
|
|
|
this._super.apply(this, arguments);
|
|
|
|
|
|
|
|
// Check for tooltip
|
|
|
|
if(this.portlet)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let content = jQuery('.tooltip', this.portlet.content);
|
2021-07-02 11:42:24 +02:00
|
|
|
if(content.length && content.children().length)
|
|
|
|
{
|
|
|
|
//Check if the tooltip is already initialized
|
|
|
|
this.portlet.content.tooltip({
|
|
|
|
items: this.portlet.content,
|
|
|
|
content: content.html(),
|
|
|
|
tooltipClass: 'portlet_' + this.portlet.id,
|
|
|
|
show: {effect: 'slideDown', delay:500},
|
|
|
|
hide: {effect: 'slideUp', delay: 500},
|
|
|
|
position: {my: "left top", at:"left bottom", collision: "flipfit"},
|
|
|
|
open: jQuery.proxy(function(event, ui) {
|
|
|
|
// Calendar specific formatting
|
|
|
|
if(ui.tooltip.has('.calendar_calEventTooltip').length)
|
|
|
|
{
|
|
|
|
ui.tooltip.removeClass("ui-tooltip");
|
|
|
|
ui.tooltip.addClass("calendar_uitooltip");
|
|
|
|
}
|
|
|
|
},this),
|
|
|
|
close: function(event,ui) {
|
|
|
|
ui.tooltip.hover(
|
|
|
|
function() {
|
|
|
|
jQuery(this).stop(true).fadeTo(100,1);
|
|
|
|
},
|
|
|
|
function() {
|
|
|
|
jQuery(this).slideUp("400",function() {jQuery(this).remove();});
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
observer: function(_msg, _app, _id, _type)
|
|
|
|
{
|
|
|
|
if(this.portlet && this.portlet.settings)
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let value = this.portlet.settings.entry || {};
|
2021-07-02 11:42:24 +02:00
|
|
|
if(value.app && value.app == _app && value.id && value.id == _id)
|
|
|
|
{
|
|
|
|
// We don't just get the updated title, in case there's a custom
|
|
|
|
// template with more fields
|
|
|
|
app.home.refresh(this.portlet.id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
app.classes.home.home_list_portlet = app.classes.home.home_portlet.extend({
|
|
|
|
observer: function(_msg, _app, _id, _type)
|
|
|
|
{
|
|
|
|
if(this.portlet && this.portlet.getWidgetById('list'))
|
|
|
|
{
|
2023-03-02 21:27:35 +01:00
|
|
|
let list = this.portlet.getWidgetById('list').options.value;
|
|
|
|
for(let i = 0; i < list.length; i++)
|
2021-07-02 11:42:24 +02:00
|
|
|
{
|
|
|
|
if(list[i].app == _app && list[i].id == _id)
|
|
|
|
{
|
|
|
|
app.home.refresh(this.portlet.id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
app.classes.home.home_weather_portlet = app.classes.home.home_portlet.extend({
|
|
|
|
init: function(portlet) {
|
|
|
|
// call parent
|
|
|
|
this._super.apply(this, arguments);
|
|
|
|
|
|
|
|
// Use location API
|
|
|
|
if(!this.portlet.options.settings && 'geolocation' in navigator)
|
|
|
|
{
|
|
|
|
navigator.geolocation.getCurrentPosition(function(position) {
|
|
|
|
if(portlet && portlet.options && portlet.options.settings &&
|
|
|
|
portlet.options.settings.position && portlet.options.settings.position == position.coords.latitude + ',' + position.coords.longitude)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
portlet._process_edit(et2_dialog.OK_BUTTON, {position: position.coords.latitude + ',' + position.coords.longitude});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
app.classes.home.home_favorite_portlet = app.classes.home.home_portlet.extend({
|
|
|
|
init: function(portlet) {
|
|
|
|
// call parent
|
|
|
|
this._super.apply(this, arguments);
|
|
|
|
|
|
|
|
// Somehow favorite got lost, or is not set
|
|
|
|
if(portlet.options && portlet.options.settings && typeof portlet.options.settings !== 'undefined' &&
|
|
|
|
!portlet.options.settings.favorite
|
|
|
|
)
|
|
|
|
{
|
|
|
|
portlet.edit_settings();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
observer: function(_msg, _app, _id, _type, _msg_type, _targetapp)
|
|
|
|
{
|
|
|
|
if(this.portlet.class.indexOf(_app) == 0 || this.portlet.class == 'home_favorite_portlet')
|
|
|
|
{
|
|
|
|
this.portlet.getWidgetById('nm').refresh(_id,_type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An example illustrating extending the base code for a application specific code.
|
|
|
|
* See also the calendar app, which needs custom handlers
|
|
|
|
*
|
|
|
|
* @type @exp;app@pro;classes@pro;home@pro;home_favorite_portlet@call;extend
|
|
|
|
* Note we put it in home, but this code should go in addressbook/js/addressbook_favorite_portlet.js
|
|
|
|
*
|
|
|
|
app.classes.home.addressbook_favorite_portlet = app.classes.home.home_favorite_portlet.extend({
|
|
|
|
|
|
|
|
observer: function(_msg, _app, _id, _type, _msg_type, _targetapp)
|
|
|
|
{
|
|
|
|
// Just checking...
|
|
|
|
debugger;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
*/
|