diff --git a/__manifest__.py b/__manifest__.py
index c5cbd7a..e46a494 100755
--- a/__manifest__.py
+++ b/__manifest__.py
@@ -28,9 +28,13 @@
'views/dss_texts.xml',
'views/dss_systems.xml',
'views/dss_screendesign.xml',
- 'views/dss_trigger_actions.xml',
'views/dss_eventdays.xml',
+ 'views/dss_trigger_actions.xml',
'views/dss_triggertypes.xml',
+ 'views/dss_trigger_groups.xml',
+ 'views/dss_maintains.xml',
+ 'views/dss_provisionstypen.xml',
+ 'views/dss_provision.xml',
'views/mainsystem_view.xml',
'views/menu.xml',
'views/company_view.xml',
diff --git a/getimg.sh b/getimg.sh
index 3b6cb2e..076f3e7 100755
--- a/getimg.sh
+++ b/getimg.sh
@@ -1 +1 @@
-curl https://upload.diebesorger.com/f.php?h=1DWnJR2Q&d=1
\ No newline at end of file
+https://apps.odoo.com/apps/modules/16.0/master_search
\ No newline at end of file
diff --git a/models/__init__.py b/models/__init__.py
index 6e17e15..23c3689 100755
--- a/models/__init__.py
+++ b/models/__init__.py
@@ -1,4 +1,5 @@
from . import dss_settings
+from . import dss_trigger
from . import dss_contract
from . import dss_projects
from . import dss_geraetetypen
@@ -7,8 +8,12 @@ from . import dss_software
from . import dss_systems
from . import dss_eventdays
from . import dss_advertisefields
-from . import dss_ads
+from . import dss_trigger
+from . import dss_triggervalues
from . import dss
+from . import dss_maintains
+from . import dss_provisionstypen
+from . import dss_provision
from . import company
diff --git a/models/company.py b/models/company.py
index 8e52005..4b77965 100755
--- a/models/company.py
+++ b/models/company.py
@@ -24,6 +24,9 @@ class ResCompany(models.Model):
dssprojekte = fields.Many2many('dss.main', readonly=1 )
dsspartner_name = fields.Char('Kundenname', default=False)
dsspartner_vorname = fields.Char('KundenVorname', default=False)
+ dssinternpartner = fields.Boolean('Mitarbeiter', default=False)
+ dssinternpartner_grafik = fields.Boolean('Grafiker', default=False)
+ dssinternpartner_technik = fields.Boolean('Techniker', default=False)
# dss_uuid = fields.Char('uuid')
dss_uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
diff --git a/models/dss.py b/models/dss.py
index f78bfc3..c2c306c 100755
--- a/models/dss.py
+++ b/models/dss.py
@@ -9,6 +9,7 @@ import logging
import base64
import subprocess
import tempfile
+
import easywebdav
import os
import os.path
@@ -365,24 +366,124 @@ class dssscreendesign(models.Model):
outlist = {'uuid' : 2222}
return outlist
-class dsstriggertypes(models.Model):
- _name = "dss.triggertypes"
- _description = "DigitalSignage Trigger Typen"
- _inherit = ['mail.thread','mail.activity.mixin']
- _rec_name = "triggername"
- uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
- date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
- date_lastedit = fields.Date('Änderungsdatum')
- user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
- user_lastedit = fields.Char('Änderungsuser')
+class dsscontractads(models.Model):
- triggername = fields.Char('Triggername',track_visibility='onchange',tracking=True)
- triggertyp = fields.Selection([('FIELD_A','Feldänderung allgemein'),('FIELD_S','Feldänderung spezifisch'),('FIELD_K','Feldwert berechnet'),('MANUAL','Menuell ausgelösst')],'Triggertyp',track_visibility='onchange',tracking=True)
- trigger_table = fields.Many2one('ir.model','Nutzbar in',help='Für welche Daten ist dieser Tigger gültig ?',track_visibility='onchange',tracking=True)
- trigger_field = fields.Many2one('ir.model.fields', help='Feldname in der Tabelle',track_visibility='onchange',tracking=True)
- trigger_operator = fields.Selection([('NGLEICH','Feld gleich dem Wert'),('GGLEICH','Feld größer/gleich dem Wert'),('KGLEICH','Feld kleiner/gleich dem Wert'),('UGLEICH','Feld ungleich dem Wert'),('ENTHAELT','Feld enthällt dem Wert')],'Feldwertoperator',track_visibility='onchange',tracking=True)
- trigger_value = fields.Char('Feldwert für Trigger',track_visibility='onchange',tracking=True)
- trigger_Aktionen = fields.Many2many('dss.triggeractions',string='Aktionen bei auslösen des Triggers',track_visibility='onchange',tracking=True)
+ def _default_work_state(self):
+ ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).id
+ _logger.debug(ds)
+ return ds
+
+ def _default_work_state_color(self):
+ ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).color
+ _logger.info(ds)
+ return ds
+
+ def _default_todo_state(self):
+ ds = self.env['dss.todostate'].search([('statusnr', '=', '0')], limit=1).id
+ if not ds: ds = 1
+ _logger.debug(ds)
+ # ds =
+ return ds
+
+ @api.model
+ def create(self, vals):
+ result = super().create(vals)
+ resstr = result.uuid
+ _logger.info(resstr)
+ for n_record in result:
+ resstr = n_record.uuid
+ _logger.info(resstr)
+ n_record.date_create = date.today()
+ n_record.user_create = self.env.user.name
+ allads = n_record.contract.ads
+ _logger.info(allads)
+ if allads != False:
+ for ad in allads:
+ ad.ad_is_lastpos = False
+ n_record.ad_is_lastpos = True
+ contract = n_record.contract
+ contract.ads_last_ad = n_record
+ return result
+
+ # @api.depends('need_media')
+ # def _getmedialist(self):
+ # mlist = []
+ # mlist = self.env['dss.mediarelations'].search([('ad','=',self.id)])
+ # if mlist != False:
+ # _logger.info('AD Need_Medias A_'+str(self)+' - Computed Medias '+str(mlist))
+ # self.need_media = mlist.ids
+ # else:
+ # self.need_media = False
+
+ _name = "dss.ads"
+ _description = "DigitalSignage Werbekampagnen Phasen"
+ _inherit = ['mail.thread', 'mail.activity.mixin']
+ # _inherit = ['mail.activity.mixin']
+ # 'mail.thread','mail.activity.mixin'
+ _rec_name = "adname"
+ # _inherit = ['mail.thread', 'mail.activity.mixin']
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False,
+ string='UUID')
+ aduuid = fields.Char(related='uuid')
+
+ date_create = fields.Date('Erstellungsdatum', default=lambda self: self._default_create_date())
+ date_lastedit = fields.Date('Änderungsdatum')
+ user_create = fields.Char('Erstellungsuser', default=lambda self: self._default_create_user())
+ user_lastedit = fields.Char('Änderungsuser')
+ # uuid = fields.Char('UUID', required=True, translate=True)
+ adname = fields.Char('Kampagnenname', required=True, tracking=True)
+ adtype = fields.Selection(
+ [('MAIN', 'Ersteinrichtung'), ('KCHN', 'Änderung durch Kunde'), ('LCHN', 'Änderung durch Logumedia'),
+ ('SONS', 'Sonstige Änderung')], tracking=True)
+ adpos = fields.Integer('Reihenfolge', default=lambda self: self._default_adpos(),
+ tracking=True)
+ ad_is_lastpos = fields.Boolean('Letzte Aktion', tracking=True)
+
+ start_date = fields.Date('gew. Ausstrahl.Begin', tracking=True)
+ end_date = fields.Date('gew. Ausstrahl.Ende', tracking=True)
+
+ contract = fields.Many2one('dss.contracts', store=True, tracking=True)
+ contract_id = fields.Char(related='contract.contract_id')
+ contract_name = fields.Char(related='contract.contract_name')
+ contract_need_media = fields.Many2many(related='contract.need_media', tracking=True)
+
+ project = fields.Many2one('dss.projects', string='Project', store=True, tracking=True)
+ project_id = fields.Integer(related='project.projectid', string='Project ID')
+
+ parent_ad = fields.Many2one('dss.ads', string='UrsprungsWerbung', store=True)
+ parent_ad_uuid = fields.Char(related='parent_ad.uuid')
+ description = fields.Text('Beschreibung')
+ amount = fields.Float('Extrakosten', tracking=True)
+ order = fields.Integer('Reihenfolge', tracking=True)
+
+ # need_media = fields.Many2many('dss.mediarelations','ad',string='Medien')
+ need_media = fields.One2many('dss.mediarelations', 'ad', string='Medien')
+
+ work_state = fields.Many2one('dss.workstate', default=_default_work_state, tracking=True)
+ work_state_color = fields.Char(related='work_state.color')
+ work_state_text = fields.Char(related='work_state.statusname')
+ work_state_info = fields.Char('Zusatzinfo')
+
+ ad_state = fields.Many2one('dss.adstate', tracking=True)
+ ad_state_color = fields.Char(related='ad_state.color')
+ ad_state_text = fields.Char(related='ad_state.statusname')
+ ad_state_func = fields.Selection(related='ad_state.func')
+
+ todo_state = fields.Many2one('dss.todostate', default=_default_todo_state,
+ tracking=True)
+ todo_state_color = fields.Char(related='todo_state.color')
+ todo_state_text = fields.Char(related='todo_state.statusname')
+ todo_state_info = fields.Char('Zusatzinfo', tracking=True)
+ todo_state_until = fields.Date('Abarbeiten bis', tracking=True)
+
+ cloud_add_directory = fields.Char('Cloud KundenKampagnen Ordner', tracking=True)
+
+ date_zuarbeit = fields.Date(string='Zuarbeit Datum', help='Zuarbeit gesendet am', tracking=True)
+ date_korrekturabzug = fields.Date(string='K.Abzug Datum', help='Korrekturabzug gesendet am')
+ date_korrekturfreigabe = fields.Date(string='K.Freigabe Datum', help='Korrekturfreigabe erhalten am',
+ tracking=True)
+ date_korrekturfreigabe_frist = fields.Date(string='K.Freigabe bis Datum', help='Korrekturfreigabe sollte bis .... erfolgen')
+ freigabe_durch_ablauf = fields.Boolean(string='Zeitablauf',help='Freigabe wurde durch Zeitablauf erreicht ?', tracking=True)
@api.model
def _default_uuid(self):
@@ -394,29 +495,320 @@ class dsstriggertypes(models.Model):
def _default_create_user(self):
return str(self.env.user.name)
-class dsstriggertypes(models.Model):
- _name = "dss.triggeractions"
- _description = "DigitalSignage Trigger Aktionen"
- _inherit = ['mail.thread','mail.activity.mixin']
- _rec_name = "triggeractionname"
- uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
- date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
+ def _default_adpos(self):
+ pos = self.env['dss.ads'].search_count([('contract_id', '=', self.contract.id)]) + 1
+ return str(pos)
+
+ @api.onchange('ad_state')
+ def _onchange_ad_state(self):
+ for record in self:
+ if record.ad_is_lastpos == True:
+ contract = record.contract
+ contract.ads_last_state = record.ad_state
+ # mtyp = self.env['dss.adstructures'].search([("uuid","=",str(record.mediastructure.uuid))])
+ # buildmediarelations(self)
+ self.date_lastedit = str(date.today())
+
+ # self.mediastructure = mtyp
+
+ def pyaction_view_ad_details(self):
+ action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
+ 'DigitalSignage.action_dss_ads_view')
+ # action['display_name'] = self.ad_name
+ # action['domain'] = '[["projectid","=","4"]]'
+ # context = action['context'].replace('', str(self.id))
+ # context = ast.literal_eval(context)
+ # context.update({
+ # 'create': self.active,
+ # 'active_test': self.active
+ # })
+ # action['context'] = context
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'form',
+ 'view_mode': 'form,tree',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '',
+ 'res_id': self.id,
+ 'display_name': ' ' + self.adname,
+ 'domain': ''
+ }
+
+ return action
+
+ def pydoviewallads(self):
+ action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
+ 'DigitalSignage.act_dss_ads_view_full')
+ action['display_name'] = self.contract_name
+ # action['domain'] = '[["projectid","=","4"]]'
+ # context = action['context'].replace('', str(self.id))
+ # context = ast.literal_eval(context)
+ # context.update({
+ # 'create': self.active,
+ # 'active_test': self.active
+ # })
+ # action['context'] = context
+ # return {
+ # 'type': 'ir.actions.act_window',
+ # 'view_type':'form',
+ # 'view_mode':'form,tree',
+ # 'res_model':'dss.ads',
+ # 'target':'current',
+ # 'context':'',
+ # 'res_id':kampagne.id,
+ # 'display_name' : 'Werbekampagne '+kampagne.adname,
+ # 'domain':'[("contract","=","context[active_id]")]'
+ # }
+ # return action
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'kanban',
+ 'view_mode': 'kanban',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '{"group_by":["parent_ad"]}',
+ # 'res_id':self.parent_ad,
+ 'display_name': 'Übersicht für ' + self.adname,
+ 'domain': [("contract", "=", self.contract.id)]
+ }
+
+ def pydonewad(self):
+ for n_record in self:
+ resstr = n_record.uuid
+ _logger.info(resstr)
+ allads = n_record.contract.ads
+ _logger.info(allads)
+ abort = False
+ if allads != False:
+ for ad in allads:
+ if ad.ad_state.func != 'FIN':
+ abort = True
+ if abort == False:
+ defadstate = self.env['dss.adstate'].search([('func', '=', 'STD')], limit=1).id
+ if not defadstate:
+ defadstate = 1
+ if self.ad_state.func == 'STD':
+ parent_id = self.id
+ else:
+ parent_id = self.parent_ad.id
+ newkampagne = self.env['dss.ads'].create({'contract': self.contract.id, 'project': self.project.id,
+ 'adname': 'AD_' + self.contract.contract_auto_name + ' ' + str(
+ date.today()), 'parent_ad': parent_id, 'adtype': 'KCHN',
+ 'ad_state': defadstate})
+ _logger.info('C_' + str(self.id) + ' Kampagne erzeugt : ' + str(newkampagne))
+ _logger.info('Prüfe Medien : C_' + str(self.id) + 'K_' + str(newkampagne.id))
+
+ mediaids = []
+ for feld in self.contract.werbe_feld_selected:
+ for media in feld.mediastructure.medias:
+ if not media:
+ _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
+ newkampagne.id) + ' Kein Medium in MedienStructur von F_' + str(feld.id) + 'M_' + str(
+ media.id))
+ else:
+ newmedia = self.env['dss.mediarelations'].create(
+ {'field': feld.id, 'contract': self.contract.id, 'project': self.project.id,
+ 'field_uuid': feld.uuid, 'ad': newkampagne.id, 'relname': media.medianame,
+ 'mediatype': media.id})
+ _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
+ newkampagne.id) + ' setze Vertrag für Medium : ' + str(newmedia.contract) + '/' + str(
+ newmedia.ad))
+ mediaids.append(newmedia.id)
+ newkampagne.write({'need_media': [(6, 0, mediaids)]})
+ _logger.info('Click auf Werbekampagne : C_' + str(self.contract.id) + 'A_' + str(
+ newkampagne.id) + ' setze Media ')
+
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'form',
+ 'view_mode': 'form,tree',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '',
+ 'res_id': newkampagne.id,
+ 'display_name': 'Änderung zur Werbekampagne ' + newkampagne.parent_ad.adname,
+ 'domain': '[("id","=","context[newkampagne.id]")]'
+ }
+ else:
+ raise ValidationError(
+ _("Kann neue Aktualisierung erst anlegen wenn alle vorherigen Schritte beendet wurden !"))
+
+ def setStandardText(self, tid):
+ text = self.env['dss.texts'].search([('text_id', '=', tid)], limit=1)
+ if text:
+ self.write({'description': text.description})
+
+
+class dsscontractads(models.Model):
+
+ def _default_work_state(self):
+ ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).id
+ _logger.debug(ds)
+ return ds
+
+ def _default_work_state_color(self):
+ ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).color
+ _logger.info(ds)
+ return ds
+
+ def _default_todo_state(self):
+ ds = self.env['dss.todostate'].search([('statusnr', '=', '0')], limit=1).id
+ if not ds: ds = 1
+ _logger.debug(ds)
+ # ds =
+ return ds
+
+ @api.model
+ def create(self, vals):
+ result = super().create(vals)
+ resstr = result.uuid
+ _logger.info(resstr)
+ for n_record in result:
+ resstr = n_record.uuid
+ _logger.info(resstr)
+ n_record.date_create = date.today()
+ n_record.user_create = self.env.user.name
+ allads = n_record.contract.ads
+ _logger.info(allads)
+ if allads != False:
+ for ad in allads:
+ ad.ad_is_lastpos = False
+ n_record.ad_is_lastpos = True
+ contract = n_record.contract
+ contract.ads_last_ad = n_record
+ return result
+
+ # @api.depends('need_media')
+ # def _getmedialist(self):
+ # mlist = []
+ # mlist = self.env['dss.mediarelations'].search([('ad','=',self.id)])
+ # if mlist != False:
+ # _logger.info('AD Need_Medias A_'+str(self)+' - Computed Medias '+str(mlist))
+ # self.need_media = mlist.ids
+ # else:
+ # self.need_media = False
+
+ _name = "dss.ads"
+ _description = "DigitalSignage Werbekampagnen"
+ _inherit = ['mail.thread', 'mail.activity.mixin','dss.triggermodel']
+ # _inherit = ['mail.activity.mixin']
+ # 'mail.thread','mail.activity.mixin'
+ _rec_name = "adname"
+ # _inherit = ['mail.thread', 'mail.activity.mixin']
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False,
+ string='UUID')
+ aduuid = fields.Char(related='uuid')
+
+ date_create = fields.Date('Erstellungsdatum', default=lambda self: self._default_create_date())
date_lastedit = fields.Date('Änderungsdatum')
- user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
+ user_create = fields.Char('Erstellungsuser', default=lambda self: self._default_create_user())
user_lastedit = fields.Char('Änderungsuser')
+ # uuid = fields.Char('UUID', required=True, translate=True)
+ adname = fields.Char('Kampagnenname', required=True, tracking=True)
+ adtype = fields.Selection(
+ [('MAIN', 'Ersteinrichtung'), ('KCHN', 'Änderung durch Kunde'), ('LCHN', 'Änderung durch Logumedia'),
+ ('SONS', 'Sonstige Änderung'),('PREA', 'Vorausplanung')], tracking=True)
+ adpos = fields.Integer('Reihenfolge', default=lambda self: self._default_adpos(),
+ tracking=True)
+ ad_is_lastpos = fields.Boolean('Letzte Aktion', tracking=True)
- triggeractionname = fields.Char('Triggeraktionname',track_visibility='onchange',tracking=True)
- triggeractiontyp = fields.Selection([('EMAIL','EMail senden'),('ACTIV','Aufgabe erzeugen'),('DSEDIT','Datensatz ändern'),('TUYA','Tuya Aktion auslössen')],'Aktionstyp',track_visibility='onchange',tracking=True)
- email_getfrom = fields.Selection([('FIX','Email fest hinterlegt'),('DATA','Email in Feld hinterlegt')],'Emailadresse von',track_visibility='onchange',tracking=True)
- email_fix_email_sender = fields.Many2one('res.partner','Absender',help='Email Sender Adresse der Aktion',track_visibility='onchange',tracking=True)
- email_fix_email = fields.Many2one('res.partner','Empfänger',help='Email Emüfänger Adresse für Benachrichtigung/Triggeraktion',track_visibility='onchange',tracking=True)
- email_data_table = fields.Many2one('ir.model','Nutzbar in',help='Für welche Daten ist diese Aktion gültig ?',track_visibility='onchange',tracking=True)
- email_data_field = fields.Many2one('ir.model.fields', help='Feldname in der Tabelle',track_visibility='onchange',tracking=True)
- email_template = fields.Many2one('mail.template','Emailvorlage',track_visibility='onchange',tracking=True)
+ contract = fields.Many2one('dss.contracts', store=True, tracking=True)
+ contract_id = fields.Char(related='contract.contract_id')
+ contract_name = fields.Char(related='contract.contract_name')
+ contract_need_media = fields.Many2many(related='contract.need_media', tracking=True)
- field = fields.Char('Feldname',help='Feldname in o.g. Tebelle.',track_visibility='onchange',tracking=True)
- type_s = fields.Char('Feldwertoperator',track_visibility='onchange',tracking=True)
- value_s = fields.Char('Feldwert für Trigger',track_visibility='onchange',tracking=True)
+ project = fields.Many2one('dss.projects', string='Project', store=True, tracking=True)
+ project_id = fields.Integer(related='project.projectid', string='Project ID')
+
+ parent_ad = fields.Many2one('dss.ads', string='UrsprungsWerbung', store=True)
+ parent_ad_uuid = fields.Char(related='parent_ad.uuid')
+ description = fields.Text('Beschreibung')
+ amount = fields.Float('Extrakosten', tracking=True)
+ order = fields.Integer('Reihenfolge', tracking=True)
+
+ # need_media = fields.Many2many('dss.mediarelations','ad',string='Medien')
+ need_media = fields.One2many('dss.mediarelations', 'ad', string='Medien')
+
+ work_state = fields.Many2one('dss.workstate', default=_default_work_state, tracking=True)
+ work_state_color = fields.Char(related='work_state.color')
+ work_state_text = fields.Char(related='work_state.statusname')
+ work_state_info = fields.Char('Zusatzinfo')
+
+ ad_state = fields.Many2one('dss.adstate', tracking=True)
+ ad_state_color = fields.Char(related='ad_state.color')
+ ad_state_text = fields.Char(related='ad_state.statusname')
+ ad_state_func = fields.Selection(related='ad_state.func')
+
+ todo_state = fields.Many2one('dss.todostate', default=_default_todo_state,
+ tracking=True)
+ todo_state_color = fields.Char(related='todo_state.color')
+ todo_state_text = fields.Char(related='todo_state.statusname')
+ todo_state_info = fields.Char('Zusatzinfo', tracking=True)
+ todo_state_until = fields.Date('Abarbeiten bis', tracking=True)
+
+ cloud_add_directory = fields.Char('Cloud KundenKampagnen Ordner', tracking=True)
+
+ date_zuarbeit = fields.Date(string='Zuarbeit Datum', help='Zuarbeit gesendet am',
+ tracking=True)
+ date_korrekturabzug = fields.Date(string='K.Abzug Datum', help='Korrekturabzug gesendet am',
+ tracking=True)
+ date_korrekturfreigabe = fields.Date(string='K.Freigabe Datum', help='Korrekturfreigabe erhalten am',
+ tracking=True)
+ date_korrekturfreigabe_ablauf = fields.Date(string='K.Freigabe Ablaufdatum', help='Korrekturfreigabe muss bis xxxx erfolgen.. sonst Autofreigabe',
+ tracking=True)
+ date_korrekturfreigabe_ablauf_compute = fields.Char(compute='_date_korrekturfreigabe_ablauf_compute')
+ date_korrekturfreigabe_ablauf_compute_int = fields.Integer(compute='_date_korrekturfreigabe_ablauf_compute_int')
+ korrekturfreigabe_ablauf_erfolgt = fields.Boolean(string='K.Freigabe durch Ablauf', help='Korrekturfreigabe erfolgte durch Zeitablauf ?',
+ tracking=True)
+ date_start_planed = fields.Date(string='geplantes Startdatum', tracking=True)
+ date_start_planed_calendar_event=fields.Integer('gepl. Start - Kal.ID',tracking=True)
+ date_start_real = fields.Date(string='wirkliches Startdatum', tracking=True)
+
+ date_remove_planed = fields.Date(string='geplantes Enddatum', tracking=True)
+ date_remove_real = fields.Date(string='wirkliches Enddatum', tracking=True)
+
+ @api.depends('date_korrekturfreigabe_ablauf')
+ def _date_korrekturfreigabe_ablauf_compute(self):
+ self.date_korrekturfreigabe_ablauf_compute = ''
+ if not self:
+ return ''
+ pass
+ else:
+ for record in self:
+ if not record:
+ return ''
+ pass
+ else:
+ if not record.date_korrekturfreigabe_ablauf:
+ return ''
+ pass
+ else:
+ datedif = record.date_korrekturfreigabe_ablauf-date.today()
+ record.date_korrekturfreigabe_ablauf_compute='Noch '+str(datedif.days)+' Tage bis Autoablauf'
+
+ @api.depends('date_korrekturfreigabe_ablauf')
+ def _date_korrekturfreigabe_ablauf_compute_int(self):
+ for record in self:
+ if record:
+ if record.date_korrekturfreigabe_ablauf:
+ datedif = record.date_korrekturfreigabe_ablauf-date.today()
+ record.date_korrekturfreigabe_ablauf_compute_int = datedif.days
+
+ def _check_date_korrekturfreigabe_ablauf(self):
+ for record in self:
+ if record:
+ if record.date_korrekturfreigabe_ablauf:
+ datedif = record.date_korrekturfreigabe_ablauf-date.today()
+ _logger.info('Contract Autokorrektur Running C_' + str(record.contract)+' / '+str(datedif))
+ if (datedif.days<=0) & ( not record.date_korrekturfreigabe):
+ _logger.info('Contract Autokorrektur Running C_' + str(record.contract)+' A_'+str(record)+' Set Values')
+ record.date_korrekturfreigabe = date.today()
+ record.korrekturfreigabe_ablauf_erfolgt = True
+ body = 'Für die Werbekampagne '+str(record.adname)+' im Vertrag '+str(record.contract.contract_auto_name)+' des Projektes '+str(record.project.projektname)+' wurde der Korekturabzug automatisch durch Zeitablauf bestätigt !'
+ record.message_post(body=body)
+ self.env['mail.message'].create({'message_type': 'comment','subtype_id': self.env.ref('mail.mt_comment').id,
+ 'model': 'mail.channel','res_id': self.env.ref('mail.channel_all_employees').id,'body': body,})
@api.model
def _default_uuid(self):
@@ -428,26 +820,249 @@ class dsstriggertypes(models.Model):
def _default_create_user(self):
return str(self.env.user.name)
-class dsstriggertypes(models.Model):
- _name = "dss.triggergroups"
- _description = "DigitalSignage Trigger Gruppen"
- _inherit = ['mail.thread','mail.activity.mixin']
- _rec_name = "triggergroupname"
-# _inherit = ['mail.thread', 'mail.activity.mixin']
- uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
- date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
- date_lastedit = fields.Date('Änderungsdatum')
- user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
- user_lastedit = fields.Char('Änderungsuser')
+ def _default_adpos(self):
+ pos = self.env['dss.ads'].search_count([('contract_id', '=', self.contract.id)]) + 1
+ return str(pos)
- triggergroupname = fields.Char('Trigger Gruppenname',track_visibility='onchange',tracking=True)
+ @api.onchange('ad_state')
+ def _onchange_ad_state(self):
+ for record in self:
+ if record.ad_is_lastpos == True:
+ contract = record.contract
+ contract.ads_last_state = record.ad_state
+ # mtyp = self.env['dss.adstructures'].search([("uuid","=",str(record.mediastructure.uuid))])
+ # buildmediarelations(self)
+ self.date_lastedit = str(date.today())
- @api.model
- def _default_uuid(self):
- return str(uuid.uuid4())
+ # self.mediastructure = mtyp
- def _default_create_date(self):
- return str(date.today())
+ @api.onchange('date_korrekturabzug')
+ def _date_korrekturabzug_change(self):
+ for record in self:
+ autodays=0
+ if record.date_korrekturabzug:
+ autodays=self.env['dss.settings'].search([],limit=1).freigabe_auto_time
+ record.date_korrekturfreigabe_ablauf = record.date_korrekturabzug + relativedelta(days=autodays)
- def _default_create_user(self):
- return str(self.env.user.name)
+ def pyaction_view_ad_details(self):
+ action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
+ 'DigitalSignage.action_dss_ads_view')
+ # action['display_name'] = self.ad_name
+ # action['domain'] = '[["projectid","=","4"]]'
+ # context = action['context'].replace('', str(self.id))
+ # context = ast.literal_eval(context)
+ # context.update({
+ # 'create': self.active,
+ # 'active_test': self.active
+ # })
+ # action['context'] = context
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'form',
+ 'view_mode': 'form,tree',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '',
+ 'res_id': self.id,
+ 'display_name': ' ' + self.adname,
+ 'domain': ''
+ }
+
+ return action
+
+ def pydoviewallads(self):
+ action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
+ 'DigitalSignage.act_dss_ads_view_full')
+ action['display_name'] = self.contract_name
+ # action['domain'] = '[["projectid","=","4"]]'
+ # context = action['context'].replace('', str(self.id))
+ # context = ast.literal_eval(context)
+ # context.update({
+ # 'create': self.active,
+ # 'active_test': self.active
+ # })
+ # action['context'] = context
+ # return {
+ # 'type': 'ir.actions.act_window',
+ # 'view_type':'form',
+ # 'view_mode':'form,tree',
+ # 'res_model':'dss.ads',
+ # 'target':'current',
+ # 'context':'',
+ # 'res_id':kampagne.id,
+ # 'display_name' : 'Werbekampagne '+kampagne.adname,
+ # 'domain':'[("contract","=","context[active_id]")]'
+ # }
+ # return action
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'kanban',
+ 'view_mode': 'kanban',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '{"group_by":["parent_ad"]}',
+ # 'res_id':self.parent_ad,
+ 'display_name': 'Übersicht für ' + self.adname,
+ 'domain': [("contract", "=", self.contract.id)]
+ }
+
+ def pydonewpread(self):
+ defadstate = self.env['dss.adstate'].search([('func', '=', 'STD')], limit=1).id
+ if not defadstate:
+ defadstate = 1
+ if self.ad_state.func == 'STD':
+ parent_id = self.id
+ else:
+ parent_id = self.parent_ad.id
+ newkampagne = self.env['dss.ads'].create({'contract': self.contract.id, 'project': self.project.id,
+ 'adname': 'PRE_' + self.contract.contract_auto_name + ' ' + str(
+ date.today()), 'parent_ad': parent_id, 'adtype': 'PREA',
+ 'ad_state': defadstate})
+ _logger.info('C_' + str(self.id) + ' Kampagne erzeugt : ' + str(newkampagne))
+ _logger.info('Erzeugte PreWerbekampagne - Prüfe Medien : C_' + str(self.id) + 'K_' + str(newkampagne.id))
+
+ mediaids = []
+ for feld in self.contract.werbe_feld_selected:
+ for media in feld.mediastructure.medias:
+ if not media:
+ _logger.info('Erzeugte PreWerbekampagne - Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
+ newkampagne.id) + ' Kein Medium in MedienStructur von F_' + str(feld.id) + 'M_' + str(
+ media.id))
+ else:
+ newmedia = self.env['dss.mediarelations'].create(
+ {'field': feld.id, 'contract': self.contract.id, 'project': self.project.id,
+ 'field_uuid': feld.uuid, 'ad': newkampagne.id, 'relname': media.medianame,
+ 'mediatype': media.id})
+ _logger.info('Erzeugte PreWerbekampagne - Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
+ newkampagne.id) + ' setze Vertrag für Medium : ' + str(newmedia.contract) + '/' + str(
+ newmedia.ad))
+ mediaids.append(newmedia.id)
+ newkampagne.write({'need_media': [(6, 0, mediaids)]})
+ _logger.info('Erzeugte PreWerbekampagne : C_' + str(self.contract.id) + 'A_' + str(
+ newkampagne.id) + ' setze Media ')
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'form',
+ 'view_mode': 'form,tree',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '',
+ 'res_id': newkampagne.id,
+ 'display_name': 'Änderung zur Werbekampagne ' + newkampagne.parent_ad.adname,
+ 'domain': '[("id","=","context[newkampagne.id]")]'
+ }
+
+ def pydonewad(self):
+ for n_record in self:
+ resstr = n_record.uuid
+ _logger.info(resstr)
+ allads = n_record.contract.ads
+ _logger.info(allads)
+ abort = False
+ if allads != False:
+ for ad in allads:
+ if ad.ad_state.func != 'FIN':
+ if ad.adtype !='PREA':
+ abort = True
+ if abort == False:
+ defadstate = self.env['dss.adstate'].search([('func', '=', 'STD')], limit=1).id
+ if not defadstate:
+ defadstate = 1
+ if self.ad_state.func == 'STD':
+ parent_id = self.id
+ else:
+ parent_id = self.parent_ad.id
+ newkampagne = self.env['dss.ads'].create({'contract': self.contract.id, 'project': self.project.id,
+ 'adname': 'AD_' + self.contract.contract_auto_name + ' ' + str(
+ date.today()), 'parent_ad': parent_id, 'adtype': 'KCHN',
+ 'ad_state': defadstate})
+ _logger.info('C_' + str(self.id) + ' Kampagne erzeugt : ' + str(newkampagne))
+ _logger.info('Prüfe Medien : C_' + str(self.id) + 'K_' + str(newkampagne.id))
+
+ mediaids = []
+ for feld in self.contract.werbe_feld_selected:
+ for media in feld.mediastructure.medias:
+ if not media:
+ _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
+ newkampagne.id) + ' Kein Medium in MedienStructur von F_' + str(feld.id) + 'M_' + str(
+ media.id))
+ else:
+ newmedia = self.env['dss.mediarelations'].create(
+ {'field': feld.id, 'contract': self.contract.id, 'project': self.project.id,
+ 'field_uuid': feld.uuid, 'ad': newkampagne.id, 'relname': media.medianame,
+ 'mediatype': media.id})
+ _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
+ newkampagne.id) + ' setze Vertrag für Medium : ' + str(newmedia.contract) + '/' + str(
+ newmedia.ad))
+ mediaids.append(newmedia.id)
+ newkampagne.write({'need_media': [(6, 0, mediaids)]})
+ _logger.info('Click auf Werbekampagne : C_' + str(self.contract.id) + 'A_' + str(
+ newkampagne.id) + ' setze Media ')
+
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'form',
+ 'view_mode': 'form,tree',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '',
+ 'res_id': newkampagne.id,
+ 'display_name': 'Änderung zur Werbekampagne ' + newkampagne.parent_ad.adname,
+ 'domain': '[("id","=","context[newkampagne.id]")]'
+ }
+ else:
+ raise ValidationError(
+ _("Kann neue Aktualisierung erst anlegen wenn alle vorherigen Schritte beendet wurden !"))
+ def pydoprecopyad(selfself):
+ for n_record in self:
+ resstr = n_record.uuid
+ _logger.info(resstr)
+ allads = n_record.contract.ads
+ _logger.info(allads)
+ abort = False
+ if allads != False:
+ for ad in allads:
+ if ad.ad_state.func != 'FIN':
+ if ad.adtype != 'PREA':
+ abort = True
+ if abort == False:
+
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_type': 'form',
+ 'view_mode': 'form,tree',
+ 'res_model': 'dss.ads',
+ 'target': 'current',
+ 'context': '',
+ 'res_id': newkampagne.id,
+ 'display_name': 'Änderung zur Werbekampagne ' + newkampagne.parent_ad.adname,
+ 'domain': '[("id","=","context[newkampagne.id]")]'
+ }
+
+ else:
+ raise ValidationError(
+ _("Kann neue Aktualisierung erst anlegen wenn alle vorherigen Schritte beendet wurden !"))
+
+ def setStandardText(self, tid):
+ text = self.env['dss.texts'].search([('text_id', '=', tid)], limit=1)
+ if text:
+ self.write({'description': text.description})
+
+ def pyaddstartdate(self):
+ self.env['calendar.event'].search([('id','=',self.date_start_planed_calendar_event)]).unlink()
+ event = {
+ 'start': self.date_start_planed.strftime('%Y-%m-%d %H:%M:%S'),
+ 'stop': self.date_start_planed.strftime('%Y-%m-%d %H:%M:%S'),
+ 'duration': 24,
+ 'allday': True,
+ 'partner_ids': (4,self.env['dss.settings'].search([],limit=1).contact_runtime_user.partner_id.id),
+ 'name': 'Geplanter Start '+self.contract.contract_auto_name,
+ 'description': 'Geplanter Start '+self.contract.contract_auto_name+' im Projekt '+self.project.projektname,
+ 'user_id': self.env['dss.settings'].search([],limit=1).contact_runtime_user.id
+ }
+ _logger.debug('linking new Startevent ' + str(event) + ' ' + str(self.uuid))
+ self.date_start_planed_calendar_event=self.env['calendar.event'].create(event).id
+ return ""
+
+ def pyaddenddate(self):
+ return 1
diff --git a/models/dss_ads.py b/models/dss_ads.py
deleted file mode 100644
index ec4c81e..0000000
--- a/models/dss_ads.py
+++ /dev/null
@@ -1,576 +0,0 @@
-import ast
-import datetime
-import json
-import re
-import uuid
-import logging
-import base64
-import subprocess
-import tempfile
-import easywebdav
-import os
-import os.path
-
-
-from odoo import api, fields, models, _
-from odoo import tools
-from odoo.exceptions import ValidationError
-from datetime import date
-from datetime import datetime
-from dateutil.relativedelta import relativedelta
-from pyffmpeg import FFmpeg
-from tuya_iot import TuyaOpenAPI, TUYA_LOGGER
-from tuya_connector import TuyaOpenAPI, TUYA_LOGGER
-import sys
-
-TUYA_LOGGER.setLevel(logging.DEBUG)
-_logger = logging.getLogger(__name__)
-
-
-class dsscontractads(models.Model):
-
- def _default_work_state(self):
- ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).id
- _logger.debug(ds)
- return ds
-
- def _default_work_state_color(self):
- ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).color
- _logger.info(ds)
- return ds
-
- def _default_todo_state(self):
- ds = self.env['dss.todostate'].search([('statusnr', '=', '0')], limit=1).id
- if not ds: ds = 1
- _logger.debug(ds)
- # ds =
- return ds
-
- @api.model
- def create(self, vals):
- result = super().create(vals)
- resstr = result.uuid
- _logger.info(resstr)
- for n_record in result:
- resstr = n_record.uuid
- _logger.info(resstr)
- n_record.date_create = date.today()
- n_record.user_create = self.env.user.name
- allads = n_record.contract.ads
- _logger.info(allads)
- if allads != False:
- for ad in allads:
- ad.ad_is_lastpos = False
- n_record.ad_is_lastpos = True
- contract = n_record.contract
- contract.ads_last_ad = n_record
- return result
-
- # @api.depends('need_media')
- # def _getmedialist(self):
- # mlist = []
- # mlist = self.env['dss.mediarelations'].search([('ad','=',self.id)])
- # if mlist != False:
- # _logger.info('AD Need_Medias A_'+str(self)+' - Computed Medias '+str(mlist))
- # self.need_media = mlist.ids
- # else:
- # self.need_media = False
-
- _name = "dss.ads"
- _description = "DigitalSignage Werbekampagnen Phasen"
- _inherit = ['mail.thread', 'mail.activity.mixin']
- # _inherit = ['mail.activity.mixin']
- # 'mail.thread','mail.activity.mixin'
- _rec_name = "adname"
- # _inherit = ['mail.thread', 'mail.activity.mixin']
- uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False,
- string='UUID')
- aduuid = fields.Char(related='uuid')
-
- date_create = fields.Date('Erstellungsdatum', default=lambda self: self._default_create_date())
- date_lastedit = fields.Date('Änderungsdatum')
- user_create = fields.Char('Erstellungsuser', default=lambda self: self._default_create_user())
- user_lastedit = fields.Char('Änderungsuser')
- # uuid = fields.Char('UUID', required=True, translate=True)
- adname = fields.Char('Kampagnenname', required=True, tracking=True)
- adtype = fields.Selection(
- [('MAIN', 'Ersteinrichtung'), ('KCHN', 'Änderung durch Kunde'), ('LCHN', 'Änderung durch Logumedia'),
- ('SONS', 'Sonstige Änderung')], tracking=True)
- adpos = fields.Integer('Reihenfolge', default=lambda self: self._default_adpos(),
- tracking=True)
- ad_is_lastpos = fields.Boolean('Letzte Aktion', tracking=True)
-
- start_date = fields.Date('gew. Ausstrahl.Begin', tracking=True)
- end_date = fields.Date('gew. Ausstrahl.Ende', tracking=True)
-
- contract = fields.Many2one('dss.contracts', store=True, tracking=True)
- contract_id = fields.Char(related='contract.contract_id')
- contract_name = fields.Char(related='contract.contract_name')
- contract_need_media = fields.Many2many(related='contract.need_media', tracking=True)
-
- project = fields.Many2one('dss.projects', string='Project', store=True, tracking=True)
- project_id = fields.Integer(related='project.projectid', string='Project ID')
-
- parent_ad = fields.Many2one('dss.ads', string='UrsprungsWerbung', store=True)
- parent_ad_uuid = fields.Char(related='parent_ad.uuid')
- description = fields.Text('Beschreibung')
- amount = fields.Float('Extrakosten', tracking=True)
- order = fields.Integer('Reihenfolge', tracking=True)
-
- # need_media = fields.Many2many('dss.mediarelations','ad',string='Medien')
- need_media = fields.One2many('dss.mediarelations', 'ad', string='Medien')
-
- work_state = fields.Many2one('dss.workstate', default=_default_work_state, tracking=True)
- work_state_color = fields.Char(related='work_state.color')
- work_state_text = fields.Char(related='work_state.statusname')
- work_state_info = fields.Char('Zusatzinfo')
-
- ad_state = fields.Many2one('dss.adstate', tracking=True)
- ad_state_color = fields.Char(related='ad_state.color')
- ad_state_text = fields.Char(related='ad_state.statusname')
- ad_state_func = fields.Selection(related='ad_state.func')
-
- todo_state = fields.Many2one('dss.todostate', default=_default_todo_state,
- tracking=True)
- todo_state_color = fields.Char(related='todo_state.color')
- todo_state_text = fields.Char(related='todo_state.statusname')
- todo_state_info = fields.Char('Zusatzinfo', tracking=True)
- todo_state_until = fields.Date('Abarbeiten bis', tracking=True)
-
- cloud_add_directory = fields.Char('Cloud KundenKampagnen Ordner', tracking=True)
-
- date_zuarbeit = fields.Date(string='Zuarbeit Datum', help='Zuarbeit gesendet am', tracking=True)
- date_korrekturabzug = fields.Date(string='K.Abzug Datum', help='Korrekturabzug gesendet am')
- date_korrekturfreigabe = fields.Date(string='K.Freigabe Datum', help='Korrekturfreigabe erhalten am',
- tracking=True)
- date_korrekturfreigabe_frist = fields.Date(string='K.Freigabe bis Datum', help='Korrekturfreigabe sollte bis .... erfolgen')
- freigabe_durch_ablauf = fields.Boolean(string='Zeitablauf',help='Freigabe wurde durch Zeitablauf erreicht ?', tracking=True)
-
- @api.model
- def _default_uuid(self):
- return str(uuid.uuid4())
-
- def _default_create_date(self):
- return str(date.today())
-
- def _default_create_user(self):
- return str(self.env.user.name)
-
- def _default_adpos(self):
- pos = self.env['dss.ads'].search_count([('contract_id', '=', self.contract.id)]) + 1
- return str(pos)
-
- @api.onchange('ad_state')
- def _onchange_ad_state(self):
- for record in self:
- if record.ad_is_lastpos == True:
- contract = record.contract
- contract.ads_last_state = record.ad_state
- # mtyp = self.env['dss.adstructures'].search([("uuid","=",str(record.mediastructure.uuid))])
- # buildmediarelations(self)
- self.date_lastedit = str(date.today())
-
- # self.mediastructure = mtyp
-
- def pyaction_view_ad_details(self):
- action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
- 'DigitalSignage.action_dss_ads_view')
- # action['display_name'] = self.ad_name
- # action['domain'] = '[["projectid","=","4"]]'
- # context = action['context'].replace('', str(self.id))
- # context = ast.literal_eval(context)
- # context.update({
- # 'create': self.active,
- # 'active_test': self.active
- # })
- # action['context'] = context
- return {
- 'type': 'ir.actions.act_window',
- 'view_type': 'form',
- 'view_mode': 'form,tree',
- 'res_model': 'dss.ads',
- 'target': 'current',
- 'context': '',
- 'res_id': self.id,
- 'display_name': ' ' + self.adname,
- 'domain': ''
- }
-
- return action
-
- def pydoviewallads(self):
- action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
- 'DigitalSignage.act_dss_ads_view_full')
- action['display_name'] = self.contract_name
- # action['domain'] = '[["projectid","=","4"]]'
- # context = action['context'].replace('', str(self.id))
- # context = ast.literal_eval(context)
- # context.update({
- # 'create': self.active,
- # 'active_test': self.active
- # })
- # action['context'] = context
- # return {
- # 'type': 'ir.actions.act_window',
- # 'view_type':'form',
- # 'view_mode':'form,tree',
- # 'res_model':'dss.ads',
- # 'target':'current',
- # 'context':'',
- # 'res_id':kampagne.id,
- # 'display_name' : 'Werbekampagne '+kampagne.adname,
- # 'domain':'[("contract","=","context[active_id]")]'
- # }
- # return action
- return {
- 'type': 'ir.actions.act_window',
- 'view_type': 'kanban',
- 'view_mode': 'kanban',
- 'res_model': 'dss.ads',
- 'target': 'current',
- 'context': '{"group_by":["parent_ad"]}',
- # 'res_id':self.parent_ad,
- 'display_name': 'Übersicht für ' + self.adname,
- 'domain': [("contract", "=", self.contract.id)]
- }
-
- def pydonewad(self):
- for n_record in self:
- resstr = n_record.uuid
- _logger.info(resstr)
- allads = n_record.contract.ads
- _logger.info(allads)
- abort = False
- if allads != False:
- for ad in allads:
- if ad.ad_state.func != 'FIN':
- abort = True
- if abort == False:
- defadstate = self.env['dss.adstate'].search([('func', '=', 'STD')], limit=1).id
- if not defadstate:
- defadstate = 1
- if self.ad_state.func == 'STD':
- parent_id = self.id
- else:
- parent_id = self.parent_ad.id
- newkampagne = self.env['dss.ads'].create({'contract': self.contract.id, 'project': self.project.id,
- 'adname': 'AD_' + self.contract.contract_auto_name + ' ' + str(
- date.today()), 'parent_ad': parent_id, 'adtype': 'KCHN',
- 'ad_state': defadstate})
- _logger.info('C_' + str(self.id) + ' Kampagne erzeugt : ' + str(newkampagne))
- _logger.info('Prüfe Medien : C_' + str(self.id) + 'K_' + str(newkampagne.id))
-
- mediaids = []
- for feld in self.contract.werbe_feld_selected:
- for media in feld.mediastructure.medias:
- if not media:
- _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
- newkampagne.id) + ' Kein Medium in MedienStructur von F_' + str(feld.id) + 'M_' + str(
- media.id))
- else:
- newmedia = self.env['dss.mediarelations'].create(
- {'field': feld.id, 'contract': self.contract.id, 'project': self.project.id,
- 'field_uuid': feld.uuid, 'ad': newkampagne.id, 'relname': media.medianame,
- 'mediatype': media.id})
- _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
- newkampagne.id) + ' setze Vertrag für Medium : ' + str(newmedia.contract) + '/' + str(
- newmedia.ad))
- mediaids.append(newmedia.id)
- newkampagne.write({'need_media': [(6, 0, mediaids)]})
- _logger.info('Click auf Werbekampagne : C_' + str(self.contract.id) + 'A_' + str(
- newkampagne.id) + ' setze Media ')
-
- return {
- 'type': 'ir.actions.act_window',
- 'view_type': 'form',
- 'view_mode': 'form,tree',
- 'res_model': 'dss.ads',
- 'target': 'current',
- 'context': '',
- 'res_id': newkampagne.id,
- 'display_name': 'Änderung zur Werbekampagne ' + newkampagne.parent_ad.adname,
- 'domain': '[("id","=","context[newkampagne.id]")]'
- }
- else:
- raise ValidationError(
- _("Kann neue Aktualisierung erst anlegen wenn alle vorherigen Schritte beendet wurden !"))
-
- def setStandardText(self, tid):
- text = self.env['dss.texts'].search([('text_id', '=', tid)], limit=1)
- if text:
- self.write({'description': text.description})
-
-
-class dsscontractads(models.Model):
-
- def _default_work_state(self):
- ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).id
- _logger.debug(ds)
- return ds
-
- def _default_work_state_color(self):
- ds = self.env['dss.workstate'].search([('statusname', '=', 'Neu')], limit=1).color
- _logger.info(ds)
- return ds
-
- def _default_todo_state(self):
- ds = self.env['dss.todostate'].search([('statusnr', '=', '0')], limit=1).id
- if not ds: ds = 1
- _logger.debug(ds)
- # ds =
- return ds
-
- @api.model
- def create(self, vals):
- result = super().create(vals)
- resstr = result.uuid
- _logger.info(resstr)
- for n_record in result:
- resstr = n_record.uuid
- _logger.info(resstr)
- n_record.date_create = date.today()
- n_record.user_create = self.env.user.name
- allads = n_record.contract.ads
- _logger.info(allads)
- if allads != False:
- for ad in allads:
- ad.ad_is_lastpos = False
- n_record.ad_is_lastpos = True
- contract = n_record.contract
- contract.ads_last_ad = n_record
- return result
-
- # @api.depends('need_media')
- # def _getmedialist(self):
- # mlist = []
- # mlist = self.env['dss.mediarelations'].search([('ad','=',self.id)])
- # if mlist != False:
- # _logger.info('AD Need_Medias A_'+str(self)+' - Computed Medias '+str(mlist))
- # self.need_media = mlist.ids
- # else:
- # self.need_media = False
-
- _name = "dss.ads"
- _description = "DigitalSignage Werbekampagnen Phasen"
- _inherit = ['mail.thread', 'mail.activity.mixin']
- # _inherit = ['mail.activity.mixin']
- # 'mail.thread','mail.activity.mixin'
- _rec_name = "adname"
- # _inherit = ['mail.thread', 'mail.activity.mixin']
- uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False,
- string='UUID')
- aduuid = fields.Char(related='uuid')
-
- date_create = fields.Date('Erstellungsdatum', default=lambda self: self._default_create_date())
- date_lastedit = fields.Date('Änderungsdatum')
- user_create = fields.Char('Erstellungsuser', default=lambda self: self._default_create_user())
- user_lastedit = fields.Char('Änderungsuser')
- # uuid = fields.Char('UUID', required=True, translate=True)
- adname = fields.Char('Kampagnenname', required=True, tracking=True)
- adtype = fields.Selection(
- [('MAIN', 'Ersteinrichtung'), ('KCHN', 'Änderung durch Kunde'), ('LCHN', 'Änderung durch Logumedia'),
- ('SONS', 'Sonstige Änderung')], tracking=True)
- adpos = fields.Integer('Reihenfolge', default=lambda self: self._default_adpos(),
- tracking=True)
- ad_is_lastpos = fields.Boolean('Letzte Aktion', tracking=True)
-
- start_date = fields.Date('gew. Ausstrahl.Begin', tracking=True)
- end_date = fields.Date('gew. Ausstrahl.Ende', tracking=True)
-
- contract = fields.Many2one('dss.contracts', store=True, tracking=True)
- contract_id = fields.Char(related='contract.contract_id')
- contract_name = fields.Char(related='contract.contract_name')
- contract_need_media = fields.Many2many(related='contract.need_media', tracking=True)
-
- project = fields.Many2one('dss.projects', string='Project', store=True, tracking=True)
- project_id = fields.Integer(related='project.projectid', string='Project ID')
-
- parent_ad = fields.Many2one('dss.ads', string='UrsprungsWerbung', store=True)
- parent_ad_uuid = fields.Char(related='parent_ad.uuid')
- description = fields.Text('Beschreibung')
- amount = fields.Float('Extrakosten', tracking=True)
- order = fields.Integer('Reihenfolge', tracking=True)
-
- # need_media = fields.Many2many('dss.mediarelations','ad',string='Medien')
- need_media = fields.One2many('dss.mediarelations', 'ad', string='Medien')
-
- work_state = fields.Many2one('dss.workstate', default=_default_work_state, tracking=True)
- work_state_color = fields.Char(related='work_state.color')
- work_state_text = fields.Char(related='work_state.statusname')
- work_state_info = fields.Char('Zusatzinfo')
-
- ad_state = fields.Many2one('dss.adstate', tracking=True)
- ad_state_color = fields.Char(related='ad_state.color')
- ad_state_text = fields.Char(related='ad_state.statusname')
- ad_state_func = fields.Selection(related='ad_state.func')
-
- todo_state = fields.Many2one('dss.todostate', default=_default_todo_state,
- tracking=True)
- todo_state_color = fields.Char(related='todo_state.color')
- todo_state_text = fields.Char(related='todo_state.statusname')
- todo_state_info = fields.Char('Zusatzinfo', tracking=True)
- todo_state_until = fields.Date('Abarbeiten bis', tracking=True)
-
- cloud_add_directory = fields.Char('Cloud KundenKampagnen Ordner', tracking=True)
-
- date_zuarbeit = fields.Date(string='Zuarbeit Datum', help='Zuarbeit gesendet am',
- tracking=True)
- date_korrekturabzug = fields.Date(string='K.Abzug Datum', help='Korrekturabzug gesendet am',
- track_visibility='onchange', tracking=True)
- date_korrekturfreigabe = fields.Date(string='K.Freigabe Datum', help='Korrekturfreigabe erhalten am',
- track_visibility='onchange', tracking=True)
-
- @api.model
- def _default_uuid(self):
- return str(uuid.uuid4())
-
- def _default_create_date(self):
- return str(date.today())
-
- def _default_create_user(self):
- return str(self.env.user.name)
-
- def _default_adpos(self):
- pos = self.env['dss.ads'].search_count([('contract_id', '=', self.contract.id)]) + 1
- return str(pos)
-
- @api.onchange('ad_state')
- def _onchange_ad_state(self):
- for record in self:
- if record.ad_is_lastpos == True:
- contract = record.contract
- contract.ads_last_state = record.ad_state
- # mtyp = self.env['dss.adstructures'].search([("uuid","=",str(record.mediastructure.uuid))])
- # buildmediarelations(self)
- self.date_lastedit = str(date.today())
-
- # self.mediastructure = mtyp
-
- def pyaction_view_ad_details(self):
- action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
- 'DigitalSignage.action_dss_ads_view')
- # action['display_name'] = self.ad_name
- # action['domain'] = '[["projectid","=","4"]]'
- # context = action['context'].replace('', str(self.id))
- # context = ast.literal_eval(context)
- # context.update({
- # 'create': self.active,
- # 'active_test': self.active
- # })
- # action['context'] = context
- return {
- 'type': 'ir.actions.act_window',
- 'view_type': 'form',
- 'view_mode': 'form,tree',
- 'res_model': 'dss.ads',
- 'target': 'current',
- 'context': '',
- 'res_id': self.id,
- 'display_name': ' ' + self.adname,
- 'domain': ''
- }
-
- return action
-
- def pydoviewallads(self):
- action = self.env['ir.actions.act_window'].with_context({'default_adid': self.id})._for_xml_id(
- 'DigitalSignage.act_dss_ads_view_full')
- action['display_name'] = self.contract_name
- # action['domain'] = '[["projectid","=","4"]]'
- # context = action['context'].replace('', str(self.id))
- # context = ast.literal_eval(context)
- # context.update({
- # 'create': self.active,
- # 'active_test': self.active
- # })
- # action['context'] = context
- # return {
- # 'type': 'ir.actions.act_window',
- # 'view_type':'form',
- # 'view_mode':'form,tree',
- # 'res_model':'dss.ads',
- # 'target':'current',
- # 'context':'',
- # 'res_id':kampagne.id,
- # 'display_name' : 'Werbekampagne '+kampagne.adname,
- # 'domain':'[("contract","=","context[active_id]")]'
- # }
- # return action
- return {
- 'type': 'ir.actions.act_window',
- 'view_type': 'kanban',
- 'view_mode': 'kanban',
- 'res_model': 'dss.ads',
- 'target': 'current',
- 'context': '{"group_by":["parent_ad"]}',
- # 'res_id':self.parent_ad,
- 'display_name': 'Übersicht für ' + self.adname,
- 'domain': [("contract", "=", self.contract.id)]
- }
-
- def pydonewad(self):
- for n_record in self:
- resstr = n_record.uuid
- _logger.info(resstr)
- allads = n_record.contract.ads
- _logger.info(allads)
- abort = False
- if allads != False:
- for ad in allads:
- if ad.ad_state.func != 'FIN':
- abort = True
- if abort == False:
- defadstate = self.env['dss.adstate'].search([('func', '=', 'STD')], limit=1).id
- if not defadstate:
- defadstate = 1
- if self.ad_state.func == 'STD':
- parent_id = self.id
- else:
- parent_id = self.parent_ad.id
- newkampagne = self.env['dss.ads'].create({'contract': self.contract.id, 'project': self.project.id,
- 'adname': 'AD_' + self.contract.contract_auto_name + ' ' + str(
- date.today()), 'parent_ad': parent_id, 'adtype': 'KCHN',
- 'ad_state': defadstate})
- _logger.info('C_' + str(self.id) + ' Kampagne erzeugt : ' + str(newkampagne))
- _logger.info('Prüfe Medien : C_' + str(self.id) + 'K_' + str(newkampagne.id))
-
- mediaids = []
- for feld in self.contract.werbe_feld_selected:
- for media in feld.mediastructure.medias:
- if not media:
- _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
- newkampagne.id) + ' Kein Medium in MedienStructur von F_' + str(feld.id) + 'M_' + str(
- media.id))
- else:
- newmedia = self.env['dss.mediarelations'].create(
- {'field': feld.id, 'contract': self.contract.id, 'project': self.project.id,
- 'field_uuid': feld.uuid, 'ad': newkampagne.id, 'relname': media.medianame,
- 'mediatype': media.id})
- _logger.info('Prüfe Medien : C_' + str(self.contract.id) + 'K_' + str(
- newkampagne.id) + ' setze Vertrag für Medium : ' + str(newmedia.contract) + '/' + str(
- newmedia.ad))
- mediaids.append(newmedia.id)
- newkampagne.write({'need_media': [(6, 0, mediaids)]})
- _logger.info('Click auf Werbekampagne : C_' + str(self.contract.id) + 'A_' + str(
- newkampagne.id) + ' setze Media ')
-
- return {
- 'type': 'ir.actions.act_window',
- 'view_type': 'form',
- 'view_mode': 'form,tree',
- 'res_model': 'dss.ads',
- 'target': 'current',
- 'context': '',
- 'res_id': newkampagne.id,
- 'display_name': 'Änderung zur Werbekampagne ' + newkampagne.parent_ad.adname,
- 'domain': '[("id","=","context[newkampagne.id]")]'
- }
- else:
- raise ValidationError(
- _("Kann neue Aktualisierung erst anlegen wenn alle vorherigen Schritte beendet wurden !"))
-
- def setStandardText(self, tid):
- text = self.env['dss.texts'].search([('text_id', '=', tid)], limit=1)
- if text:
- self.write({'description': text.description})
-
diff --git a/models/dss_advertisefields.py b/models/dss_advertisefields.py
old mode 100644
new mode 100755
diff --git a/models/dss_contract.py b/models/dss_contract.py
old mode 100644
new mode 100755
index 4fe7b94..7915a8d
--- a/models/dss_contract.py
+++ b/models/dss_contract.py
@@ -64,7 +64,7 @@ class dsscontracts(models.Model):
_name = "dss.contracts"
_description = "DigitalSignage Vertraege"
_rec_name = "contract_auto_name"
- _inherit = ['mail.thread','mail.activity.mixin']
+ _inherit = ['mail.thread','mail.activity.mixin','dss.triggermodel']
uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
cruuid = fields.Char(related='uuid')
contract_id = fields.Char("Kundennummer",store=True,tracking=True)
@@ -74,6 +74,18 @@ class dsscontracts(models.Model):
contract_state_order = fields.Integer(related='contract_state.order',store=True)
contract_auto_id = fields.Char("Kundennummer",tracking=True,help="Wird berechnet aus Projektnummer + Kunden ID")
contract_auto_name = fields.Char('Vertragskennug',tracking=True,help="Wird berechnet aus Kundennummer + Vertragskennung")
+ contract_payment_done = fields.Boolean('Zahlungeingang erfolgt',tracking=True)
+ contract_payment_done_date = fields.Date('Zahlungeingang Datum',tracking=True)
+
+ remark = fields.Html('Bemerkung',tracking=True)
+
+ contract_writer = fields.Many2one('res.partner', tracking=True)
+ vertragssumme = fields.Float('Vertragssumme :', tracking=True)
+ provisionstyp = fields.Many2one('dss.provisionstypen', tracking=True)
+ provisionsteilcalc = fields.Float('Berechnete Provision :',computed='_compute_prov')
+ provisions = fields.Many2many('dss.provision',tracking=True)
+ provisionspayedpercent = fields.Float('Ausgezahlte Provision % :',computed='_compute_provisionspayedpercent')
+ grafiker = fields.Many2one('res.partner', placeholder="wenn vom Projekt abweichend", domain="['&',('dssinternpartner','=',True),('dssinternpartner_grafik','=',True)]", tracking=True, string="abw. Grafiker",help="Grafiker nur wenn abweichend vom Projektgrafiker")
contract_remark = fields.Html('Vertragshinweise',tracking=True)
@@ -81,6 +93,8 @@ class dsscontracts(models.Model):
project_id = fields.Integer(related='project.projectid', string='Project ID')
projectIid = fields.Integer('Project IID',tracking=True)
project_ad_structure = fields.Many2one(related='project.grundsystem_default_adstructure', string='Aufbau')
+ project_grundsystem_typ = fields.Char(related='project.grundsystem_typ', tracking=True)
+ project_grafiker = fields.Many2one(related="project.project_grafiker",tracking=True)
client = fields.Many2one('res.partner',string="Kunde (wenn angelegt)",domain="['&',('dsspartner','=',True),('dsspartner_werbung','=',True)]",tracking=True,help="Nur zu Benutzen wenn Kunden als Kontakt angelegt wurde")
client_id = fields.Char(string="KundenID (2Stellen)",help="Nur der 2 stellige letzte Teil der Kundennummer",tracking=True)
@@ -133,6 +147,8 @@ class dsscontracts(models.Model):
contract_date = fields.Date('Vertragsdatum',tracking=True)
start_date = fields.Date('Start/Auslief.datum',tracking=True)
+ contract_cancel_mon = fields.Integer('Kündigungsfrist Monaten',tracking=True)
+ contract_cancel_date = fields.Date('Kündigungsfrist Errechnet', tracking=True)
contract_auto_extend = fields.Boolean('autom. V.Verlängerung',tracking=True)
contract_auto_extend_time = fields.Char('Verl. Monate',tracking=True)
@@ -159,13 +175,12 @@ class dsscontracts(models.Model):
ads_last_work_state = fields.Many2one(related='ads_last_ad.work_state', string="Arbeitschritt letzt Kamp",help="Arbeitsstatus des letzten Werbekampagnen Eintrags",store=True,readonly=False,group_expand='_expand_ads_last_work_state')
ads_last_work_state_color = fields.Char(related='ads_last_ad.work_state_color')
ads_last_work_state_text = fields.Char(related='ads_last_ad.work_state_text',store=True,readonly=False)
- ads_last_work_state_info = fields.Char(related='ads_last_ad.work_state_info')
+ ads_last_work_state_info = fields.Char(related='ads_last_ad.work_state_info',store=True,readonly=False)
ads_last_todo_state = fields.Many2one(related='ads_last_ad.todo_state',string="Aufgabe letzt Kamp", help="Aufgabenstatus des letzten Werbekampagnen Eintrags",store=True,readonly=False,group_expand='_expand_ads_last_todo_state')
ads_last_todo_state_until = fields.Date(related='ads_last_ad.todo_state_until')
ads_last_todo_state_color = fields.Char(related='ads_last_ad.todo_state_color')
ads_last_todo_state_text = fields.Char(related='ads_last_ad.todo_state_text',store=True,readonly=False)
-
vnnox_zugang_erstellt = fields.Boolean('Vnnox Zugang ?',tracking=True)
vnnox_zugang_username = fields.Char('Vnnox Username',tracking=True)
vnnox_zugang_password = fields.Char('Vnnox Passwort',tracking=True)
@@ -174,7 +189,7 @@ class dsscontracts(models.Model):
xibo_zugang_erstellt = fields.Boolean('Xibo Zugang ?',tracking=True)
xibo_zugang_username = fields.Char('Xibo Username',tracking=True)
xibo_zugang_password = fields.Char('Xibo Passwort',tracking=True)
- vnnox_zugang_gesendet = fields.Boolean('Xibo Zugang gesendet?',tracking=True)
+ xibo_zugang_gesendet = fields.Boolean('Xibo Zugang gesendet?',tracking=True)
lmw_zugang_erstellt = fields.Boolean('LMW Zugang ?',tracking=True)
lmw_zugang_username = fields.Char('LMW Username',tracking=True)
@@ -204,6 +219,34 @@ class dsscontracts(models.Model):
cloudlink = fields.Char('Cloud Verzeichnis',help='Verzeichnis für den Kunde innerhalb des Projekt Ordners')
+ @api.depends('vertragssumme')
+ def _compute_prov(self):
+ _logger.info('contract Provision Berechnung : C_' + str(self.id));
+ if self.provisionstyp.provisionbase == 'VSUM':
+ if not self.provisionstyp.provisioncalc:
+ prozent=self.provisionstyp.provisionprozent
+ basis=self.vertragssumme
+ value = (basis/100)*prozent
+ self.provisionsteilcalc = value
+
+ def _compute_provisionspayedpercent(self):
+ addval = 0
+ for prov in self.provisions:
+ _logger.info('contract Provision Payed_part : C_' + str(self.id)+' / '+str(prov.provisionprozent));
+ addval = addval + prov.provisionprozent
+ _logger.info('Contract Provision payed : C_' + str(self.id)+' V '+str(addval));
+ self.provisionspayedpercent = addval
+
+ @api.onchange('provisionstyp')
+ def _onchange_provtype(self):
+ self._compute_prov()
+ self._compute_provisionspayedpercent()
+
+ @api.onchange('vertragssumme')
+ def _onchange_vsumme(self):
+ self._compute_prov()
+ self._compute_provisionspayedpercent()
+
@api.model
def _expand_ads_last_work_state(self,states,domain,order):
if dss_settings.dssSettings._get_settingvalue(self,'show_expand_contract_group_ad_work_state'):
@@ -248,12 +291,8 @@ class dsscontracts(models.Model):
if resstr is None :
resstr = 'nicht ermittelbar'
_logger.info('Contract_client_id_change : C_' + str(self.id) + ' - '+str(cidstr)+' vs. '+str(self.contract_auto_id)+'/'+str(resstr)+' vs '+str(self.contract_auto_name));
- if not self.contract_auto_name:
- self.contract_auto_name = resstr
- if not self.contract_auto_id:
- self.contract_auto_id = cidstr
- if self.contract_auto_id == "":
- self.contract_auto_id = cidstr
+ self.contract_auto_name = resstr
+ self.contract_auto_id = cidstr
@api.onchange('project_id')
def _onchange_project_id(self):
@@ -263,16 +302,12 @@ class dsscontracts(models.Model):
else:
cname = record.contract_name
resstr = "%s%s %s" % (record.project_id,record.client_id,cname)
- cidstr = "%s%s" % (record.project_id,record.client_id)
+ cidstr: String = "%s%s" % (record.project_id,record.client_id)
if resstr is None :
resstr = 'nicht ermittelbar'
_logger.info('project_id_change : C_' + str(self.id) + ' - '+str(cidstr)+' vs. '+str(self.contract_auto_id)+'/'+str(resstr)+' vs '+str(self.contract_auto_name));
- if not self.contract_auto_name:
- self.contract_auto_name = resstr
- if not self.contract_auto_id:
- self.contract_auto_id = cidstr
- if self.contract_auto_id == "":
- self.contract_auto_id = cidstr
+ self.contract_auto_name = resstr
+ self.contract_auto_id = cidstr
@api.onchange('contract_name')
def _onchange_contract_name(self):
@@ -319,6 +354,66 @@ class dsscontracts(models.Model):
# contract=super().create(vals)
# return contract
+ def action_info_click_ls(self):
+ view=self.env.ref("DigitalSignage.dss_contracts_infofeld_form")
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_mode': 'form' ,
+ 'view_id': view.id,
+ 'res_model': 'dss.contracts' ,
+ 'target': 'new' ,
+ 'display_name' : 'Text ändern',
+ 'views':[(view.id,'form')],
+ 'res_id': self.id ,
+ }
+
+ def action_state_click_ls(self):
+ view=self.env.ref("DigitalSignage.dss_contracts_viewstate_ls_form")
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_mode': 'form' ,
+ 'view_id': view.id,
+ 'res_model': 'dss.contracts' ,
+ 'target': 'new' ,
+ 'display_name' : 'Status ändern',
+ 'views':[(view.id,'form')],
+ 'res_id': self.id ,
+ }
+
+ def action_state_click_lws(self):
+ view=self.env.ref("DigitalSignage.dss_contracts_viewstate_lws_form")
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_mode': 'form' ,
+ 'view_id': view.id,
+ 'res_model': 'dss.contracts' ,
+ 'target': 'new' ,
+ 'display_name' : 'Status ändern',
+ 'views':[(view.id,'form')],
+ 'res_id': self.id ,
+ }
+
+ def action_state_click_lds(self):
+ view=self.env.ref("DigitalSignage.dss_contracts_viewstate_lds_form")
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_mode': 'form' ,
+ 'view_id': view.id,
+ 'res_model': 'dss.contracts' ,
+ 'target': 'new' ,
+ 'display_name' : 'Status ändern',
+ 'views':[(view.id,'form')],
+ 'res_id': self.id ,
+ }
+
+# return {
+ #'effect':{
+# 'fadeout':'slow',
+# 'message':'Status Geklickt',
+# 'type':'rainbow_man',
+# }
+# }
+ ''
@api.model
def _default_uuid(self):
@@ -387,7 +482,7 @@ class dsscontracts(models.Model):
return ""
def pyaction_dss_contract_addcalendar_rf_runtime(self):
- self.runtime_calendar_event.unlink()
+ self.env['calendar.event'].search([('id', '=', self.runtime_calendar_event)]).unlink()
event = {
'start': self.runtime_finish.strftime('%Y-%m-%d %H:%M:%S'),
'stop': self.runtime_finish.strftime('%Y-%m-%d %H:%M:%S'),
@@ -416,9 +511,9 @@ class dsscontracts(models.Model):
else:
addyears = 0
addmonths = bmonths - (addyears*12)
- enddatum = startdatum + relativedelta(months=addmonths,years=addyears)
+ enddatum = startdatum + relativedelta(months=addmonths,years=addyears,days=-1)
elif self.runtimesystem == "T":
- enddatum = startdatum + relativedelta(days=self.runtime_t)
+ enddatum = startdatum + relativedelta(days=self.runtime_t-1)
enddatum = enddatum + relativedelta(days=self.runtime_t)
elif self.runtimesystem == "E":
enddatum = startdatum
@@ -433,6 +528,67 @@ class dsscontracts(models.Model):
self.runtime_finish = enddatum
+ if not self.contract_cancel_mon:
+ cancelmonths = 0
+ else:
+ cancelmonths = self.contract_cancel_mon
+
+ self.contract_cancel_date = enddatum - relativedelta(months=cancelmonths)
+
+ def pyaction_dss_contract_calc_runtime_end(self):
+ if self.start_date:
+ startdatum= self.start_date
+ else:
+ startdatum= self.contract_date
+ if startdatum :
+ if self.runtimesystem == "M":
+ bmonths = 0
+ bmonths = self.runtime_m+self.runtime_bonus_m
+ if bmonths > 12:
+ addyears=bmonths//12
+ else:
+ addyears = 0
+ addmonths = bmonths - (addyears*12)
+ enddatum = startdatum + relativedelta(months=addmonths,years=addyears,days=-1)
+ elif self.runtimesystem == "T":
+ enddatum = startdatum + relativedelta(days=self.runtime_t-1)
+ enddatum = enddatum + relativedelta(days=self.runtime_t)
+ elif self.runtimesystem == "E":
+ enddatum = startdatum
+ if self.runtime_events:
+ mydate = startdatum
+ for event in self.runtime_events:
+ if mydate.strftime('%Y-%m-%d') < event.eventenddate.strftime('%Y-%m-%d'):
+ mydate = event.eventenddate
+ enddatum = mydate
+ elif self.runtimesystem == "S":
+ enddatum = self.runtime_finish
+
+ if not self.contract_cancel_mon:
+ cancelmonths = 0
+ else:
+ cancelmonths = self.contract_cancel_mon
+
+ self.contract_cancel_date = enddatum - relativedelta(months=cancelmonths)
+
+ def pyopen_provadd_form_view(self):
+ _logger.info('Open Add Provision')
+ view=self.env.ref("DigitalSignage.dss_provision_add_form")
+ self._compute_prov()
+ provision=self.env['dss.provision'].create({'contract':self.id,'provisiongesamt':self.provisionsteilcalc,'paydate':date.today()})
+ _logger.info('Open Add Provision '+str(view)+' P_'+str(provision))
+# self.provisions=[(4,[provision.id])]
+ return {
+ 'type': 'ir.actions.act_window',
+ 'view_mode': 'form' ,
+ 'view_type': 'form' ,
+ 'view_id': view.id,
+ 'res_model': 'dss.provision' ,
+ 'target': 'new' ,
+ 'display_name' : 'Neu Provision einfügen',
+ 'views':[(view.id,'form')],
+ 'res_id':provision.id,
+ }
def tokampagne(self):
_logger.info('Click auf Werbekampagne : C_'+str(self.id)+' - prüfe Letzte Aktuelle Kampagne')
@@ -458,7 +614,7 @@ class dsscontracts(models.Model):
resstr = 'nicht ermittelbar'
_logger.debug(resstr)
self.contract_auto_name = resstr
- _logger.info('Click auf Werbekampagne : C_' + str(self.id) + 'A_' + str(ds.id) + ' - Contract_autoname gesetzt '+ str(resstr)+'/'+str(cidstr))
+ _logger.info('Click auf Werbekampagne : C_' + str(self.id) + 'A_' + str(ds.id) + ' - Contract_autoname gesetzt '+ str(resstr))
# prüfen ob LetzteKampagne medien hat
_logger.info('Click auf Werbekampagne : C_'+str(self.id)+'A_'+str(ds.id)+' - Suche evtl Medien der Kampagne ')
medias = self.env['dss.mediarelations'].search(['&',('contract','=',self.id),('ad','=',ds.id)])
@@ -696,3 +852,33 @@ class dsscontracts(models.Model):
for record in self:
self.tokampagne()
+
+ def trigger_action_init(self):
+ _logger.info('Contract Cron Running - Start '+str(self)+' start searching')
+ jobcontracts = self.env['dss.contracts'].search([('run_trigger', '=', True)])
+ _logger.info('Contract Cron Running - Start '+str(self)+' Found '+str(len(jobcontracts))+' Trigger')
+ for record in jobcontracts:
+ if not record:
+ _logger.info('Contract Cron Running No Contracts Delivered - searching')
+ else:
+ _logger.info('Contract Cron Running C_' + str(record.id) + ' Check Trigger')
+ if record.run_trigger:
+ _logger.info('Contract Cron Running C_' + str(record.id) + ' Prüfe Triggergruppe TG_' + str(record.triggergroup.id))
+ triggergroup = record.triggergroup
+ if not triggergroup:
+ _logger.info('Contract Cron Running C_' + str(record.id) + ' Triggergruppe TG_' + str(record.triggergroup.id)+' not found')
+ pass
+ else:
+ for trigger in triggergroup.triggers:
+ if not trigger:
+ pass
+ else:
+ _logger.info('Contract Cron Running C_' + str(record) + ' in Triggergruppe TG_' + str(record.triggergroup)+' Prüfe Trigger T_'+str(trigger))
+ if trigger._check_trigger():
+ _logger.info('Contract Cron Trigger T_' + str(trigger)+' executed ')
+
+
+ _logger.info('Contract Cron Running - Finished')
+
+
+
diff --git a/models/dss_dialog.py b/models/dss_dialog.py
old mode 100644
new mode 100755
diff --git a/models/dss_eventdays.py b/models/dss_eventdays.py
old mode 100644
new mode 100755
diff --git a/models/dss_geraetetypen.py b/models/dss_geraetetypen.py
old mode 100644
new mode 100755
diff --git a/models/dss_maintains.py b/models/dss_maintains.py
new file mode 100755
index 0000000..b272c8c
--- /dev/null
+++ b/models/dss_maintains.py
@@ -0,0 +1,62 @@
+
+import ast
+import datetime
+import json
+import re
+import uuid
+import logging
+import base64
+import subprocess
+import tempfile
+import easywebdav
+import os
+import os.path
+
+
+from odoo import api, fields, models, _
+from odoo import tools
+from odoo.exceptions import ValidationError
+from datetime import date
+from datetime import datetime
+from dateutil.relativedelta import relativedelta
+from pyffmpeg import FFmpeg
+from tuya_iot import TuyaOpenAPI, TUYA_LOGGER
+from tuya_connector import TuyaOpenAPI, TUYA_LOGGER
+import sys
+
+TUYA_LOGGER.setLevel(logging.DEBUG)
+_logger = logging.getLogger(__name__)
+
+
+class dssmaintaintasks(models.Model):
+ _name = "dss.maintaintask"
+ _description = "DigitalSignage Wartungsarbeiten"
+ _inherit = ['mail.thread','mail.activity.mixin']
+ _rec_name = "maintainname"
+ maintainname = fields.Char('Einsatzbezeichnung', required=True)
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+
+ date_create = fields.Date('Erstellungsdatum', default=lambda self: self._default_create_date())
+ date_lastedit = fields.Date('Änderungsdatum')
+ user_create = fields.Char('Erstellungsuser', default=lambda self: self._default_create_user())
+ user_lastedit = fields.Char('Änderungsuser')
+
+ maintaintyp = fields.Selection([('AUF','Aufbau/Installation'),('REP','Reparatur'), ('CLE','Reinigung'), ('INS','Inspektion'), ('ABB','Demontage'), ('SON','Sonstiges')],tracking=True)
+ project = fields.Many2one('dss.projects', required=True, tracking=True)
+ maintain_tag = fields.Date('Einsatztag', required=True, tracking=True)
+ maintain_time_h = fields.Integer('Einsatzzeit Volle Stunden', required=True, tracking=True)
+ maintain_time_m = fields.Integer('Einsatzzeit Rest Minuten', required=True, tracking=True)
+# maintain_people = fields.Many2many('res.partner','Einsatzbeteiligte', domain="['&',('dssinternpartner','=',True),('dssinternpartner_grafik','=',True)]", tracking=True,required=True)
+ maintain_reason = fields.Char('Einsatzgrund', required=True)
+ maintain_text = fields.Html('Einsatzablauf', required=True)
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
+
+ def _default_create_date(self):
+ return str(date.today())
+
+ def _default_create_user(self):
+ return str(self.env.user.name)
+
diff --git a/models/dss_projects.py b/models/dss_projects.py
old mode 100644
new mode 100755
index 43adde6..8f59245
--- a/models/dss_projects.py
+++ b/models/dss_projects.py
@@ -7,6 +7,8 @@ import logging
import base64
import subprocess
import tempfile
+#from symbol import return_stmt
+
import easywebdav
import os
import os.path
@@ -47,6 +49,7 @@ class dssprojects(models.Model):
description = fields.Text('Beschreibung',tracking=True)
systemname = fields.Many2one('dss.systems',tracking=True)
grundsystemname = fields.Many2one('dss.systemtypen',tracking=True)
+ grundsystem_typ = fields.Char(related='grundsystemname.kennung',tracking=True)
grundsystemicon = fields.Image(related='grundsystemname.icon',tracking=True)
grundsystem_default_adstructure = fields.Many2one(related='grundsystemname.default_adstructure',tracking=True)
grundsystemicon5050 = fields.Image(related='grundsystemname.icon_5050')
@@ -68,14 +71,22 @@ class dssprojects(models.Model):
vertragsschreiber = fields.Many2one('res.partner',domain="['&',('dsspartner','=',True),('dsspartner_vertrieb','=',True)]",tracking=True)
standortpartner = fields.Many2one('res.partner',domain="['&',('dsspartner','=',True),('dsspartner_standort','=',True)]",tracking=True)
vertriebspartner = fields.Many2many('res.partner',domain="['&',('dsspartner','=',True),('dsspartner_vertrieb','=',True)]",tracking=True)
+ project_grafiker = fields.Many2one('res.partner',domain="['&',('dssinternpartner','=',True),('dssinternpartner_grafik','=',True)]",tracking=True)
zeiten_on = fields.Datetime('veraltet',tracking=True)
zeiten_off = fields.Datetime('veraltet',tracking=True)
zeitenf_on = fields.Float('Einschaltzeit',tracking=True)
zeitenf_off = fields.Float('Ausschaltzeit',tracking=True)
errichtet_am = fields.Date('Errichtungstag',tracking=True)
- standort_inout = fields.Selection([('indoor','im Gebäude'), ('outdoor','Aussenbereich'), ('semiindoor','Überdachter Aussenbereich'),('Divers','Andere Art')],tracking=True);
+ standort_inout = fields.Selection([('indoor','im Gebäude'), ('outdoor','Aussenbereich'), ('semiindoor','Überdachter Aussenbereich'),('Divers','Andere Art')],tracking=True)
+
+ date_start_compute = fields.Date('Kalender Start intern',compute='_compute_calendar_start')
cloudlink = fields.Char('Cloud Verzeichnis',help='Verzeichnis für das Project innerhalb des Cloud Struktur')
+ maintains = fields.Many2many('dss.maintaintask',string="Einsätze am Projekt")
+
+ @api.model
+ def _compute_calendar_start(self):
+ self.date_start_compute=date.today()
@api.model
def _default_uuid(self):
@@ -102,7 +113,7 @@ class dssprojects(models.Model):
# context = "{
# 'create': self.active,
# 'active_test': self.active
-# }
+# }7
# action['context'] = context
# action['domain'] = domain
# return action
diff --git a/models/dss_provision.py b/models/dss_provision.py
new file mode 100644
index 0000000..a6f09f1
--- /dev/null
+++ b/models/dss_provision.py
@@ -0,0 +1,91 @@
+# -*- coding: utf-8 -*
+
+import ast
+import datetime
+import json
+import re
+import uuid
+import logging
+import base64
+import subprocess
+import tempfile
+import easywebdav
+import os
+import os.path
+
+
+from odoo import api, fields, models, _
+from odoo import tools
+from . import dss_settings
+from odoo.exceptions import ValidationError
+from datetime import date
+from datetime import datetime
+from dateutil.relativedelta import relativedelta
+from pyffmpeg import FFmpeg
+from tuya_iot import TuyaOpenAPI, TUYA_LOGGER
+from tuya_connector import TuyaOpenAPI, TUYA_LOGGER
+import sys
+
+TUYA_LOGGER.setLevel(logging.DEBUG)
+_logger = logging.getLogger(__name__)
+
+class dssprovision(models.Model):
+
+ _name = "dss.provision"
+ _description = "DigitalSignage Provisionen"
+ _rec_name = "provisionname"
+ _inherit = ['mail.thread','mail.activity.mixin','dss.triggermodel']
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+ date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
+ user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
+ provisionname = fields.Char('Kurzbezeichnung der Provision',tracking=True)
+ contract = fields.Many2one('dss.contracts' , string='Vertrag', store=True,tracking=True)
+ project = fields.Many2one(related='contract.project' , string='Project', store=True,tracking=True)
+ client = fields.Many2one(related='contract.client',string="Kunde (wenn angelegt)")
+ contract_writer = fields.Many2one(related='contract.contract_writer',string="Vertragsschreiber", store=True)
+ provisionbase = fields.Selection([('VZAHL','Volle Zahlung'),('TZAHL','Teil Betrag FIX'),('PZAHL','Teil Betrag Prozent')], tracking=True, store=True)
+ provisiongesamt = fields.Float('Gesamtsumme Provision', tracking=True)
+ provisionprozent = fields.Float('Ausgezahlte Prozent',tracking=True)
+ provisionteil = fields.Float('Auszahlungs-Summe', tracking=True)
+
+ paydate = fields.Date('Zahlungsdatum', tracking=True)
+
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
+
+ def _default_create_date(self):
+ return str(date.today())
+
+ def _default_create_user(self):
+ return str(self.env.user.name)
+
+
+ @api.onchange('provisionbase')
+ def _onchange_provisionbase(self):
+ if self.provisionbase == "PZAHL":
+ _logger.info('Calc Provisionpart : ' + str(self.provisionbase) + ' \% ' + str(self.provisionprozent));
+ try:
+ self.provisionteil = self.provisiongesamt/100*self.provisionprozent
+ except:
+ self.provisionteil = 0
+ elif self.provisionbase == "VZAHL":
+ _logger.info('Calc Provisionpart : ' + str(self.provisionbase) + ' \V ' + str(self.provisiongesamt));
+ self.provisionteil = self.provisiongesamt
+ self.provisionprozent = 100
+ else:
+ _logger.info('Calc Provisionpart : ' + str(self.provisionbase) + ' \T ' + str(self.provisionteil));
+ self.provisionteil = self.provisionteil
+
+ @api.onchange('provisionprozent')
+ def _onchange_provisionprozent(self):
+ self._onchange_provisionbase()
+
+ def pyadd_entry_tocontract(self):
+ self._onchange_provisionbase()
+ self.contract.provisions=[(4,self.id)]
+ self.contract._compute_provisionspayedpercent()
+
+ def pycancel_entry(self):
+ self.unlink()
diff --git a/models/dss_provisionstypen.py b/models/dss_provisionstypen.py
new file mode 100644
index 0000000..43e0fcf
--- /dev/null
+++ b/models/dss_provisionstypen.py
@@ -0,0 +1,47 @@
+# -*- coding: utf-8 -*
+
+import ast
+import datetime
+import json
+import re
+import uuid
+import logging
+import base64
+import subprocess
+import tempfile
+import easywebdav
+import os
+import os.path
+
+
+from odoo import api, fields, models, _
+from odoo import tools
+from . import dss_settings
+from odoo.exceptions import ValidationError
+from datetime import date
+from datetime import datetime
+from dateutil.relativedelta import relativedelta
+from pyffmpeg import FFmpeg
+from tuya_iot import TuyaOpenAPI, TUYA_LOGGER
+from tuya_connector import TuyaOpenAPI, TUYA_LOGGER
+import sys
+
+TUYA_LOGGER.setLevel(logging.DEBUG)
+_logger = logging.getLogger(__name__)
+
+class dssprovisionstypen(models.Model):
+
+ _name = "dss.provisionstypen"
+ _description = "DigitalSignage Provisionen"
+ _rec_name = "provisionstyp"
+ _inherit = ['mail.thread','mail.activity.mixin','dss.triggermodel']
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+ provisionbase = fields.Selection([('VSUM','Vertragssumme'),('FSUM','Fester Betrag')], tracking=True)
+ provisionstyp = fields.Char('Kurzbezeichnung der Provisions Art',tracking=True)
+ provisionprozent = fields.Float('Prozent der Basis',tracking=True)
+ provisioncalc = fields.Char('Berechnung',help="Beispiel : =%10(B-F(300)) ist 10% der (Basissumme - 300€) oder =%5(B)-F(200) ist 5% der Basissumme und dann minus 200€ )",tracking=True)
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
+
diff --git a/models/dss_settings.py b/models/dss_settings.py
old mode 100644
new mode 100755
index fa7ddff..57a1a98
--- a/models/dss_settings.py
+++ b/models/dss_settings.py
@@ -48,6 +48,8 @@ class dssSettings(models.Model):
show_expand_contract_group_ad_state = fields.Boolean('Jeden Kampagnen-Status anzeigen ?',help='im Groupierungsmodus der Verträge nach Kampagnenstatus alle (auch unbenutzte) Statusspalten anzeigen ?',tracking=True)
show_expand_contract_group_ad_work_state = fields.Boolean('Jeden Kampagnen-Arbeitsstatus anzeigen ?',help='im Groupierungsmodus der Verträge nach Kampagnen-Arbeitsstatus alle (auch unbenutzte) Statusspalten anzeigen ?',tracking=True)
show_expand_contract_group_ad_todo_state = fields.Boolean('Jeden Kampagnen-Aufgabenschritt anzeigen ?',help='im Groupierungsmodus der Verträge nach Kampagnen-Aufgabenschritt alle (auch unbenutzte) Aufgabenspalten anzeigen ?',tracking=True)
+ freigabe_auto_time = fields.Integer('Auto Freigabetage',help='Automatische Freigabe nach x Tagen',tracking=True)
+
def _get_settingvalue(self,valuename):
settings = (self.env['dss.settings'].search([],limit=1))
wert = settings._origin.read([valuename])[0][valuename]
diff --git a/models/dss_software.py b/models/dss_software.py
old mode 100644
new mode 100755
diff --git a/models/dss_systems.py b/models/dss_systems.py
old mode 100644
new mode 100755
index 33a037a..aafbfc2
--- a/models/dss_systems.py
+++ b/models/dss_systems.py
@@ -42,7 +42,7 @@ class dsssystems(models.Model):
standort = fields.Char('Hauptstandort des Systems',tracking=True)
- farbe = fields.Integer('Grundfarbe')
+ farbe = fields.Char('Grundfarbe')
has_heizung = fields.Boolean('Mit Heizsystem')
has_klima = fields.Boolean('Mit Klimasystem')
diff --git a/models/dss_systemtypen.py b/models/dss_systemtypen.py
old mode 100644
new mode 100755
index 35952ca..637cf3b
--- a/models/dss_systemtypen.py
+++ b/models/dss_systemtypen.py
@@ -40,8 +40,13 @@ class dsssystemtypen(models.Model):
farbe = fields.Char('Grundfarbe')
icon = fields.Image()
icon_5050 = fields.Image("Icon 50",compute='_compute_icon_5050')
- default_adstructure = fields.Many2one('dss.adstructures',String='Standard-Werbeaufbau')
- default_cloud_path = fields.Char('Standard Cloud Path')
+ default_adstructure = fields.Many2one('dss.adstructures',String='Standard-Werbeaufbau', tracking=True)
+ default_cloud_path = fields.Char('Standard Cloud Path', tracking=True)
+ email_template_welcome = fields.Many2one('mail.template',string='Mailvorlage für Willkommen', tracking=True)
+ email_template_zuarbeit = fields.Many2one('mail.template',string='Mailvorlage für Zuarbeiterrinnerung', tracking=True)
+ email_template_abzug = fields.Many2one('mail.template',string='Mailvorlage für Korrekturabzug', tracking=True)
+ email_template_start = fields.Many2one('mail.template',string='Mailvorlage für Auslieferung', tracking=True)
+
@api.model
def _default_uuid(self):
return str(uuid.uuid4())
diff --git a/models/dss_trigger.py b/models/dss_trigger.py
new file mode 100644
index 0000000..e05a996
--- /dev/null
+++ b/models/dss_trigger.py
@@ -0,0 +1,135 @@
+import uuid
+import logging
+
+from odoo import api, fields, models, _
+from odoo import tools
+from odoo.exceptions import ValidationError
+import sys
+
+_logger = logging.getLogger(__name__)
+
+class dsstriggergroups(models.Model):
+ _name = "dss.triggergroups"
+ _description = "DigitalSignage Trigger Gruppen"
+ _inherit = ['mail.thread','mail.activity.mixin']
+ _rec_name = "triggergroupname"
+# _inherit = ['mail.thread', 'mail.activity.mixin']
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+ date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
+ date_lastedit = fields.Date('Änderungsdatum')
+ user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
+ user_lastedit = fields.Char('Änderungsuser')
+
+ triggergroupname = fields.Char('Trigger Gruppenname',tracking=True)
+ trigger_table = fields.Many2one('ir.model','Nutzbar in',help='Für welche Daten ist dieser Tigger gültig ?',track_visibility='onchange',tracking=True)
+ triggers = fields.Many2many('dss.triggertypes',string='Enthaltene Trigger',tracking=True)
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
+
+ def _default_create_date(self):
+ return str(date.today())
+
+ def _default_create_user(self):
+ return str(self.env.user.name)
+
+
+class dsstriggertypes(models.Model):
+ _name = "dss.triggertypes"
+ _description = "DigitalSignage Trigger Typen"
+ _inherit = ['mail.thread','mail.activity.mixin']
+ _rec_name = "triggername"
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+ date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
+ date_lastedit = fields.Date('Änderungsdatum')
+ user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
+ user_lastedit = fields.Char('Änderungsuser')
+
+ triggername = fields.Char('Triggername',tracking=True)
+ trigger_active = fields.Boolean('Trigger Aktiv ?',tracking=True)
+ triggertyp = fields.Selection([('FIELD_A','Feldänderung allgemein'),('FIELD_S','Feldänderung spezifisch'),('FIELD_K','Feldwert berechnet'),('MANUAL','Menuell ausgelösst')],'Triggertyp',tracking=True)
+ trigger_table = fields.Many2one('ir.model','Nutzbar in',help='Für welche Daten ist dieser Tigger gültig ?',tracking=True)
+ trigger_field = fields.Many2one('ir.model.fields', help='Feldname in der Tabelle',tracking=True)
+ trigger_operator = fields.Selection([('NGLEICH','Feld gleich dem Wert'),('GGLEICH','Feld größer/gleich dem Wert'),('KGLEICH','Feld kleiner/gleich dem Wert'),('UGLEICH','Feld ungleich dem Wert'),('ENTHAELT','Feld enthällt dem Wert')],'Feldwertoperator',tracking=True)
+ trigger_init_trigger = fields.Boolean('Ausl. bei 1. Änderung ?',help='Soll der Trigger bereits beim setzen des 1. Wertes auslösen?',tracking=True)
+ trigger_value = fields.Char('Feldwert für Trigger',tracking=True)
+ trigger_old_value = fields.Char('Letzter-Feldwert für Trigger',tracking=True)
+ trigger_aktionen = fields.Many2many('ir.actions.server',string='Aktionen bei auslösen des Triggers',tracking=True)
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
+
+ def _default_create_date(self):
+ return str(date.today())
+
+ def _default_create_user(self):
+ return str(self.env.user.name)
+
+ def _check_trigger(self):
+ for record in self:
+ _logger.info('Trigger Check TR_' + str(record)+' Begin')
+ if record.trigger_active:
+ _logger.info('Trigger Check TR_' + str(record) + ' Check Field F_'+str(record.trigger_field.name)+' of model M_'+str(record.trigger_table.model)+' Feld-Typ :'+str(record.trigger_field.ttype))
+ if record.trigger_field.ttype in ['char','text','html']:
+ val = self.env[record.trigger_table.model]._origin.read([record.trigger_field.name])
+ elif record.trigger_field.ttype in ['selection']:
+ val = self.env[record.trigger_table.model]._origin.read([record.trigger_field.name])[0]
+ elif record.trigger_field.ttype in ['one2many']:
+ val = self.env[record.trigger_table.model]._origin.read([record.trigger_field.name]).ids
+ elif record.trigger_field.ttype in ['many2one']:
+ val = self.env[record.trigger_table.model]._origin.read([record.trigger_field.name])
+ else:
+ val = False
+ record.trigger_old_value = str(val)
+ _logger.info('Trigger Check TR_' + str(record)+' oldValue '+str(val))
+ _logger.info('Trigger Check TR_' + str(record)+' Ende')
+ return ""
+
+class dsstriggeractions(models.Model):
+ _name = "dss.triggeractions"
+ _description = "DigitalSignage Trigger Aktionen"
+ _inherit = ['mail.thread','mail.activity.mixin']
+ _rec_name = "triggeractionname"
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+ date_create = fields.Date('Erstellungsdatum',default=lambda self: self._default_create_date())
+ date_lastedit = fields.Date('Änderungsdatum')
+ user_create = fields.Char('Erstellungsuser',default=lambda self: self._default_create_user())
+ user_lastedit = fields.Char('Änderungsuser')
+
+ triggeractionname = fields.Char('Triggeraktionname',track_visibility='onchange',tracking=True)
+ triggeractiontyp = fields.Selection([('EMAIL','EMail senden'),('ACTIV','Aufgabe erzeugen'),('DSEDIT','Datensatz ändern'),('TUYA','Tuya Aktion auslössen')],'Aktionstyp',tracking=True)
+ email_getfrom = fields.Selection([('FIX','Email fest hinterlegt'),('DATA','Email in Feld hinterlegt')],'Emailadresse von',tracking=True)
+ email_fix_email_sender = fields.Many2one('res.partner','Absender',help='Email Sender Adresse der Aktion',tracking=True)
+ email_fix_email = fields.Many2one('res.partner','Empfänger',help='Email Emüfänger Adresse für Benachrichtigung/Triggeraktion',tracking=True)
+ email_data_table = fields.Many2one('ir.model','Nutzbar in',help='Für welche Daten ist diese Aktion gültig ?',tracking=True)
+ email_data_field = fields.Many2one('ir.model.fields', help='Feldname in der Tabelle',tracking=True)
+ email_template = fields.Many2one('mail.template','Emailvorlage',tracking=True)
+
+ field = fields.Char('Feldname',help='Feldname in o.g. Tebelle.',tracking=True)
+ type_s = fields.Char('Feldwertoperator',tracking=True)
+ value_s = fields.Char('Feldwert für Trigger',tracking=True)
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
+
+ def _default_create_date(self):
+ return str(date.today())
+
+ def _default_create_user(self):
+ return str(self.env.user.name)
+
+
+class dsstriggermodel(models.Model):
+ _name = "dss.triggermodel"
+ _description = "DigitalSignage Trigger Model für Inherit"
+ _rec_name = "trigger_uuid"
+ trigger_uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+ run_trigger = fields.Boolean('Trigger aktiv',tracking=True)
+ triggergroup = fields.Many2one('dss.triggergroups',string='Triggergruppe',tracking=True,default=True)
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
diff --git a/models/dss_triggervalues.py b/models/dss_triggervalues.py
new file mode 100644
index 0000000..346b35d
--- /dev/null
+++ b/models/dss_triggervalues.py
@@ -0,0 +1,21 @@
+from odoo import api, fields, models, _
+from odoo import tools
+
+
+class dssadstatus(models.Model):
+ _name = "dss.triggervalues"
+ _description = "DigitalSignage Trigger Werte"
+# _rec_name = "statusname"
+# _inherit = ['mail.thread', 'mail.activity.mixin']
+ uuid = fields.Char(default=lambda self: self._default_uuid(), required=True, readonly=True, copy=False, string='UUID')
+# uuid = fields.Char('UUID', required=True, translate=True)
+ valuename = fields.Char('Wertname', required=True)
+ value_old_bool = fields.Boolean('Alter Boolean_Wert', required=True)
+ value_old_string = fields.Char('Alter String_Wert', required=True)
+ value_old_integer = fields.Integer('Alter Integer_Wert', required=True)
+ icon = fields.Image()
+ order = fields.Integer('Reihenfolge')
+
+ @api.model
+ def _default_uuid(self):
+ return str(uuid.uuid4())
diff --git a/odoolog b/odoolog
new file mode 120000
index 0000000..217f935
--- /dev/null
+++ b/odoolog
@@ -0,0 +1 @@
+/var/log/odoo
\ No newline at end of file
diff --git a/security/groups.xml b/security/groups.xml
index 3119259..54cfb2f 100755
--- a/security/groups.xml
+++ b/security/groups.xml
@@ -22,4 +22,17 @@
+
+
+ Systemtechniker
+
+
+
+
+
+ Provisionsinformationen
+
+
+
+
\ No newline at end of file
diff --git a/security/ir.model.access.csv b/security/ir.model.access.csv
index 855c8b9..3f5b4a6 100755
--- a/security/ir.model.access.csv
+++ b/security/ir.model.access.csv
@@ -25,3 +25,6 @@ digitalsignage_dss_screendesign_group_user,access.dss.screendesign,model_dss_scr
digitalsignage_dss_triggertypes_group_user,access.dss.triggertypes,model_dss_triggertypes,base.group_user,1,1,1,1
digitalsignage_dss_triggeractions_group_user,access.dss.triggeractions,model_dss_triggeractions,base.group_user,1,1,1,1
digitalsignage_dss_triggergroups_group_user,access.dss.triggergroups,model_dss_triggergroups,base.group_user,1,1,1,1
+digitalsignage_dss_maintaintask_group_user,access.dss.maintaintask,model_dss_maintaintask,base.group_user,1,1,1,1
+digitalsignage_dss_provision_group_user,access.dss.provision,model_dss_provision,base.group_user,1,1,1,1
+digitalsignage_dss_provisionstypen_group_user,access.dss.provisionstypen,model_dss_provisionstypen,DigitalSignage.group_prov_user,1,1,1,1
diff --git a/static/src/css/dss.css b/static/src/css/dss.css
index 08339bf..70c2865 100755
--- a/static/src/css/dss.css
+++ b/static/src/css/dss.css
@@ -34,6 +34,22 @@ width:90px;
.o_FormRenderer_chatterContainer.oe_chatter.o-aside
{
- max-width: 20%; // The width in pixels
+ max-width: 30%; // The width in pixels
+
+}
+
+.my_prov_class [data-id="provisionname"] {
+
+ width:330px !important;
+
+ min-width: 130px !important;
+
+}
+
+.my_prov_class [data-id="Sl No"] {
+
+ width:130px !important;
+
+ min-width: 30px !important;
}
\ No newline at end of file
diff --git a/static/src/helpers/img/pat-blue.jpg b/static/src/helpers/img/pat-blue.jpg
new file mode 100644
index 0000000..3a553ff
Binary files /dev/null and b/static/src/helpers/img/pat-blue.jpg differ
diff --git a/static/src/helpers/img/pat-orange.jpg b/static/src/helpers/img/pat-orange.jpg
new file mode 100644
index 0000000..a9c5e78
Binary files /dev/null and b/static/src/helpers/img/pat-orange.jpg differ
diff --git a/static/src/helpers/img/pat-red.jpg b/static/src/helpers/img/pat-red.jpg
new file mode 100644
index 0000000..23143dd
Binary files /dev/null and b/static/src/helpers/img/pat-red.jpg differ
diff --git a/static/src/helpers/img/pat-yellow.jpg b/static/src/helpers/img/pat-yellow.jpg
new file mode 100644
index 0000000..2554bda
Binary files /dev/null and b/static/src/helpers/img/pat-yellow.jpg differ
diff --git a/static/src/helpers/v2/app.js b/static/src/helpers/v2/app.js
new file mode 100644
index 0000000..9c74f49
--- /dev/null
+++ b/static/src/helpers/v2/app.js
@@ -0,0 +1 @@
+!function(){"use strict";var t={572:function(t,e,n){n.r(e),e.default='
\r\n\r\n \r\n\r\n \r\n\r\n
\r\n
\r\n
\r\n
\r\n
\r\n
Download DayPilot Lite for JavaScript (open-source)
\r\n
See also the advanced features of
DayPilot Pro (commercial).
\r\n
\r\n
\r\n
\r\n
\r\n\r\n
\r\n \r\n\r\n
\r\n'},722:function(t,e){Object.defineProperty(e,"t",{value:!0}),e.i=void 0;var n=function(){function t(){}return t.o=navigator&&navigator.userAgent&&(-1!==navigator.userAgent.indexOf("MSIE")||-1!==navigator.userAgent.indexOf("Trident")),t}();e.i=n},143:function(t,e){Object.defineProperty(e,"t",{value:!0}),e.u=void 0;var n=function(){function t(){}return t.get=function(t){return document.getElementById(t)},t.query=function(t){return Array.apply(null,document.querySelectorAll(t))},t.l=function(t,e){return t.getElementsByClassName(e)[0]},t.create=function(t){return document.createElement(t)},t.h=function(){return document.createDocumentFragment()},t}();e.u=n},730:function(t,e){Object.defineProperty(e,"t",{value:!0}),e.v=void 0;var n=function(){function t(){}return t.prototype.m=function(){var t=this;document.addEventListener("DOMContentLoaded",(function(){Array.apply(null,document.querySelectorAll(".track-download")).forEach((function(e){e.addEventListener("click",(function(n){t.track(e,"/action/trialdownload")||n.preventDefault()}))}))}))},t.prototype.download=function(t){var e=document.createElement("a");e.href=t,e.download=t.split("/").pop(),document.body.appendChild(e),e.click(),document.body.removeChild(e)},t.prototype.track=function(t,e){var n=this,i=window.ga;return void 0!==i&&i("send","pageview",e),"_blank"===t.target||(setTimeout((function(){n.download(t.href)}),150),!1)},t}();e.v=n},447:function(t,e,n){var i=this&&this.p||function(t,e,n,i){return new(n||(n=Promise))((function(r,a){function o(t){try{c(i.next(t))}catch(t){a(t)}}function u(t){try{c(i.g(t))}catch(t){a(t)}}function c(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(o,u)}c((i=i.apply(t,e||[])).next())}))},r=this&&this.k||function(t,e){var n,i,r,a,o={label:0,O:function(){if(1&r[0])throw r[1];return r[1]},M:[],T:[]};return a={next:u(0),g:u(1),_:u(2)},"function"==typeof Symbol&&(a[Symbol.iterator]=function(){return this}),a;function u(u){return function(c){return function(u){if(n)throw new TypeError("Generator is already executing.");for(;a&&(a=0,u[0]&&(o=0)),o;)try{if(n=1,i&&(r=2&u[0]?i._:u[0]?i.g||((r=i._)&&r.call(i),0):i.next)&&!(r=r.call(i,u[1])).done)return r;switch(i=0,r&&(u=[2&u[0],r.value]),u[0]){case 0:case 1:r=u;break;case 4:return o.label++,{value:u[1],done:!1};case 5:o.label++,i=u[1],u=[0];continue;case 7:u=o.T.pop(),o.M.pop();continue;default:if(!(r=o.M,(r=r.length>0&&r[r.length-1])||6!==u[0]&&2!==u[0])){o=0;continue}if(3===u[0]&&(!r||u[1]>r[0]&&u[1]=o.scrollHeight?r.classList.add("no-gradient-bottom"):r.classList.remove("no-gradient-bottom"),0==o.scrollTop?r.classList.add("no-gradient-top"):r.classList.remove("no-gradient-top"))};c(),o.addEventListener("scroll",c),window.addEventListener("resize",c),new MutationObserver(c).observe(o,{childList:!0,subtree:!0}),o.addEventListener("mouseover",(function(e){var n=e.target;if("A"===n.tagName){var i=n.getAttribute("href");i&&t.H(i)}})),o.addEventListener("touchstart",(function(e){var n=e.target;if("A"===n.tagName){var i=n.getAttribute("href");i&&t.H(i)}})),i=o.querySelector(".active"),document.body.className="dpw-sidebar-left"}else document.body.className="dpw-no-sidebar";var s=u.u.l(n,"toolbar"),l=this.q();s.appendChild(l);var f=u.u.l(n,"placeholder");this.placeholder=f,f.appendChild(this.B(document)),d.i.o?f.appendChild(this.content):f.appendChild(this.content.content),new h.K(this.placeholder).Y(),document.body.insertBefore(n,document.body.firstChild),this.search.m(),i&&this.U(i),this.location.X&&!this.location.sandbox&&this.N.m()},t.prototype.B=function(t){var e=t.title.split("|")[0].trim(),n=t.createElement("h2");return n.className="title",n.innerHTML=e,n},t.prototype.U=function(t){var e=document.querySelector(".dp-menu");if(e){var n=e.getBoundingClientRect(),i=t.getBoundingClientRect();i.topn.bottom&&(e.scrollTop+=i.bottom-n.bottom+85)}},t.prototype.F=function(){var t=this,e=u.u.create("div");e.className="dp-menu";var n=u.u.create("ul"),i=this.location.Z;return this.R.getItems(i).forEach((function(e){var i=e.filename;"index.html"!==i||t.location.filesystem||(i="./");var r=t.$(e.text,i,e.filename===t.location.filename,e.tt);n.appendChild(r.div)})),e.appendChild(n),e},t.prototype.$=function(t,e,n,i){var r=this,a=u.u.create("li");if("string"==typeof e){var o=document.createElement("a");if(o.href=e,o.title=t,n&&(o.className="active"),this.D&&o.addEventListener("click",(function(t){window.innerWidth<=1e3||(t.preventDefault(),r.et(e))})),i){var c=u.u.create("span");c.innerText="NEW",c.className="new",o.appendChild(c)}var s=u.u.create("span");s.innerText=t,o.appendChild(s),a.appendChild(o)}else{var l=u.u.create("strong");l.innerText=t,a.appendChild(l)}return{div:a}},t.prototype.q=function(){var t=this,e=u.u.h(),n=this.location.W?"":"../",i=this.location.filesystem?"index.html":"",r=this.location.sandbox?"Sandbox":"Demo",a=this.nt(r,n+i,this.location.W);return e.appendChild(a.div),this.R.it().forEach((function(r){var a="".concat(n).concat(r.Z,"/").concat(i),o=r.Z===t.location.Z,u=t.nt(r.text,a,o);e.appendChild(u.div)})),e},t.prototype.nt=function(t,e,n){var i=u.u.create("div");i.className="dpw-header-item";var r=document.createElement("a");return r.href=e,r.innerText=t,n&&(r.className="dpw-header-item-selected"),i.appendChild(r),{div:i,a:r}},t.prototype.J=function(){var t=this;window.addEventListener("popstate",(function(e){var n=window.location.pathname,i=n.substring(n.lastIndexOf("/")+1);""!==i&&"index.html"!==i||(i="./"),t.et(i,!0)}))},t.prototype.H=function(t){return i(this,void 0,void 0,(function(){var e,n,i;return r(this,(function(r){switch(r.label){case 0:return this.P[t]?[2,this.P[t]]:[4,fetch(t)];case 1:return(e=r.O()).ok?[4,e.text()]:[2,null];case 2:return-1===(n=r.O()).indexOf("")?[2,null]:(i=new v(n),this.rt(i.ot),this.P[t]=i,[2,i])}}))}))},t.prototype.et=function(t,e){return i(this,void 0,void 0,(function(){var n,i,a,o,c,s,l,d,f,v,m;return r(this,(function(r){switch(r.label){case 0:return n=window.location.href,(i=this.P[t])?[3,2]:[4,this.H(t)];case 1:i=r.O(),r.label=2;case 2:return i?(a=i.ot,o=a.querySelector("#content").content,this.rt(a),o.querySelectorAll("script").forEach((function(t){var e=t.parentNode,n=document.createElement("script");n.innerHTML="(function() {".concat(t.innerHTML,"})();"),e.replaceChild(n,t)})),this.placeholder.innerHTML="",this.placeholder.appendChild(this.B(a)),this.placeholder.appendChild(o),(c=a.querySelector("script[src*='jquery']"))&&(s=c.getAttribute("src"))&&(document.querySelector('script[src="'.concat(s,'"]'))||((l=document.createElement("script")).setAttribute("src",s),document.head.appendChild(l))),new h.K(this.placeholder).Y(),d=a.querySelector("title"),document.title=d.innerText,e||window.history.pushState({},"",t),this.search.m(),f=u.u.l(document.body,"dp-menu"),(v=f.querySelector("a.active"))&&v.classList.remove("active"),(m=f.querySelector('a[href="'.concat(t,'"]')))&&(m.classList.add("active"),this.U(m)),this.ut({referrer:n}),[2]):[2]}}))}))},t.prototype.ut=function(t){var e=window.__sendPageView;e&&e(window.location.href,t)},t.prototype.rt=function(t){t.querySelectorAll("link[rel=stylesheet]").forEach((function(t){var e=t.getAttribute("href");if(e&&!document.querySelector('link[href="'.concat(e,'"]'))){var n=document.createElement("link");n.setAttribute("rel","stylesheet"),n.setAttribute("href",e),document.head.appendChild(n)}}))},t}();e.S=f;var v=function(){function t(t){this.ct=t}return Object.defineProperty(t.prototype,"ot",{get:function(){return(new DOMParser).parseFromString(this.ct,"text/html")},enumerable:!1,configurable:!0}),t}();e.j=v},11:function(t,e){Object.defineProperty(e,"t",{value:!0}),e.C=void 0;var n=function(){function t(t,e,n){this.st=["demo","demo2","sandbox","lite"],this.pathname=n,this.host=e,this.protocol=t,this.lt()}return Object.defineProperty(t.prototype,"W",{get:function(){return"ROOT"===this.Z},enumerable:!1,configurable:!0}),t.test=function(){},t.prototype.lt=function(){this.filename=this.ht(this.pathname),this.Z=this.dt(this.pathname),this.sandbox=this.ft(this.pathname),this.X=this.vt(this.host),this.filesystem=this.xt(this.protocol)},t.prototype.ht=function(t){var e=t.substring(t.lastIndexOf("/")+1);return""===e&&(e="index.html"),e},t.prototype.xt=function(t){return"file:"===t},t.prototype.dt=function(t){var e=t.lastIndexOf("/"),n=t.lastIndexOf("/",e-1),i=t.substring(n+1,e);return"/"===i&&(i="ROOT"),-1!==this.st.indexOf(i)&&(i="ROOT"),i},t.prototype.vt=function(t){return"javascript.daypilot.org"===t},t.prototype.ft=function(t){return 0===t.indexOf("/sandbox")},t.prototype.print=function(){window.console.log(this.pathname,this.Z,this.filename)},t}();e.C=n},463:function(t,e,n){Object.defineProperty(e,"t",{value:!0}),e.A=void 0;var i=n(143),r=function(){function t(){this.wt=null}return t.prototype.m=function(){var t=this,e=this.bt=i.u.get("search-box-input");e.addEventListener("keyup",(function(n){"Escape"!==n.key&&"Esc"!==n.key||(e.value=""),t.gt(e.value)})),i.u.get("search-box-clear").addEventListener("click",(function(n){t.yt(),e.focus()}))},t.prototype.yt=function(){this.bt.value="",this.gt("")},t.prototype.gt=function(t){var e=!t||""===t.trim();e?this.kt():this.Ot(),i.u.query(".menu li").forEach((function(n){var i=n.getElementsByTagName("a")[0],r=i&&-1!==i.innerText.toLowerCase().indexOf(t.toLowerCase()),a=e||r;n.style.display=a?"":"none"}))},t.prototype.Ot=function(){if(null==this.wt){var t=this.Mt();t&&(this.wt=t.offsetHeight,t.style.height=this.wt+"px")}},t.prototype.kt=function(){var t=this.Mt();this.wt=null,t&&(t.style.height="")},t.prototype.Mt=function(){return i.u.query(".dp-menu")[0]},t}();e.A=r},151:function(t,e,n){Object.defineProperty(e,"t",{value:!0}),e.K=void 0;var i=n(143),r=n(827),a=n(722),o=function(){function t(t){this.placeholder=t}return t.prototype.Y=function(){if(null!==this.Tt()){var t=this._t();this.placeholder.appendChild(t)}},t.prototype._t=function(){var t=this,e=i.u.create("div");e.className="space";var n=i.u.create("button");return n.className="button-source",n.innerText="Show source",n.onclick=function(){t.Et(),n.style.display="none"},e.appendChild(n),e},t.prototype.Et=function(){var t,e=this.Tt(),n=a.i.o?e.innerText:e.innerHTML,o=n;if(n.startsWith("(function() {")){var u=n.indexOf("(function() {"),c=n.lastIndexOf("})();");o=n.substring(u+13,c)}a.i.o?((t=i.u.create("div")).innerHTML=""+r.St.jt(o)+"
",this.placeholder.appendChild(t)):((t=i.u.create("pre")).innerText=r.St.jt(o),this.placeholder.appendChild(t))},t.prototype.Tt=function(){return this.placeholder.querySelector("script:not([src])")},t}();e.K=o},4:function(t,e){Object.defineProperty(e,"t",{value:!0}),e.L=void 0;var n=[{text:"Calendar",Z:"calendar",children:[{text:"Main"},{text:"JavaScript Event Calendar",filename:"index.html"},{text:"View Types"},{text:"Day View",filename:"day.html"},{text:"Days View",filename:"days.html"},{text:"Week View",filename:"week.html"},{text:"Resources View",filename:"resources.html"},{text:"Resource Calendar"},{text:"Resource Images",filename:"resourceimages.html",tt:!0},{text:"Resource Icons",filename:"resourceicons.html",tt:!0},{text:"Grid"},{text:"Height",filename:"height.html"},{text:"Highlighting Today",filename:"today.html",tt:!0},{text:"Lunch Break",filename:"lunch.html",tt:!0},{text:"Text in Cells",filename:"textincells.html",tt:!0},{text:"Navigation"},{text:"Navigator",filename:"navigator.html"},{text:"Date Picker",filename:"datepicker.html"},{text:"Next/Previous",filename:"nextprevious.html"},{text:"Themes"},{text:"Green Theme",filename:"themegreen.html"},{text:"Transparent Theme",filename:"themetransparent.html"},{text:"White Theme",filename:"themewhite.html"},{text:"Events"},{text:"Event Active Areas",filename:"eventareas.html"},{text:"Event Context Menu",filename:"eventmenu.html"},{text:"Event Customization",filename:"eventcustomization.html"},{text:"Event Icons",filename:"eventicons.html"},{text:"Event Deleting",filename:"eventdeleting.html"},{text:"Event Types",filename:"eventtypes.html"},{text:"Localization"},{text:"Localization",filename:"localization.html"},{text:"Frameworks"},{text:"Angular",filename:"angular.html"},{text:"React",filename:"react.html"},{text:"Vue",filename:"vue.html"}]},{text:"Month",Z:"month",children:[{text:"Main"},{text:"JavaScript Monthly Event Calendar",filename:"index.html"},{text:"Navigation"},{text:"Navigator",filename:"navigator.html"},{text:"Date Picker",filename:"datepicker.html"},{text:"Next/Previous",filename:"nextprevious.html"},{text:"Themes"},{text:"Green Theme",filename:"themegreen.html"},{text:"Transparent Theme",filename:"themetransparent.html"},{text:"White Theme",filename:"themewhite.html"},{text:"Events"},{text:"Event Active Areas",filename:"eventareas.html"},{text:"Event Context Menu",filename:"eventmenu.html"},{text:"Event Customization",filename:"eventcustomization.html"},{text:"Event Deleting",filename:"eventdeleting.html"},{text:"Grid"},{text:"Cell Customization",filename:"cellcustomization.html",tt:!0},{text:"Frameworks"},{text:"Angular",filename:"angular.html"},{text:"React",filename:"react.html"},{text:"Vue",filename:"vue.html"},{text:"Localization"},{text:"Localization",filename:"localization.html"}]},{text:"Scheduler",Z:"scheduler",children:[{text:"Main"},{text:"JavaScript Scheduler",filename:"index.html"}]}],i=function(){function t(){this.Dt={},this.load()}return t.prototype.getItems=function(t){var e,n;return null!==(n=null===(e=this.Dt[t])||void 0===e?void 0:e.children)&&void 0!==n?n:[]},t.prototype.it=function(){return n},t.prototype.load=function(){var t=this;n.forEach((function(e){t.Dt[e.Z]=e}))},t}();e.L=i},827:function(t,e){Object.defineProperty(e,"t",{value:!0}),e.St=void 0;var n=function(){function t(){}return t.jt=function(t){var e=this,n=t.split(/\r?\n/),i=100;n.forEach((function(t){if(!(0===t.trim().length)){var n=e.zt(t);na,.dpw-header-item>a:visited{display:inline-block;margin-right:30px;padding:10px 0}.dpw-header-item a.dpw-header-item-selected{border-bottom:4px solid #00639e;box-sizing:border-box}.dpw-header .dpw-header-right{position:absolute;right:0;top:0}.dpw-header form{display:inline-block}.dpw-header .dpw-header-right .dpw-header-search fieldset{border:0;padding:5px 0}.dpw-header .dpw-header-right .dpw-header-search input{font-size:14px;padding:2px 5px}.dpw-subheader{border-bottom:1px solid #ccc;box-shadow:0 5px 10px -5px rgba(0,0,0,.2);min-height:40px;overflow-x:auto;overflow-y:hidden;white-space:nowrap}.dpw-subheader-inner{font-size:14px;margin:0 auto;max-width:1600px}.dpw-subheader-item{display:inline-block}.dpw-subheader a,.dpw-subheader a:visited{color:#666;text-decoration:none}.dpw-subheader>a,.dpw-subheader>a:visited{display:inline-block;margin-right:30px;padding:10px 0}.dpw-title{background-color:#f0f0f0;border-bottom:1px solid #ccc;box-shadow:0 5px 10px -5px rgba(0,0,0,.2)}.dpw-title-inner{margin:0 auto;max-width:1600px;padding:20px 0}.dpw-title h1{margin:0;padding:0}.dpw-main{display:flex;flex-direction:row;margin:20px auto 0;max-width:1600px;overflow:hidden}.dpw-body-inner{line-height:1.6}.section-highlight{box-sizing:border-box;margin:30px 0;padding:30px 0;position:relative}.section-highlight:before{background-color:#f0f0f0;content:"";height:100%;left:0;margin-left:-1000px;margin-right:-1000px;position:absolute;right:0;top:0;z-index:-1}.dpw-body-edit{background-color:red;bottom:0;color:#fff;padding:5px;position:fixed;right:0}.dpw-body-edit a{color:#fff}.dpw-body h1{font-size:30px;margin:0;padding-bottom:30px}.dpw-body h2{font-size:30px;margin:0;padding-top:40px}.dpw-body img{height:auto;max-width:100%}.dpw-body a,.dpw-sidebar a{color:#1155a3}.dpw-body pre{line-height:1.42;overflow:auto;overflow-x:auto;overflow-y:hidden;padding:.5em;width:100%}.dpw-body em,.dpw-body pre{background-color:#f7f7f9;border:1px solid #d9d9d9;box-sizing:border-box;font-size:13px}.dpw-body em{font-family:monospace;font-style:normal;padding:0 2px}.dpw-body ul{list-style-image:url();list-style-position:inside;margin:.2em 0 1em;padding:0}.dpw-body ul li{padding-left:12px;text-indent:-12px}.dpw-body ul li div{display:inline}p.image-left img{float:left;margin-right:20px;max-width:50%}p.image-right img{float:right;margin-left:20px;max-width:50%}h1,h2,h3,h4,table{clear:both}.dpw-body h2 a{color:#000;text-decoration:none}a.inline-btn{background-color:#00639e}a.inline-btn,a.inline-btn-lighter{box-sizing:border-box;color:#fff;display:inline-block;padding:10px 40px;text-decoration:none}a.inline-btn-lighter{background-color:#3495cf}table td a.inline-btn{text-align:center;width:100%}.new-label{color:#e69138;font-size:12px;font-weight:700}.breadcrumbs{color:#666;font-size:12px}.breadcrumbs a{color:#666}.dpw-footer{background:url("");margin-top:40px;min-height:40px}.dpw-footer a{color:#fff;text-decoration:none}.dpw-footer a:hover{text-decoration:underline}.dpw-footer-inner{color:#fff;font-size:11px;margin:0 auto;max-width:1600px;padding-bottom:40px;padding-top:20px}.dpw-footer ul{margin:0;padding:0}.dpw-footer ul li{list-style:none}.dpw-four-cols .dpw-col{display:inline-block;vertical-align:top;width:24%}.dpw-cookie-main{background-color:#fff;border:1px solid gray;bottom:10px;font-family:Open Sans,sans-serif;font-size:12px;max-width:400px;padding:10px;right:10px}.dpw-cookie-main-buttons{margin-top:10px;text-align:right}.dpw-cookie-main-ok{background-color:#333;color:#fff;display:inline-block;padding:5px 15px;text-decoration:none}.dpw-no-sidebar .dpw-body-inner,.dpw-no-sidebar .dpw-footer-inner,.dpw-no-sidebar .dpw-header-inner,.dpw-no-sidebar .dpw-subheader-inner,.dpw-no-sidebar .dpw-title-inner{max-width:1600px}.dpw-body-inner,.dpw-footer-inner,.dpw-header-inner,.dpw-subheader-inner,.dpw-title-inner{box-sizing:border-box;padding-left:5px;padding-right:5px}.dpw-no-sidebar .dpw-sidebar{display:none}.dpw-no-sidebar .dpw-main{display:block;max-width:none}.dpw-no-sidebar .dpw-body-inner{margin-left:auto;margin-right:auto}.dpw-sidebar-right .dpw-body{flex-grow:1;order:1}.dpw-sidebar-right .dpw-sidebar{margin-left:10px;order:2;width:220px}.dpw-sidebar-left .dpw-body{flex-grow:1}.dpw-sidebar-left .dpw-sidebar{margin-right:10px;width:220px}.dpw-sidebar-left h1,.dpw-sidebar-left h2,.dpw-sidebar-left h3,.dpw-sidebar-left h4,.dpw-sidebar-left table{clear:none}.dpw-sidebar-bottom .dpw-body-inner,.dpw-sidebar-bottom .dpw-footer-inner,.dpw-sidebar-bottom .dpw-header-inner,.dpw-sidebar-bottom .dpw-subheader-inner,.dpw-sidebar-bottom .dpw-title-inner{max-width:870px}.dpw-sidebar-bottom .dpw-main{display:block;max-width:none}.dpw-sidebar-bottom .dpw-body-inner,.dpw-sidebar-bottom .dpw-sidebar{margin-left:auto;margin-right:auto}.dpw-sidebar-bottom .dpw-sidebar{max-width:870px}.dpw-sidebar img{height:auto;max-width:100%}a.dpw-button-action{color:#fff;text-decoration:none}.dpw-button-action{background-color:#3c78d8;border:1px solid #15c;border-radius:2px;color:#fff;cursor:pointer;padding:6px 20px}.dpw-button-action.disabled{background-color:#a7c1ed;border:1px solid #759ee3;cursor:not-allowed}.dp-menu{box-sizing:border-box;font-size:13px;padding:2px}.dp-menu ul{background:-webkit-gradient(linear,left top,right top,from(#fff),to(#f9f9f9));background:-webkit-linear-gradient(left,#fff,#f9f9f9);background:-moz-linear-gradient(left,#fff 0,#f9f9f9);background:-ms-linear-gradient(left,#fff 0,#f9f9f9);background:-o-linear-gradient(left,#fff 0,#f9f9f9);background:linear-gradient(left,#fff,#f9f9f9);border:0;filter:progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff",endColorStr="#f9f9f9",GradientType=1);list-style:none;margin:0;padding:0}.dp-menu{margin-bottom:50px}.dp-menu>ul{border-bottom:1px solid #d9d9d9;border-left:1px solid #d9d9d9;border-right:1px solid #d9d9d9}.dp-menu li{xwidth:100%;overflow:hidden;text-overflow:ellipsis;vertical-align:bottom;white-space:nowrap}.dp-menu ul ul{border-bottom:none;margin-top:0}.dp-menu a,.dp-menu strong{border-top:1px solid #d9d9d9;color:#333;display:block;overflow:hidden;padding:8px 5px;text-decoration:none;text-overflow:ellipsis;white-space:nowrap}.dp-menu a:hover{xtext-decoration:underline}.dp-menu a.active{xfont-weight:bold;background:#002449;color:#fff}.dp-menu li li a{padding-left:20px}.dp-menu li li li a{padding-left:35px}.dp-menu li li li li a{padding-left:50px}.dp-menu li li li li li a{padding-left:65px}#content{max-width:1360px}.button-source{background-color:#f3f3f3;border:1px solid silver;border-radius:2px;color:#333;cursor:pointer;margin-right:5px;padding:8px 16px}.toolbar .dpw-header-item:first-child{font-weight:700}.space{margin:10px 0}.note{margin-bottom:10px;margin-top:10px}.new{background-color:#87ab13;border-radius:2px;color:#fff;display:inline-block;font-size:8px;margin-right:3px;padding:2px;position:relative;top:-1px}.download a:not(.inline-btn){color:#00639e}.search{padding:2px}.search-box{position:relative}.search input[type=text]{border:1px solid #ccc;border-radius:2px;box-sizing:border-box;margin-right:2px;padding:5px 20px 5px 5px;width:180px;width:100%}.search input::-ms-clear{display:none}.search-box button{background-color:transparent;border:1px solid transparent;border-radius:2px;color:#666;cursor:pointer;outline:none;padding:5px 7px;position:absolute;right:0;top:0}.list-front{margin-top:20px;position:relative}.list-front div{display:inline-block;margin-bottom:20px;vertical-align:top;width:290px}.list-front a img{border:0;box-shadow:0 0 5px rgba(0,0,0,.2)}.img-link{display:inline-block;overflow:hidden;position:relative;transition:opacity .3s ease-in-out}.img-link .banner{background:linear-gradient(90deg,rgba(36,37,42,.8),rgba(64,67,77,.8));border-radius:25px;bottom:50px;box-sizing:border-box;color:#fff;font-size:20px;font-weight:700;left:50px;padding:10px;position:absolute;right:50px;text-align:center;text-shadow:2px 2px 4px rgba(0,0,0,.2);transform:translateY(20%);transition:all .3s ease-in-out;visibility:hidden}.img-link:hover .banner{transform:translateY(0);visibility:visible}a.img-link{color:#fff;opacity:.7;text-decoration:none}a.img-link:hover{opacity:1}@media (-ms-high-contrast:none){template{opacity:0}.placeholder template{opacity:1}}@media screen and (max-width:1000px){.dpw-sidebar-right .dpw-body-inner{margin-right:0}.dpw-sidebar-right .dpw-sidebar{box-sizing:border-box;-moz-box-sizing:border-box;float:left;margin-left:0;width:100%}.dpw-sidebar-left .dpw-main{display:flex;flex-direction:column}.dpw-sidebar-left .dpw-body{margin-left:0;order:1}.dpw-sidebar-left .dpw-sidebar{box-sizing:border-box;-moz-box-sizing:border-box;margin-left:0;margin-top:20px;order:2;width:100%}}@media screen and (max-width:600px){.dpw-header-search input[type=text]{max-width:100px}.dpw-four-cols .dpw-col{width:49%}.breadcrumbs-segment,.breadcrumbs-segment-this,.breadcrumbs-separator{display:none}.breadcrumbs-segment-last:before,.breadcrumbs-segment.breadcrumbs-segment-last{display:inline-block;font-size:14px}.breadcrumbs-segment-last:before{content:"\ab";margin-right:5px}}@media screen and (max-width:400px){.dpw-four-cols .dpw-col{width:99%}}.dpw-body h2.title{padding-top:0}.dp-menu li{position:relative}.dp-menu a,.dp-menu a:after{transition:opacity .15s ease-in-out}.dp-menu a:after{background-color:#cccccc33;bottom:0;content:"";left:0;opacity:0;position:absolute;right:0;top:0}.dp-menu a:hover:after{opacity:1}.dp-menu a.active:hover:after{opacity:0}.dpw-body code{background:rgba(0,0,0,.1);border-radius:5px;color:rgba(0,0,0,.8);font-family:monospace;font-size:.8rem;font-style:normal;font-weight:700;padding:.2rem .4rem}@media screen and (min-width:1001px){.dp-menu{margin-bottom:0;max-height:calc(100vh - 284px)}.dp-menu,.dpw-body{overflow-y:auto;scrollbar-width:thin}.dpw-body{max-height:calc(100vh - 250px)}.dp-menu::-webkit-scrollbar,.dpw-body::-webkit-scrollbar{width:8px}.dp-menu::-webkit-scrollbar-track,.dpw-body::-webkit-scrollbar-track{background:none}.dp-menu::-webkit-scrollbar-thumb,.dpw-body::-webkit-scrollbar-thumb{background-color:rgba(0,0,0,.2);border-radius:4px}.dp-menu::-webkit-scrollbar-thumb:hover,.dpw-body::-webkit-scrollbar-thumb:hover{background-color:rgba(0,0,0,.4)}.dpw-footer{margin-top:5px}.dpw-sidebar.menu .search{padding-right:10px}@supports (-moz-appearance:none){.dp-menu,.dpw-sidebar.menu .search{padding-right:8px}}@media (pointer:coarse){.dpw-sidebar.menu .search{padding-right:2px}}.dp-menu,.dpw-sidebar.menu{position:relative}.dpw-sidebar.menu:after,.dpw-sidebar.menu:before{background:linear-gradient(180deg,#ffffffbb,transparent);content:"";height:40px;left:0;margin-right:10px;pointer-events:none;position:absolute;right:0;z-index:2}.dpw-sidebar.menu:before{top:35px}.dpw-sidebar.menu:after{background:linear-gradient(0deg,#ffffffbb,transparent);bottom:0}.dpw-sidebar.menu.no-gradient-bottom:after,.dpw-sidebar.menu.no-gradient-top:before{display:none}}.note{align-items:center;background-color:#f9f9f9;border:1px solid #e0e0e0;border-radius:8px;box-shadow:0 2px 8px rgba(0,0,0,.1);color:#333;font-size:14px;margin:20px 0;padding:12px 20px 12px 55px;position:relative}.note:after{border:3px solid #666;border-radius:50%;box-sizing:border-box;color:#666;content:"i";font-size:18px;font-weight:700;height:30px;justify-content:center;left:15px;position:absolute;top:50%;transform:translateY(-50%);width:30px}.note:after,.tools{align-items:center;display:flex}.tools{background-color:#f9f9f9;border:1px solid #e0e0e0;border-radius:8px;box-shadow:0 2px 8px rgba(0,0,0,.1);flex-wrap:wrap;justify-content:start;margin:20px 0;padding:10px 20px}.tools a,.tools button{background-color:#e0e0e0;border:none;border-radius:15px;color:#333;cursor:pointer;margin:0 10px;outline:none;padding:5px 15px;text-decoration:none;transition:all .3s}.tools a:hover,.tools button:hover{background-color:#007bff;color:#fff}.tools button{background-color:#666;color:#fff;padding:7px 17px}.tools select{background-color:#fff;border:1px solid #e0e0e0;border-radius:15px;cursor:pointer;margin:0 10px;outline:none;padding:5px 15px;transition:border-color .3s}.tools select:focus{border-color:#007bff}.tools select option{padding:5px 10px}.tools label{align-items:center;cursor:pointer;display:flex;margin:0 10px}.tools input[type=radio]{margin-bottom:4px;margin-right:5px}.tools input[type=checkbox]{margin-right:5px}.tools input[type=text]{border:1px solid #e0e0e0;border-radius:15px;margin:0 10px;outline:none;padding:5px 10px;transition:border-color .3s}.tools input[type=text]:focus{border-color:#007bff}
\ No newline at end of file
diff --git a/static/src/helpers/v2/reporter.js b/static/src/helpers/v2/reporter.js
new file mode 100644
index 0000000..41d3cff
--- /dev/null
+++ b/static/src/helpers/v2/reporter.js
@@ -0,0 +1 @@
+(new(function(){function n(){}return n.prototype.init=function(){this.t(5),this.i(5)},n.prototype.send=function(n,t,i){try{var o=new XMLHttpRequest;o.open("POST","https://log.daypilot.org/api/v1/message/error",!0);var r=JSON.stringify({key:"3-m81KH05IiFxaf4ZdDy1n_4AOsK_Ub9k9Lc_A6AnGRoynqRNwd0WRUiYhXh74UOppGhK5vs3LqvtLsdDhUe8A",message:n,filename:t,stacktrace:i,url:document.location.href});o.send(r)}catch(n){}},n.prototype.i=function(n){var t=this,i=0;document.addEventListener("DOMContentLoaded",(function(){Array.apply(null,document.querySelectorAll("img")).forEach((function(o){o.addEventListener("error",(function(r){if(!(i>=n)){i+=1;var e=o.src;t.send("error loading image",e,o.outerHTML)}}))}))}))},n.prototype.t=function(n){var t=this,i=0;window.onerror=function(o,r,e,c,a){try{if(i>=n)return;i+=1,window.setTimeout((function(){t.send(o,r,a?a.stack:"")}),100)}catch(n){}}},n}())).init();
\ No newline at end of file
diff --git a/static/src/img/calendar.png b/static/src/img/calendar.png
new file mode 100644
index 0000000..b67ffd8
Binary files /dev/null and b/static/src/img/calendar.png differ
diff --git a/static/src/img/calendar_rf.png b/static/src/img/calendar_rf.png
new file mode 100644
index 0000000..d9e8997
Binary files /dev/null and b/static/src/img/calendar_rf.png differ
diff --git a/static/src/img/conv30.png b/static/src/img/conv30.png
new file mode 100644
index 0000000..a50f61f
Binary files /dev/null and b/static/src/img/conv30.png differ
diff --git a/static/src/img/conv40.png b/static/src/img/conv40.png
new file mode 100644
index 0000000..2fe31a5
Binary files /dev/null and b/static/src/img/conv40.png differ
diff --git a/static/src/img/conv50.png b/static/src/img/conv50.png
new file mode 100644
index 0000000..cbab98b
Binary files /dev/null and b/static/src/img/conv50.png differ
diff --git a/static/src/img/heart30.png b/static/src/img/heart30.png
new file mode 100644
index 0000000..c6067a7
Binary files /dev/null and b/static/src/img/heart30.png differ
diff --git a/static/src/img/heart40.png b/static/src/img/heart40.png
new file mode 100644
index 0000000..e79ba0a
Binary files /dev/null and b/static/src/img/heart40.png differ
diff --git a/static/src/img/heart50.png b/static/src/img/heart50.png
new file mode 100644
index 0000000..26c2252
Binary files /dev/null and b/static/src/img/heart50.png differ
diff --git a/static/src/img/img_107_b.jpg b/static/src/img/img_107_b.jpg
new file mode 100644
index 0000000..4bbf0ec
Binary files /dev/null and b/static/src/img/img_107_b.jpg differ
diff --git a/static/src/img/img_108_b.jpg b/static/src/img/img_108_b.jpg
new file mode 100644
index 0000000..ec64b07
Binary files /dev/null and b/static/src/img/img_108_b.jpg differ
diff --git a/static/src/img/img_113_b.jpg b/static/src/img/img_113_b.jpg
new file mode 100644
index 0000000..46c9453
Binary files /dev/null and b/static/src/img/img_113_b.jpg differ
diff --git a/static/src/img/img_115_b.jpg b/static/src/img/img_115_b.jpg
new file mode 100644
index 0000000..8cd72fb
Binary files /dev/null and b/static/src/img/img_115_b.jpg differ
diff --git a/static/src/img/img_117_b.jpg b/static/src/img/img_117_b.jpg
new file mode 100644
index 0000000..abfbb6a
Binary files /dev/null and b/static/src/img/img_117_b.jpg differ
diff --git a/static/src/img/img_119_b.jpg b/static/src/img/img_119_b.jpg
new file mode 100644
index 0000000..8a4dcd0
Binary files /dev/null and b/static/src/img/img_119_b.jpg differ
diff --git a/static/src/img/img_134_b.jpg b/static/src/img/img_134_b.jpg
new file mode 100644
index 0000000..8644560
Binary files /dev/null and b/static/src/img/img_134_b.jpg differ
diff --git a/static/src/img/img_14_b.jpg b/static/src/img/img_14_b.jpg
new file mode 100644
index 0000000..3b86854
Binary files /dev/null and b/static/src/img/img_14_b.jpg differ
diff --git a/static/src/img/img_15_b.jpg b/static/src/img/img_15_b.jpg
new file mode 100644
index 0000000..8644560
Binary files /dev/null and b/static/src/img/img_15_b.jpg differ
diff --git a/static/src/img/img_17_b.jpg b/static/src/img/img_17_b.jpg
new file mode 100644
index 0000000..e7e97cc
Binary files /dev/null and b/static/src/img/img_17_b.jpg differ
diff --git a/static/src/img/img_18_b.jpg b/static/src/img/img_18_b.jpg
new file mode 100644
index 0000000..310ebf7
Binary files /dev/null and b/static/src/img/img_18_b.jpg differ
diff --git a/static/src/img/img_19_b.jpg b/static/src/img/img_19_b.jpg
new file mode 100644
index 0000000..65bb8f5
Binary files /dev/null and b/static/src/img/img_19_b.jpg differ
diff --git a/static/src/img/img_21_b.jpg b/static/src/img/img_21_b.jpg
new file mode 100644
index 0000000..4b16535
Binary files /dev/null and b/static/src/img/img_21_b.jpg differ
diff --git a/static/src/img/img_4_b.jpg b/static/src/img/img_4_b.jpg
new file mode 100644
index 0000000..a3dd97c
Binary files /dev/null and b/static/src/img/img_4_b.jpg differ
diff --git a/static/src/img/img_5_b.jpg b/static/src/img/img_5_b.jpg
new file mode 100644
index 0000000..2b50d65
Binary files /dev/null and b/static/src/img/img_5_b.jpg differ
diff --git a/static/src/img/img_62_b.jpg b/static/src/img/img_62_b.jpg
new file mode 100644
index 0000000..4e4d56b
Binary files /dev/null and b/static/src/img/img_62_b.jpg differ
diff --git a/static/src/img/img_63_b.jpg b/static/src/img/img_63_b.jpg
new file mode 100644
index 0000000..6277cd8
Binary files /dev/null and b/static/src/img/img_63_b.jpg differ
diff --git a/static/src/img/img_65_b.jpg b/static/src/img/img_65_b.jpg
new file mode 100644
index 0000000..a905ca9
Binary files /dev/null and b/static/src/img/img_65_b.jpg differ
diff --git a/static/src/img/img_66_b.jpg b/static/src/img/img_66_b.jpg
new file mode 100644
index 0000000..b6b9111
Binary files /dev/null and b/static/src/img/img_66_b.jpg differ
diff --git a/static/src/img/img_67_b.jpg b/static/src/img/img_67_b.jpg
new file mode 100644
index 0000000..ee826d1
Binary files /dev/null and b/static/src/img/img_67_b.jpg differ
diff --git a/static/src/img/img_68_b.jpg b/static/src/img/img_68_b.jpg
new file mode 100644
index 0000000..e234596
Binary files /dev/null and b/static/src/img/img_68_b.jpg differ
diff --git a/static/src/img/img_69_b.jpg b/static/src/img/img_69_b.jpg
new file mode 100644
index 0000000..169f33a
Binary files /dev/null and b/static/src/img/img_69_b.jpg differ
diff --git a/static/src/img/img_70_b.jpg b/static/src/img/img_70_b.jpg
new file mode 100644
index 0000000..132dec1
Binary files /dev/null and b/static/src/img/img_70_b.jpg differ
diff --git a/static/src/img/img_71_b.jpg b/static/src/img/img_71_b.jpg
new file mode 100644
index 0000000..cefcc0c
Binary files /dev/null and b/static/src/img/img_71_b.jpg differ
diff --git a/static/src/img/img_72_b.jpg b/static/src/img/img_72_b.jpg
new file mode 100644
index 0000000..f543941
Binary files /dev/null and b/static/src/img/img_72_b.jpg differ
diff --git a/static/src/img/img_73_b.jpg b/static/src/img/img_73_b.jpg
new file mode 100644
index 0000000..a01fbab
Binary files /dev/null and b/static/src/img/img_73_b.jpg differ
diff --git a/static/src/img/img_74_b.jpg b/static/src/img/img_74_b.jpg
new file mode 100644
index 0000000..056424c
Binary files /dev/null and b/static/src/img/img_74_b.jpg differ
diff --git a/static/src/img/img_76_b.jpg b/static/src/img/img_76_b.jpg
new file mode 100644
index 0000000..9fd6786
Binary files /dev/null and b/static/src/img/img_76_b.jpg differ
diff --git a/static/src/img/img_77_b.jpg b/static/src/img/img_77_b.jpg
new file mode 100644
index 0000000..9ecbed6
Binary files /dev/null and b/static/src/img/img_77_b.jpg differ
diff --git a/static/src/img/img_78_b.jpg b/static/src/img/img_78_b.jpg
new file mode 100644
index 0000000..ae2c55e
Binary files /dev/null and b/static/src/img/img_78_b.jpg differ
diff --git a/static/src/img/img_79_b.jpg b/static/src/img/img_79_b.jpg
new file mode 100644
index 0000000..75c94fe
Binary files /dev/null and b/static/src/img/img_79_b.jpg differ
diff --git a/static/src/img/img_81_b.jpg b/static/src/img/img_81_b.jpg
new file mode 100644
index 0000000..59a7d26
Binary files /dev/null and b/static/src/img/img_81_b.jpg differ
diff --git a/static/src/img/img_83_b.jpg b/static/src/img/img_83_b.jpg
new file mode 100644
index 0000000..210958f
Binary files /dev/null and b/static/src/img/img_83_b.jpg differ
diff --git a/static/src/img/img_84_b.jpg b/static/src/img/img_84_b.jpg
new file mode 100644
index 0000000..b70e93a
Binary files /dev/null and b/static/src/img/img_84_b.jpg differ
diff --git a/static/src/img/img_85_b.jpg b/static/src/img/img_85_b.jpg
new file mode 100644
index 0000000..2f41cc2
Binary files /dev/null and b/static/src/img/img_85_b.jpg differ
diff --git a/static/src/img/img_86_b.jpg b/static/src/img/img_86_b.jpg
new file mode 100644
index 0000000..d4a10fe
Binary files /dev/null and b/static/src/img/img_86_b.jpg differ
diff --git a/static/src/img/img_87_b.jpg b/static/src/img/img_87_b.jpg
new file mode 100644
index 0000000..11c2192
Binary files /dev/null and b/static/src/img/img_87_b.jpg differ
diff --git a/static/src/img/img_88_b.jpg b/static/src/img/img_88_b.jpg
new file mode 100644
index 0000000..e9f96e5
Binary files /dev/null and b/static/src/img/img_88_b.jpg differ
diff --git a/static/src/img/img_90_b.jpg b/static/src/img/img_90_b.jpg
new file mode 100644
index 0000000..9c625ae
Binary files /dev/null and b/static/src/img/img_90_b.jpg differ
diff --git a/static/src/img/img_91_b.jpg b/static/src/img/img_91_b.jpg
new file mode 100644
index 0000000..cb4e056
Binary files /dev/null and b/static/src/img/img_91_b.jpg differ
diff --git a/static/src/img/img_92_b.jpg b/static/src/img/img_92_b.jpg
new file mode 100644
index 0000000..2181263
Binary files /dev/null and b/static/src/img/img_92_b.jpg differ
diff --git a/static/src/img/img_93_b.jpg b/static/src/img/img_93_b.jpg
new file mode 100644
index 0000000..de31d06
Binary files /dev/null and b/static/src/img/img_93_b.jpg differ
diff --git a/static/src/img/img_95_b.jpg b/static/src/img/img_95_b.jpg
new file mode 100644
index 0000000..f8881b7
Binary files /dev/null and b/static/src/img/img_95_b.jpg differ
diff --git a/static/src/img/img_96_b.jpg b/static/src/img/img_96_b.jpg
new file mode 100644
index 0000000..c47aca9
Binary files /dev/null and b/static/src/img/img_96_b.jpg differ
diff --git a/static/src/img/img_97_b.jpg b/static/src/img/img_97_b.jpg
new file mode 100644
index 0000000..82d8498
Binary files /dev/null and b/static/src/img/img_97_b.jpg differ
diff --git a/static/src/img/img_98_b.jpg b/static/src/img/img_98_b.jpg
new file mode 100644
index 0000000..6a726c8
Binary files /dev/null and b/static/src/img/img_98_b.jpg differ
diff --git a/static/src/img/img_99_b.jpg b/static/src/img/img_99_b.jpg
new file mode 100644
index 0000000..08e1dd0
Binary files /dev/null and b/static/src/img/img_99_b.jpg differ
diff --git a/static/src/img/m0.png b/static/src/img/m0.png
new file mode 100644
index 0000000..329ff52
Binary files /dev/null and b/static/src/img/m0.png differ
diff --git a/static/src/img/m1.png b/static/src/img/m1.png
new file mode 100644
index 0000000..511fb29
Binary files /dev/null and b/static/src/img/m1.png differ
diff --git a/static/src/img/m2.png b/static/src/img/m2.png
new file mode 100644
index 0000000..c632633
Binary files /dev/null and b/static/src/img/m2.png differ
diff --git a/static/src/img/m3.png b/static/src/img/m3.png
new file mode 100644
index 0000000..9f30b30
Binary files /dev/null and b/static/src/img/m3.png differ
diff --git a/static/src/img/m4.png b/static/src/img/m4.png
new file mode 100644
index 0000000..0d3f826
Binary files /dev/null and b/static/src/img/m4.png differ
diff --git a/static/src/img/m5.png b/static/src/img/m5.png
new file mode 100644
index 0000000..61387d2
Binary files /dev/null and b/static/src/img/m5.png differ
diff --git a/static/src/img/people35.png b/static/src/img/people35.png
new file mode 100644
index 0000000..6a5a209
Binary files /dev/null and b/static/src/img/people35.png differ
diff --git a/static/src/img/people45.png b/static/src/img/people45.png
new file mode 100644
index 0000000..219c62e
Binary files /dev/null and b/static/src/img/people45.png differ
diff --git a/static/src/img/people55.png b/static/src/img/people55.png
new file mode 100644
index 0000000..6e88d1e
Binary files /dev/null and b/static/src/img/people55.png differ
diff --git a/static/src/img/pin.png b/static/src/img/pin.png
new file mode 100644
index 0000000..3f830a3
Binary files /dev/null and b/static/src/img/pin.png differ
diff --git a/static/src/img/pin2.png b/static/src/img/pin2.png
new file mode 100644
index 0000000..3ed727e
Binary files /dev/null and b/static/src/img/pin2.png differ
diff --git a/static/src/img/pin_display.png b/static/src/img/pin_display.png
new file mode 100644
index 0000000..7c293ce
Binary files /dev/null and b/static/src/img/pin_display.png differ
diff --git a/static/src/img/pin_display_touch.png b/static/src/img/pin_display_touch.png
new file mode 100644
index 0000000..7ce5922
Binary files /dev/null and b/static/src/img/pin_display_touch.png differ
diff --git a/static/src/img/pin_led.png b/static/src/img/pin_led.png
new file mode 100644
index 0000000..269f3f5
Binary files /dev/null and b/static/src/img/pin_led.png differ
diff --git a/static/src/js/daypilot-all.min.d.ts b/static/src/js/daypilot-all.min.d.ts
new file mode 100644
index 0000000..bbefd00
--- /dev/null
+++ b/static/src/js/daypilot-all.min.d.ts
@@ -0,0 +1,1212 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+type GlobalDate = Date;
+
+export module DayPilot {
+
+ export class CalendarPropsAndEvents {
+ backendUrl?: string;
+ businessBeginsHour?: number;
+ businessEndsHour?: number;
+ cellHeight?: number;
+ columnMarginRight?: number;
+ columnsLoadMethod?: "POST" | "GET";
+ contextMenu?: DayPilot.Menu;
+ days?: number;
+ doubleClickTimeout?: number;
+ durationBarVisible?: boolean;
+ eventClickHandling?: "Enabled" | "Disabled" | "CallBack" | "ContextMenu";
+ eventDeleteHandling?: "Update" | "Disabled" | "CallBack";
+ eventMoveHandling?: "Update" | "CallBack" | "Disabled";
+ eventResizeHandling?: "Update" | "CallBack" | "Disabled";
+ eventRightClickHandling?: "ContextMenu" | "Enabled" | "Disabled";
+ headerClickHandling?: "Enabled" | "Disabled";
+ headerDateFormat?: string;
+ headerHeight?: number;
+ headerTextWrappingEnabled?: boolean;
+ height?: number;
+ heightSpec?: "BusinessHours" | "BusinessHoursNoScroll" | "Full";
+ hideUntilInit?: boolean;
+ hourWidth?: number;
+ initScrollPos?: number;
+ loadingLabelText?: string;
+ loadingLabelHtml?: string;
+ loadingLabelVisible?: boolean;
+ locale?: string;
+ showToolTip?: boolean;
+ startDate?: DayPilot.Date | string;
+ theme?: string;
+ timeFormat?: "Auto" | "Clock12Hours" | "Clock24Hours";
+ timeRangeSelectedHandling?: "Enabled" | "Disabled" | "CallBack";
+ viewType?: "Day" | "Days" | "Week" | "WorkWeek" | "Resources";
+ visible?: boolean;
+ weekStarts?: "Auto" | number;
+ xssProtection?: "Enabled" | "Disabled";
+
+ onAfterEventRender?: EventHandler;
+
+ onBeforeCellRender?: EventHandler;
+ onBeforeEventRender?: EventHandler;
+ onBeforeHeaderRender?: EventHandler;
+
+ onEventClick?: EventHandler;
+ onEventClicked?: EventHandler;
+ onEventDelete?: EventHandler;
+ onEventDeleted?: EventHandler;
+ onEventMove?: EventHandler;
+ onEventMoved?: EventHandler;
+ onEventResize?: EventHandler;
+ onEventResized?: EventHandler;
+ onEventRightClick?: EventHandler;
+ onEventRightClicked?: EventHandler;
+ onHeaderClick?: EventHandler;
+ onHeaderClicked?: EventHandler;
+ onTimeRangeSelect?: EventHandler;
+ onTimeRangeSelected?: EventHandler;
+
+ }
+
+ export class CalendarConfig extends CalendarPropsAndEvents {
+ columns?: CalendarColumnData[];
+ events?: EventData[];
+ }
+
+ export class Calendar extends CalendarPropsAndEvents {
+ v: string;
+ columns: {
+ list: CalendarColumnData[];
+ load(url: string,
+ success: (args: { data: any; preventDefault(): void; }) => void,
+ error: (args: { request: XMLHttpRequest, exception: any; }) => void
+ ): void;
+ };
+ events: {
+ list: EventData[];
+ add(e: DayPilot.Event | EventData): void;
+ find(id: string): DayPilot.Event;
+ find(filter: (e: DayPilot.Event) => boolean): DayPilot.Event;
+ load(url: string,
+ success: (args: { data: any; preventDefault(): void; }) => void,
+ error: (args: { request: XMLHttpRequest, exception: any; }) => void
+ ): void;
+ remove(e: DayPilot.Event): void;
+ remove(id: EventId): void;
+ update(e: DayPilot.Event | EventData): void;
+ };
+
+ constructor(id: string | HTMLElement, options?: CalendarConfig);
+
+ clearSelection(): void;
+
+ dispose(): void;
+
+ disposed(): boolean;
+
+ getSelection(): DayPilot.Selection;
+
+ hide(): void;
+
+ init(): void;
+
+ show(): void;
+
+ update(options?: CalendarConfig): void;
+
+ visibleStart(): DayPilot.Date;
+
+ visibleEnd(): DayPilot.Date;
+
+ }
+
+ export interface CalendarColumnData {
+ name: string;
+ id?: ResourceId;
+ start?: DayPilot.Date | string;
+ html?: string;
+ toolTip?: string;
+ tags?: any;
+ }
+
+ export interface CalendarAfterEventRenderArgs {
+ readonly e: DayPilot.Event;
+ readonly div: HTMLElement;
+ }
+
+ export interface CalendarBeforeCellRenderArgs {
+ readonly cell: {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly resource: ResourceId;
+ readonly properties: {
+ html: string;
+ business: boolean;
+ backColor: string;
+ backImage: string;
+ backRepeat: string;
+ cssClass: string;
+ };
+ };
+ }
+
+ export interface CalendarBeforeEventRenderArgs {
+ readonly control: DayPilot.Calendar;
+ readonly data: EventData;
+ }
+
+ export interface CalendarBeforeHeaderRenderArgs {
+ readonly header: {
+ readonly id: ResourceId;
+ readonly start: DayPilot.Date;
+ readonly name: string;
+ readonly children: CalendarColumnData[];
+ html: string;
+ backColor: string;
+ cssClass?: string;
+ verticalAlignment?: "top" | "center" | "bottom";
+ toolTip: string;
+ areas: AreaData[];
+ };
+ readonly column: Column;
+ }
+
+ export interface CalendarEventClickArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ readonly ctrl: boolean;
+ readonly meta: boolean;
+ readonly originalEvent: MouseEvent;
+ preventDefault(): void;
+ }
+
+
+ export interface CalendarEventClickedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ readonly ctrl: boolean;
+ readonly meta: boolean;
+ readonly originalEvent: MouseEvent;
+ }
+
+ export interface CalendarEventRightClickArgs {
+ readonly e: DayPilot.Event;
+ preventDefault(): void;
+ }
+
+ export interface CalendarEventRightClickedArgs {
+ readonly e: DayPilot.Event;
+ }
+
+ export interface CalendarEventDeleteArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ preventDefault(): void;
+ }
+
+
+ export interface CalendarEventDeletedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ }
+
+
+ export interface CalendarEventMoveArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ readonly newResource: ResourceId;
+ readonly ctrl: boolean;
+ readonly shift: boolean;
+
+ preventDefault(): void;
+ }
+
+
+ export interface CalendarEventMovedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ readonly newResource: ResourceId;
+ readonly ctrl: boolean;
+ readonly shift: boolean;
+ }
+
+
+ export interface CalendarEventResizeArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+
+ preventDefault(): void;
+ }
+
+
+ export interface CalendarEventResizedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Calendar;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ }
+
+ export interface CalendarHeaderClickArgs {
+ readonly column: Column;
+ readonly originalEvent: MouseEvent;
+ readonly shift: boolean;
+ readonly meta: boolean;
+ readonly ctrl: boolean;
+ preventDefault(): void;
+ }
+
+ export interface CalendarHeaderClickedArgs {
+ readonly column: Column;
+ readonly originalEvent: MouseEvent;
+ readonly shift: boolean;
+ readonly meta: boolean;
+ readonly ctrl: boolean;
+ }
+
+ export interface CalendarTimeRangeSelectArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly resource: ResourceId;
+ readonly control: DayPilot.Calendar;
+ preventDefault(): void;
+ }
+
+
+ export interface CalendarTimeRangeSelectedArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly resource: ResourceId;
+ readonly control: DayPilot.Calendar;
+ }
+
+ export class Column {
+ readonly id: ResourceId;
+ readonly start: DayPilot.Date;
+ readonly name: string;
+ readonly data: CalendarColumnData;
+ }
+
+ export class MonthPropsAndEvents {
+ backendUrl?: string;
+ cellHeaderClickHandling?: "Enabled" | "Disabled";
+ cellHeaderHeight?: number;
+ cellHeight?: number;
+ cellMarginBottom?: number;
+ contextMenu?: DayPilot.Menu;
+ eventBarVisible?: boolean;
+ eventClickHandling?: "Enabled" | "Disabled" | "CallBack" | "ContextMenu";
+ eventRightClickHandling?: "ContextMenu" | "Enabled" | "Disabled";
+ eventHeight?: number;
+ eventDeleteHandling?: "Update" | "Disabled";
+ eventMoveHandling?: "Update" | "CallBack" | "Notify" | "Disabled";
+ eventResizeHandling?: "Update" | "CallBack" | "Notify" | "Disabled";
+ headerClickHandling?: "Enabled" | "Disabled" | "CallBack";
+ headerHeight?: number;
+ hideUntilInit?: boolean;
+ lineSpace?: number;
+ locale?: string;
+ showToolTip?: boolean;
+ startDate?: DayPilot.Date | string;
+ theme?: string;
+ timeRangeSelectedHandling?: "Enabled" | "Disabled" | "CallBack";
+ visible?: boolean;
+ weekStarts?: "Auto" | number;
+ width?: string;
+ xssProtection?: "Enabled" | "Disabled";
+
+ onAfterEventRender?: EventHandler;
+
+ onBeforeEventRender?: EventHandler;
+ onBeforeCellRender?: EventHandler;
+
+ onCellHeaderClick?: EventHandler;
+ onCellHeaderClicked?: EventHandler;
+ onEventClick?: EventHandler;
+ onEventClicked?: EventHandler;
+ onEventDelete?: EventHandler;
+ onEventDeleted?: EventHandler;
+ onEventMove?: EventHandler;
+ onEventMoved?: EventHandler;
+ onEventResize?: EventHandler;
+ onEventResized?: EventHandler;
+ onEventRightClick?: EventHandler;
+ onEventRightClicked?: EventHandler;
+ onTimeRangeSelect?: EventHandler;
+ onTimeRangeSelected?: EventHandler;
+ }
+
+ export class MonthConfig extends MonthPropsAndEvents {
+ events?: EventData[];
+ }
+
+ export class Month extends MonthPropsAndEvents {
+ v: string;
+ events: {
+ list: EventData[];
+ add(e: DayPilot.Event | EventData): void;
+ find(id: string): DayPilot.Event;
+ find(filter: (e: DayPilot.Event) => boolean): DayPilot.Event;
+ load(url: string,
+ success: (args: { data: any; preventDefault(): void; }) => void,
+ error: (args: { request: XMLHttpRequest, exception: any; }) => void
+ ): void;
+ remove(e: DayPilot.Event): void;
+ remove(id: EventId): void;
+ update(e: DayPilot.Event): void;
+ };
+
+ constructor(id: string | HTMLElement, options?: MonthConfig);
+
+ clearSelection(): void;
+
+ dispose(): void;
+
+ disposed(): boolean;
+
+ init(): void;
+
+ show(): void;
+
+ hide(): void;
+
+ update(options?: MonthConfig): void;
+
+ visibleStart(): DayPilot.Date;
+
+ visibleEnd(): DayPilot.Date;
+
+ }
+
+ export interface MonthAfterEventRenderArgs {
+ readonly e: DayPilot.Event;
+ readonly div: HTMLElement;
+ }
+
+ export interface MonthBeforeEventRenderArgs {
+ readonly control: DayPilot.Month;
+ readonly data: EventData;
+ }
+
+ export interface MonthBeforeCellRenderArgs {
+ readonly control: DayPilot.Month;
+ readonly cell: {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly properties: {
+ business: boolean;
+ headerHtml: string;
+ html: string;
+ backColor: string;
+ }
+ }
+ }
+
+ export interface MonthCellHeaderClickArgs {
+ readonly control: DayPilot.Month;
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ preventDefault(): void;
+ }
+
+ export interface MonthCellHeaderClickedArgs {
+ readonly control: DayPilot.Month;
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ }
+
+ export interface MonthEventClickArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ readonly div: HTMLElement;
+ readonly originalEvent: MouseEvent;
+ readonly meta: boolean;
+ readonly ctrl: boolean;
+ preventDefault(): void;
+ }
+
+ export interface MonthEventClickedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ readonly div: HTMLElement;
+ readonly originalEvent: MouseEvent;
+ readonly meta: boolean;
+ readonly ctrl: boolean;
+ }
+
+ export interface MonthEventRightClickArgs {
+ readonly e: DayPilot.Event;
+ preventDefault(): void;
+ }
+
+ export interface MonthEventRightClickedArgs {
+ readonly e: DayPilot.Event;
+ }
+
+ export interface MonthEventDeleteArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ preventDefault(): void;
+ }
+
+ export interface MonthEventDeletedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ }
+
+ export interface MonthEventMoveArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ readonly ctrl: boolean;
+ readonly shift: boolean;
+ preventDefault(): void;
+ }
+
+
+ export interface MonthEventMovedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ readonly ctrl: boolean;
+ readonly shift: boolean;
+ }
+
+
+ export interface MonthEventResizeArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ preventDefault(): void;
+ }
+
+
+ export interface MonthEventResizedArgs {
+ readonly e: DayPilot.Event;
+ readonly control: DayPilot.Month;
+ readonly newStart: DayPilot.Date;
+ readonly newEnd: DayPilot.Date;
+ }
+
+
+ export interface MonthTimeRangeSelectArgs {
+ readonly control: DayPilot.Month;
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ preventDefault(): void;
+ }
+
+
+ export interface MonthTimeRangeSelectedArgs {
+ readonly control: DayPilot.Month;
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ }
+
+ export class NavigatorPropsAndEvents {
+ cellHeight?: number;
+ cellWidth?: number;
+ command?: string;
+ dayHeaderHeight?: number;
+ freeHandSelectionEnabled?: boolean;
+ locale?: string;
+ orientation?: "Vertical" | "Horizontal";
+ rowsPerMonth?: "Auto" | "Six";
+ selectionDay?: DayPilot.Date;
+ selectionEnd?: DayPilot.Date;
+ selectionStart?: DayPilot.Date;
+ selectMode?: "Day" | "Week" | "Month" | "None";
+ showMonths?: number;
+ showWeekNumbers?: boolean;
+ skipMonths?: number;
+ startDate?: DayPilot.Date | string;
+ theme?: string;
+ titleHeight?: number;
+ weekStarts?: "Auto" | number;
+ weekNumberAlgorithm?: "Auto" | "US" | "ISO8601";
+ timeRangeSelectedHandling?: "Bind" | "None";
+ visibleRangeChangedHandling?: "Enabled" | "Disabled" | "CallBack";
+
+ onBeforeCellRender?: EventHandler;
+ onTimeRangeSelect?: EventHandler;
+ onTimeRangeSelected?: EventHandler;
+ onVisibleRangeChange?: EventHandler;
+ onVisibleRangeChanged?: EventHandler;
+ }
+
+ export class NavigatorConfig extends NavigatorPropsAndEvents {
+ events?: EventData[];
+ }
+
+ interface NavigatorSelectOptions {
+ dontFocus?: boolean;
+ dontNotify?: boolean;
+ }
+
+ export class Navigator extends NavigatorPropsAndEvents {
+ v: string;
+ events: {
+ list: EventDataShort[];
+ };
+
+ constructor(id: string | HTMLElement, options?: NavigatorConfig);
+
+ init(): void;
+
+ dispose(): void;
+
+ update(options?: NavigatorConfig): void;
+
+ select(date: DayPilot.Date | string, options?: NavigatorSelectOptions): void;
+ select(start: DayPilot.Date | string, end: DayPilot.Date | string, options?: NavigatorSelectOptions): void;
+
+ hide(): void;
+
+ show(): void;
+
+ visibleEnd(): DayPilot.Date;
+
+ visibleStart(): DayPilot.Date;
+ }
+
+ interface NavigatorBeforeCellRenderArgs {
+ readonly cell: {
+ readonly day: DayPilot.Date;
+ readonly isCurrentMonth: boolean;
+ readonly isToday: boolean;
+ readonly isWeekend: boolean;
+ html: string;
+ cssClass: string;
+ };
+ }
+
+ interface NavigatorTimeRangeSelectArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly day: DayPilot.Date;
+ readonly days: number;
+ readonly mode: "Day" | "Week" | "Month" | "None" | "FreeHand";
+ preventDefault(): void;
+ }
+
+ interface NavigatorTimeRangeSelectedArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly day: DayPilot.Date;
+ readonly days: number;
+ readonly mode: "Day" | "Week" | "Month" | "None" | "FreeHand";
+ }
+
+ interface NavigatorVisibleRangeChangeArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ preventDefault(): void;
+ }
+
+ interface NavigatorVisibleRangeChangedArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ }
+
+ export class Locale {
+ datePattern: string;
+ dateTimePattern: string;
+ dayNames: string[];
+ dayNamesShort: string[];
+ monthNames: string[];
+ monthNamesShort: string[];
+ timeFormat: "Clock12Hours" | "Clock24Hours";
+ timePattern: string;
+ weekStarts: number;
+
+ constructor(id: string, properties: {
+ dayNames: string[];
+ dayNamesShort: string[];
+ monthNames: string[];
+ monthNamesShort: string[];
+ timePattern: string;
+ datePattern: string;
+ dateTimePattern: string;
+ timeFormat: "Clock12Hours" | "Clock24Hours";
+ weekStarts: number;
+ });
+
+ static register(locale: DayPilot.Locale): void;
+ static find(id: string): DayPilot.Locale;
+ }
+
+
+ export class MenuPropsAndEvents {
+ hideOnMouseOut?: boolean;
+ items?: MenuItemData[];
+ menuTitle?: string;
+ onShow?: EventHandler;
+ onHide?: EventHandler;
+ showMenuTitle?: boolean;
+ zIndex?: number;
+ theme?: string;
+ }
+
+ export class MenuConfig extends MenuPropsAndEvents {
+ }
+
+
+ export class Menu extends MenuPropsAndEvents {
+ v: string;
+ constructor(options?: MenuConfig);
+
+ show(target?: any): void;
+ hide(): void;
+
+ static hide(): void;
+ }
+
+ export interface MenuShowArgs {
+ readonly source: any;
+ readonly menu: DayPilot.Menu;
+ preventDefault(): void;
+ }
+
+ export interface MenuHideArgs {
+ }
+
+ export class MenuBar {
+ items: any[];
+
+ constructor(id: string, options?: any);
+
+ init(): void;
+
+ dispose(): void;
+ }
+
+ export interface MenuItemData {
+ action?: "CallBack" | "PostBack";
+ command?: string;
+ cssClass?: string;
+ disabled?: boolean;
+ hidden?: boolean;
+ href?: string;
+ icon?: string;
+ image?: string;
+ items?: MenuItemData[];
+ onClick?: EventHandler;
+ symbol?: string;
+ tags?: any;
+ target?: string;
+ text?: string;
+ html?: string;
+ }
+
+ export interface MenuItemClickArgs {
+ readonly item: MenuItemData;
+ readonly source: any;
+ readonly originalEvent: MouseEvent;
+ preventDefault(): void;
+ }
+
+
+ export class SwitcherPropsAndEvents {
+ selectedClass?: string;
+
+ onChange?: EventHandler;
+ onChanged?: EventHandler;
+ onSelect?: EventHandler;
+ }
+
+ export class SwitcherConfig extends SwitcherPropsAndEvents {
+ triggers?: SwitcherTrigger[];
+ navigator?: DayPilot.Navigator;
+ }
+
+ export class Switcher extends SwitcherPropsAndEvents {
+ constructor(options?: SwitcherConfig);
+ readonly active: SwitcherView;
+
+ addTrigger(id: string | HTMLElement, view: SwitcherViewControl): void;
+ addNavigator(navigator: DayPilot.Navigator): void;
+ select(triggerId: string): void;
+
+ events: {
+ load(url: string,
+ success: (args: { data: any; preventDefault(): void; }) => void,
+ error: (args: { request: XMLHttpRequest, exception: any; }) => void
+ ): void;
+ }
+
+ }
+
+ export interface SwitcherView {
+ control: SwitcherViewControl;
+ }
+
+ export interface SwitcherTrigger {
+ id: string | HTMLElement;
+ view: SwitcherViewControl;
+ }
+
+ export interface SwitcherChangeArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly day: DayPilot.Date;
+ readonly target: SwitcherView;
+ preventDefault(): void;
+ }
+
+ export interface SwitcherChangedArgs {
+ readonly start: DayPilot.Date;
+ readonly end: DayPilot.Date;
+ readonly day: DayPilot.Date;
+ readonly target: SwitcherView;
+ }
+
+ export interface SwitcherSelectArgs {
+ readonly source: HTMLElement;
+ readonly target: SwitcherViewControl;
+ }
+
+ export type SwitcherViewControl = DayPilot.Calendar | DayPilot.Month;
+
+
+ export class Date {
+ constructor(str?: string | DayPilot.Date);
+ constructor(date: GlobalDate, isLocal?: boolean);
+
+ addDays(days: number): DayPilot.Date;
+
+ addHours(hours: number): DayPilot.Date;
+
+ addMilliseconds(millis: number): DayPilot.Date;
+
+ addMinutes(minutes: number): DayPilot.Date;
+
+ addMonths(months: number): DayPilot.Date;
+
+ addSeconds(seconds: number): DayPilot.Date;
+
+ addTime(ticks: number): DayPilot.Date;
+ addTime(duration: DayPilot.Duration): DayPilot.Date;
+
+ addYears(years: number): DayPilot.Date;
+
+ dayOfWeek(): number;
+
+ dayOfWeekISO(): number;
+
+ dayOfYear(): number;
+
+ daysInMonth(): number;
+
+ daysInYear(): number;
+
+ equals(another: DayPilot.Date): boolean;
+
+ firstDayOfMonth(): DayPilot.Date;
+
+ firstDayOfWeek(locale?: string | DayPilot.Locale): DayPilot.Date;
+ firstDayOfWeek(firstDayOfWeek?: number): DayPilot.Date;
+
+ firstDayOfYear(): DayPilot.Date;
+
+ getDatePart(): DayPilot.Date;
+
+ getDay(): number;
+
+ getDayOfWeek(): number;
+
+ getYear(): number;
+
+ getHours(): number;
+
+ getMilliseconds(): number;
+
+ getMinutes(): number;
+
+ getMonth(): number;
+
+ getSeconds(): number;
+
+ getTime(): number;
+
+ getTimePart(): number;
+
+ getTotalTicks(): number;
+
+ getYear(): number;
+
+ lastDayOfMonth(): DayPilot.Date;
+
+ toDate(): GlobalDate;
+
+ toDateLocal(): GlobalDate;
+
+ toString(pattern?: string, locale?: string | DayPilot.Locale): string;
+
+ toStringSortable(): string;
+
+ weekNumber(): number;
+
+ weekNumberISO(): number;
+
+ static fromYearMonthDay(year: number, month: number, day: number): DayPilot.Date;
+ static parse(input: string, pattern: string, locale?: string | DayPilot.Locale): DayPilot.Date;
+ static today(): DayPilot.Date;
+ static now(): DayPilot.Date;
+ static Cache: DayPilotDateCache;
+ }
+
+ export class DayPilotDateCache {
+ static clear(): void;
+ }
+
+ export class Util {
+ static overlaps(start1: DayPilot.Date, end1: DayPilot.Date, start2: DayPilot.Date, end2: DayPilot.Date): boolean;
+ static overlaps(start1: number, end1: number, start2: number, end2: number): boolean;
+ static escapeHtml(text: string): string;
+ }
+
+ export class ColorUtil {
+ static darker(color: string, steps?: number): string;
+ static lighter(color: string, steps?: number): string;
+ static contrasting(color: string, light?: string, dark?: string): string;
+ }
+
+ export class Http {
+ static get(url: string, options?: HttpGetOptions): Promise;
+ static post(url: string, data?: object | string, options?: HttpPostOptions): Promise;
+ static put(url: string, data?: object | string, options?: HttpPutOptions): Promise;
+ static delete(url: string, options?: HttpDeleteOptions): Promise;
+ }
+
+ export interface HttpDeleteOptions {
+ headers?: HttpHeaders;
+ }
+
+ export interface HttpGetOptions {
+ headers?: HttpHeaders;
+ }
+
+ export interface HttpPostOptions {
+ headers?: HttpHeaders;
+ contentType?: string;
+ }
+
+ export interface HttpPutOptions {
+ headers?: HttpHeaders;
+ contentType?: string;
+ }
+
+ export interface HttpPromiseArgs {
+ readonly data: any;
+ readonly request: XMLHttpRequest;
+ }
+
+ export interface HttpHeaders {
+ [header: string]: string;
+ }
+
+ export class Duration {
+
+ ticks: number;
+
+ constructor(ticks: number);
+
+ constructor(start: DayPilot.Date | string, end: DayPilot.Date | string);
+
+ toString(pattern?: string): string;
+
+ totalMilliseconds(): number;
+
+ totalSeconds(): number;
+
+ totalMinutes(): number;
+
+ totalHours(): number;
+
+ totalDays(): number;
+
+ milliseconds(): number;
+
+ seconds(): number;
+
+ minutes(): number;
+
+ hours(): number;
+
+ days(): number;
+
+ add(d: DayPilot.Duration): DayPilot.Duration;
+
+ static ofWeeks(i: number): DayPilot.Duration;
+
+ static ofDays(i: number): DayPilot.Duration;
+
+ static ofHours(i: number): DayPilot.Duration;
+
+ static ofMinutes(i: number): DayPilot.Duration;
+
+ static ofSeconds(i: number): DayPilot.Duration;
+
+ }
+
+ export class Event {
+ data: any;
+
+ constructor(data: EventData);
+
+ start(): DayPilot.Date;
+ start(newStart: DayPilot.Date): void;
+
+ end(): DayPilot.Date;
+ end(newEnd: DayPilot.Date): void;
+
+ id(): EventId;
+
+ text(): string;
+ text(newText: string): void;
+
+ resource(): string;
+ resource(newResource: string): void;
+
+ duration(): DayPilot.Duration;
+ }
+
+ export class Selection {
+ start: DayPilot.Date;
+ end: DayPilot.Date;
+ resource: string;
+ }
+
+ export interface EventDataShort {
+ start: string | DayPilot.Date;
+ end: string | DayPilot.Date;
+ }
+
+ export interface EventData {
+ start: string | DayPilot.Date;
+ end: string | DayPilot.Date;
+ id: EventId;
+ text: string;
+ resource?: ResourceId;
+
+ areas?: AreaData[];
+ backColor?: string;
+ barBackColor?: string;
+ barColor?: string;
+ barHidden?: boolean;
+ borderColor?: string;
+ cssClass?: string;
+ fontColor?: string;
+ html?: string;
+ tags?: any;
+ }
+
+ export interface AreaData {
+ action?: "Default" | "None" | "ContextMenu" | "ResizeEnd" | "ResizeStart" | "Move";
+ backColor?: string;
+ background?: string;
+ bottom?: number | string;
+ cssClass?: string;
+ fontColor?: string;
+ height?: number | string;
+ horizontalAlignment?: HorizontalAlignment;
+ html?: string;
+ icon?: string;
+ id?: AreaId;
+ image?: string;
+ left?: number | string;
+ menu?: Menu | string;
+ onClick?: (args: any) => void;
+ onClicked?: (args: any) => void;
+ onMouseEnter?: (args: any) => void;
+ onMouseLeave?: (args: any) => void;
+ padding?: number;
+ right?: number | string;
+ style?: string;
+ symbol?: string;
+ text?: string;
+ toolTip?: string;
+ top?: number | string;
+ verticalAlignment?: VerticalAlignment;
+ visibility?: "Hover" | "Visible" | "TouchVisible";
+ width?: number | string;
+ }
+
+
+ export function guid(): string;
+
+ export interface EventHandler {
+ (args: T): void;
+ }
+
+ export type ResourceId = string | number;
+ export type EventId = string | number;
+ export type AreaId = string | number;
+
+ export type HorizontalAlignment = "right" | "center" | "left";
+ export type VerticalAlignment = "top" | "center" | "bottom";
+
+ // modal
+
+ export class ModalPropsAndEvents {
+ autoFocus?: boolean;
+ autoStretch?: boolean;
+ autoStretchFirstLoadOnly?: boolean;
+ container?: HTMLElement;
+ disposeOnClose?: boolean;
+ dragDrop?: boolean;
+ focus?: string | { id: string, value: string | number };
+ height?: number;
+ left?: number;
+ loadingHtml?: string;
+ maxHeight?: number;
+ scrollWithPage?: boolean;
+ theme?: string;
+ top?: number;
+ useIframe?: boolean;
+ width?: number;
+ zIndex?: number;
+
+ onClose?: EventHandler;
+ onClosed?: EventHandler;
+ onShow?: EventHandler;
+ }
+
+ export class ModalConfig extends ModalPropsAndEvents {
+ }
+
+ export class Modal extends ModalPropsAndEvents {
+ constructor(options?: ModalConfig)
+
+ close(result?: any): void;
+
+ closeSerialized(): void;
+
+ showHtml(html: string | HTMLElement): void;
+
+ showUrl(url: string): void;
+
+ stretch(): void;
+
+ static close(): void;
+
+ static opener(): void;
+
+ static prompt(message: string, defaultValue?: string, options?: ModalPromptConfig): Promise;
+
+ static alert(message: string, options?: ModalAlertConfig): Promise;
+
+ static confirm(message: string, options?: ModalConfirmConfig): Promise;
+
+ static form(form?: ModalFormItem[], data?: any, options?: ModalFormConfig): Promise;
+ }
+
+ export class ModalAlertConfig extends ModalConfig {
+ okText?: string;
+ }
+
+ export class ModalConfirmConfig extends ModalConfig {
+ okText?: string;
+ cancelText?: string;
+ }
+
+ export class ModalPromptConfig extends ModalConfig {
+ okText?: string;
+ cancelText?: string;
+ }
+
+ export class ModalFormConfig extends ModalConfig {
+ okText?: string;
+ cancelText?: string;
+ locale?: string;
+ plugins?: any;
+ }
+
+ export interface ModalCloseArgs {
+ canceled: boolean;
+ result: any;
+ backgroundClick: boolean;
+
+ preventDefault(): void;
+ }
+
+ export interface ModalClosedArgs {
+ canceled: boolean;
+ result: any;
+ backgroundClick: boolean;
+ }
+
+ export interface ModalShowArgs {
+ root: Node
+ }
+
+
+ export interface ModalFormItem {
+ id?: string;
+ name?: string;
+ type?: "text" | "date" | "searchable" | "select" | "radio" | "checkbox" | "table" | "title" | "image" | "html" | "textarea" | "scrollable" | string;
+ image?: string;
+ dateFormat?: string;
+ disabled?: boolean;
+ cssClass?: string;
+ options?: ModalFormOption[];
+ children?: ModalFormItem[];
+ columns?: ModalFormTableColumns[];
+ onValidate?: EventHandler;
+ onNewRow?: EventHandler;
+ height?: number;
+ text?: string;
+ html?: string;
+ }
+
+ export interface ModalFormOption {
+ id: string | number;
+ name?: string;
+ children?: ModalFormItem[];
+ }
+
+ export interface ModalFormTableColumns {
+ id: string;
+ name: string;
+ type?: "text" | "number" | "select";
+ options?: ModalFormOption[];
+ }
+
+ export interface ModalFormItemValidationArgs {
+ value: any;
+ result: any;
+ valid: boolean;
+ message: string;
+ }
+
+ export interface ModalFormTableItemNewRowArgs {
+ value: any;
+ result: any;
+ }
+
+}
diff --git a/static/src/js/daypilot-all.min.js b/static/src/js/daypilot-all.min.js
new file mode 100644
index 0000000..1e858f2
--- /dev/null
+++ b/static/src/js/daypilot-all.min.js
@@ -0,0 +1,16 @@
+/*
+DayPilot Lite
+Copyright (c) 2005 - 2024 Annpoint s.r.o.
+https://www.daypilot.org/
+Licensed under Apache Software License 2.0
+Version: 2024.3.539-lite
+*/
+var DayPilot={};if("undefined"==typeof DayPilot)var DayPilot={};if(function(){function e(e){var t=DayPilot.Date.Cache.Ticks;if(t[e])return DayPilot.Stats.cacheHitsTicks+=1,t[e];var i,a=new Date(e),n=a.getUTCMilliseconds();i=0===n?"":n<10?".00"+n:n<100?".0"+n:"."+n;var o=a.getUTCSeconds();o<10&&(o="0"+o);var r=a.getUTCMinutes();r<10&&(r="0"+r);var s=a.getUTCHours();s<10&&(s="0"+s);var l=a.getUTCDate();l<10&&(l="0"+l);var d=a.getUTCMonth()+1;d<10&&(d="0"+d);var c=a.getUTCFullYear();if(c<=0)throw"The minimum year supported is 1.";c<10?c="000"+c:c<100?c="00"+c:c<1e3&&(c="0"+c);var u=c+"-"+d+"-"+l+"T"+s+":"+r+":"+o+i;return t[e]=u,u}function t(e,t){return!DayPilot.Util.isNullOrUndefined(e)&&(!DayPilot.Util.isNullOrUndefined(t)&&e.toLocaleLowerCase()===t.toLocaleLowerCase())}function i(e){e=Math.min(e,255),e=Math.max(e,0);var t=e.toString(16);return e<16?"0"+t:t}if("undefined"==typeof DayPilot.$){"undefined"==typeof DayPilot.Global&&(DayPilot.Global={}),DayPilot.$=function(e){return document.getElementById(e)},Object.defineProperty(DayPilot,"isKhtml",{get:function(){return"undefined"!=typeof navigator&&navigator.userAgent&&navigator.userAgent.indexOf("KHTML")!==-1}}),DayPilot.mo2=function(e,t){if(t=t||window.event,"undefined"!=typeof t.offsetX){var i={x:t.offsetX+1,y:t.offsetY+1};if(!e)return i;for(var a=t.srcElement;a&&a!==e;)"SPAN"!==a.tagName&&(i.x+=a.offsetLeft,a.offsetTop>0&&(i.y+=a.offsetTop-a.scrollTop)),a=a.offsetParent;return a?i:null}if("undefined"!=typeof t.layerX){var i={x:t.layerX,y:t.layerY,src:t.target};if(!e)return i;for(var a=t.target;a&&"absolute"!==a.style.position&&"relative"!==a.style.position;)a=a.parentNode,DayPilot.isKhtml&&(i.y+=a.scrollTop);for(;a&&a!==e;)i.x+=a.offsetLeft,i.y+=a.offsetTop-a.scrollTop,a=a.offsetParent;return a?i:null}return null},DayPilot.mo3=function(e,t){var i,a=DayPilot.page(t);if(a)if(e){var n=DayPilot.abs(e);if(!n)throw new Error("no abs");i={x:a.x-n.x,y:a.y-n.y}}else i={x:a.x,y:a.y};else if(i=DayPilot.mo2(e,t),!i)return null;return i.shift=t.shiftKey,i.meta=t.metaKey,i.ctrl=t.ctrlKey,i.alt=t.altKey,i},DayPilot.browser={},Object.defineProperty(DayPilot.browser,"hover",{get:function(){return!window.matchMedia("(any-hover: none)").matches}}),DayPilot.touch={},DayPilot.page=function(e){var t=e.changedTouches&&e.changedTouches[0]?e.changedTouches[0]:e;return"undefined"!=typeof t.pageX?{x:t.pageX,y:t.pageY}:"undefined"!=typeof e.clientX&&document.body&&document.documentElement?{x:e.clientX+document.body.scrollLeft+document.documentElement.scrollLeft,y:e.clientY+document.body.scrollTop+document.documentElement.scrollTop}:null},DayPilot.abs=function(e,t){if(!e)return null;if(e.getBoundingClientRect){var i=DayPilot.absBoundingClientBased(e);if(t){var a=DayPilot.absOffsetBased(e,!1),t=DayPilot.absOffsetBased(e,!0);i.x+=t.x-a.x,i.y+=t.y-a.y,i.w=t.w,i.h=t.h}return i}return DayPilot.absOffsetBased(e,t)},DayPilot.absBoundingClientBased=function(e){var t=e.getBoundingClientRect();return{x:t.left+window.pageXOffset,y:t.top+window.pageYOffset,w:e.clientWidth,h:e.clientHeight,toString:function(){return"x:"+this.x+" y:"+this.y+" w:"+this.w+" h:"+this.h}}},DayPilot.absOffsetBased=function(e,t){for(var i={x:e.offsetLeft,y:e.offsetTop,w:e.clientWidth,h:e.clientHeight,toString:function(){return"x:"+this.x+" y:"+this.y+" w:"+this.w+" h:"+this.h}};e.offsetParent;)e=e.offsetParent,i.x-=e.scrollLeft,i.y-=e.scrollTop,t&&(i.x<0&&(i.w+=i.x,i.x=0),i.y<0&&(i.h+=i.y,i.y=0),e.scrollLeft>0&&i.x+i.w>e.clientWidth&&(i.w-=i.x+i.w-e.clientWidth),e.scrollTop&&i.y+i.h>e.clientHeight&&(i.h-=i.y+i.h-e.clientHeight)),i.x+=e.offsetLeft,i.y+=e.offsetTop;var a=DayPilot.pageOffset();return i.x+=a.x,i.y+=a.y,i},DayPilot.isArray=function(e){return"[object Array]"===Object.prototype.toString.call(e)},DayPilot.distance=function(e,t){return Math.sqrt(Math.pow(e.x-t.x,2)+Math.pow(e.y-t.y,2))},DayPilot.sheet=function(){if("undefined"==typeof window){var e={};return e.add=function(){},e.commit=function(){},e}var t=document.createElement("style");t.setAttribute("type","text/css"),t.styleSheet||t.appendChild(document.createTextNode("")),(document.head||document.getElementsByTagName("head")[0]).appendChild(t);var i=!!t.styleSheet,e={};return e.rules=[],e.commit=function(){i&&(t.styleSheet.cssText=this.rules.join("\n"))},e.add=function(e,a,n){return i?void this.rules.push(e+"{"+a+"}"):void(t.sheet.insertRule?("undefined"==typeof n&&(n=t.sheet.cssRules.length),t.sheet.insertRule(e+"{"+a+"}",n)):t.sheet.addRule&&t.sheet.addRule(e,a,n))},e},DayPilot.gs=function(e,t){return window.getComputedStyle(e,null).getPropertyValue(t)||""},DayPilot.StyleReader=function(e){this.get=function(t){return e?DayPilot.gs(e,t):null},this.getPx=function(e){var t=this.get(e);return t.indexOf("px")===-1?void 0:parseInt(t,10)}},function(){if(!DayPilot.Global.defaultCss){var e=DayPilot.sheet();e.add(".calendar_default_main","border: 1px solid #c0c0c0; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;"),e.add(".calendar_default_main *, .calendar_default_main *:before, .calendar_default_main *:after","box-sizing: content-box;"),e.add(".calendar_default_rowheader_inner,.calendar_default_cornerright_inner,.calendar_default_corner_inner,.calendar_default_colheader_inner,.calendar_default_alldayheader_inner","color: #333;background: #f3f3f3;"),e.add(".calendar_default_cornerright_inner","position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;\tborder-bottom: 1px solid #c0c0c0;"),e.add(".calendar_default_direction_rtl .calendar_default_cornerright_inner","border-right: 1px solid #c0c0c0;"),e.add(".calendar_default_rowheader_inner","font-size: 16pt;text-align: right; position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0; padding: 3px;"),e.add(".calendar_default_direction_rtl .calendar_default_rowheader_inner","border-right: none;"),e.add(".calendar_default_corner_inner","position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;"),e.add(".calendar_default_direction_rtl .calendar_default_corner_inner","border-right: none;"),e.add(".calendar_default_rowheader_minutes","font-size:10px;vertical-align: super;padding-left: 2px;padding-right: 2px;"),e.add(".calendar_default_colheader_inner","position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0; display: flex; align-items: center; justify-content: center; font-size: 13px;"),e.add(".calendar_default_cell_inner","position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #ddd;border-bottom: 1px solid #ddd; background: #f9f9f9;"),e.add(".calendar_default_cell_business .calendar_default_cell_inner","background: #fff"),e.add(".calendar_default_alldayheader_inner","text-align: center;position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;"),e.add(".calendar_default_message","opacity: 0.9; padding: 10px; color: #ffffff;background: #ffa216;"),e.add(".calendar_default_alldayevent_inner,.calendar_default_event_inner","color: #333; border: 1px solid #999;"),e.add(".calendar_default_event_bar","top: 0px;bottom: 0px;left: 0px;width: 6px;background-color: #9dc8e8;"),e.add(".calendar_default_event_bar_inner","position: absolute;width: 6px;background-color: #1066a8;"),e.add(".calendar_default_alldayevent_inner,.calendar_default_event_inner",'background: #fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");'),e.add(".calendar_default_selected .calendar_default_event_inner","background: #ddd;"),e.add(".calendar_default_alldayevent_inner","position: absolute;top: 2px;bottom: 2px;left: 2px;right: 2px;overflow:hidden;padding: 2px;margin-right: 1px; font-size: 13px;"),e.add(".calendar_default_event_withheader .calendar_default_event_inner","padding-top: 15px;"),e.add(".calendar_default_event","cursor: default;"),e.add(".calendar_default_event_inner","position: absolute;overflow: hidden;top: 0px;bottom: 0px;left: 0px;right: 0px;padding: 2px 2px 2px 8px; font-size: 13px;"),e.add(".calendar_default_event_delete","background: url() center center no-repeat; opacity: 0.6; cursor: pointer;"),e.add(".calendar_default_event_delete:hover","opacity: 1;-ms-filter: none;"),e.add(".calendar_default_scroll_up","background: url();"),e.add(".calendar_default_scroll_down","background: url();"),e.add(".calendar_default_now","background-color: red;"),e.add(".calendar_default_now:before","content: ''; top: -5px; border-width: 5px; border-color: transparent transparent transparent red; border-style: solid; width: 0px; height:0px; position: absolute; -moz-transform: scale(.9999);"),e.add(".calendar_default_shadow_top",'box-sizing: border-box; padding:2px;border:1px solid #ccc;background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");'),e.add(".calendar_default_shadow_bottom",'box-sizing: border-box; padding:2px;border:1px solid #ccc;background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");'),e.add(".calendar_default_crosshair_vertical, .calendar_default_crosshair_horizontal, .calendar_default_crosshair_left, .calendar_default_crosshair_top","background-color: gray; opacity: 0.2;"),e.add(".calendar_default_loading","background-color: orange; color: white; padding: 2px;"),e.add(".calendar_default_scroll","background-color: #f3f3f3;"),e.add(".calendar_default_event_moving_source","opacity: 0.5;"),e.add(".calendar_default_shadow_inner","box-sizing: border-box; background-color: #bbbbbb;border: 1px solid #888888;opacity: 0.5;height: 100%;"),e.add(".calendar_default_shadow","box-shadow: 0 2px 5px rgba(0,0,0,.2);"),e.add(".menu_default_main","user-select:none; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif;font-size: 13px;border: 1px solid #dddddd;background-color: white;padding: 0px;cursor: default;background-image: url();background-repeat: repeat-y;xborder-radius: 5px;-moz-box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);-webkit-box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);"),e.add(".menu_default_main, .menu_default_main *, .menu_default_main *:before, .menu_default_main *:after","box-sizing: content-box;"),e.add(".menu_default_title","background-color: #f2f2f2;border-bottom: 1px solid gray;padding: 4px 4px 4px 37px;"),e.add(".menu_default_main a","padding: 2px 2px 2px 35px;color: black;text-decoration: none;cursor: default;"),e.add(".menu_default_main.menu_default_withchildren a","padding: 2px 35px 2px 35px;"),e.add(".menu_default_main a img","margin-left: 6px;margin-top: 2px;"),e.add(".menu_default_item_text","display: block;height: 20px;line-height: 20px; overflow:hidden;padding-left: 2px;padding-right: 20px; white-space: nowrap;"),e.add(".menu_default_main a:hover","background-color: #f3f3f3;"),e.add(".menu_default_main div div","border-top: 1px solid #dddddd;margin-top: 2px;margin-bottom: 2px;margin-left: 28px;"),e.add(".menu_default_main a.menu_default_item_disabled","color: #ccc"),e.add(".menu_default_item_haschildren.menu_default_item_haschildren_active","background-color: #f3f3f3;"),e.add(".menu_default_item_haschildren a:before","content: ''; border-width: 5px; border-color: transparent transparent transparent #666; border-style: solid; width: 0px; height:0px; position: absolute; right: 5px; margin-top: 5px;"),e.add(".menu_default_item_icon","position: absolute; top:0px; left: 0px; padding: 2px 2px 2px 8px;"),e.add(".menu_default_item a i","height: 20px;line-height: 20px;"),e.add(".menu_default_item .menu_default_item_symbol","width: 18px; height: 18px; color: #999; margin-left: 6px;margin-top: 2px;"),e.add(".menubar_default_main","border-bottom: 1px solid #ccc; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px; user-select:none;"),e.add(".menubar_default_item","display: inline-block; padding: 6px 10px; cursor: default;"),e.add(".menubar_default_item:hover","background-color: #f2f2f2;"),e.add(".menubar_default_item_active","background-color: #f2f2f2;"),e.add(".scheduler_default_selected .scheduler_default_event_inner","background: #ddd;"),e.add(".scheduler_default_main","border: 1px solid #c0c0c0;font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;"),e.add(".scheduler_default_timeheader","cursor: default;color: #333;"),e.add(".scheduler_default_message","opacity: 0.9;filter: alpha(opacity=90);padding: 10px; color: #ffffff;background: #ffa216;"),e.add(".scheduler_default_timeheadergroup,.scheduler_default_timeheadercol","color: #333;background: #f3f3f3;"),e.add(".scheduler_default_rowheader,.scheduler_default_corner","color: #333;background: #f3f3f3;"),e.add(".scheduler_default_rowheader_inner","position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #eee;padding: 2px;"),e.add(".scheduler_default_timeheadergroup, .scheduler_default_timeheadercol","text-align: center;"),e.add(".scheduler_default_timeheadergroup_inner","position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;"),e.add(".scheduler_default_timeheadercol_inner","position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #c0c0c0;"),e.add(".scheduler_default_divider","background-color: #c0c0c0;"),e.add(".scheduler_default_divider_horizontal","background-color: #c0c0c0;"),e.add(".scheduler_default_matrix_vertical_line","background-color: #eee;"),e.add(".scheduler_default_matrix_vertical_break","background-color: #000;"),e.add(".scheduler_default_matrix_horizontal_line","background-color: #eee;"),e.add(".scheduler_default_resourcedivider","background-color: #c0c0c0;"),e.add(".scheduler_default_shadow_inner","background-color: #666666;opacity: 0.5;filter: alpha(opacity=50);height: 100%;xborder-radius: 5px;"),e.add(".scheduler_default_event","color:#333; font-size: 13px;"),e.add(".scheduler_default_event_inner","position:absolute;top:0px;left:0px;right:0px;bottom:0px;padding:5px 2px 2px 2px;overflow:hidden;border:1px solid #ccc;"),e.add(".scheduler_default_event_bar","top:0px;left:0px;right:0px;height:4px;background-color:#9dc8e8;"),e.add(".scheduler_default_event_bar_inner","position:absolute;height:4px;background-color:#1066a8;"),e.add(".scheduler_default_event_inner",'background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");'),e.add(".scheduler_default_event_float_inner","padding:6px 2px 2px 8px;"),e.add(".scheduler_default_event_float_inner:after",'content:"";border-color: transparent #666 transparent transparent;border-style:solid;border-width:5px;width:0;height:0;position:absolute;top:8px;left:-4px;'),e.add(".scheduler_default_columnheader_inner","font-weight: bold;"),e.add(".scheduler_default_columnheader_splitter","background-color: #666;opacity: 0.5;filter: alpha(opacity=50);"),e.add(".scheduler_default_columnheader_cell_inner","padding: 2px;"),e.add(".scheduler_default_cell","background-color: #f9f9f9;"),e.add(".scheduler_default_cell.scheduler_default_cell_business","background-color: #fff;"),e.add(".navigator_default_main","border-left: 1px solid #c0c0c0;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;background-color: white;color: #000000; box-sizing: content-box;"),e.add(".navigator_default_main *, .navigator_default_main *:before, .navigator_default_main *:after","box-sizing: content-box;"),e.add(".navigator_default_month","font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 12px;"),e.add(".navigator_default_day","color: black;"),e.add(".navigator_default_weekend","background-color: #f0f0f0;"),e.add(".navigator_default_dayheader","color: black;"),e.add(".navigator_default_line","border-bottom: 1px solid #c0c0c0;"),e.add(".navigator_default_dayother","color: gray;"),e.add(".navigator_default_todaybox","border: 1px solid red;"),e.add(".navigator_default_title, .navigator_default_titleleft, .navigator_default_titleright","border-top: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;color: #333;background: #f3f3f3;"),e.add(".navigator_default_busy","font-weight: bold;"),e.add(".navigator_default_cell","text-align: center;"),e.add(".navigator_default_select .navigator_default_cell_box","background-color: #FFE794; opacity: 0.5;"),e.add(".navigator_default_title","text-align: center;"),e.add(".navigator_default_titleleft, .navigator_default_titleright","text-align: center;"),e.add(".navigator_default_dayheader","text-align: center;"),e.add(".navigator_default_weeknumber","text-align: center; color: #999;"),e.add(".navigator_default_cell_text","cursor: pointer;"),e.add(".month_default_main","border: 1px solid #c0c0c0;font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;color: #333;"),e.add(".month_default_main *, .month_default_main *:before, .month_default_main *:after","box-sizing: content-box;"),e.add(".month_default_cell_inner","border-right: 1px solid #ddd;border-bottom: 1px solid #ddd;position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;background-color: #f9f9f9;"),e.add(".month_default_cell_business .month_default_cell_inner","background-color: #fff;"),e.add(".month_default_cell_header","text-align: right; padding: 4px; box-sizing: border-box;"),e.add(".month_default_header_inner","position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;cursor: default;color: #333;background: #f3f3f3; overflow:hidden; display: flex; align-items: center; justify-content: center;"),e.add(".month_default_message","padding: 10px;opacity: 0.9; color: #ffffff;background: #ffa216;"),e.add(".month_default_event_inner","position: absolute;top: 0px;bottom: 0px;left: 1px;right: 1px;overflow:hidden;padding: 2px;padding-left: 10px;color: #333;background: #fff;background: linear-gradient(to bottom, #ffffff 0%, #eeeeee);border: 1px solid #999;border-radius: 0px;display: flex; align-items: center; font-size: 13px;"),e.add(".month_default_event_continueright .month_default_event_inner","border-top-right-radius: 0px;border-bottom-right-radius: 0px;border-right-style: dotted;"),e.add(".month_default_event_continueleft .month_default_event_inner","border-top-left-radius: 0px;border-bottom-left-radius: 0px;border-left-style: dotted;"),e.add(".month_default_event_bar","top: 0px;bottom: 0px;left: 0px;width: 6px;"),e.add(".month_default_event_bar_inner","position: absolute;width: 6px;background-color: #1066a8;"),e.add(".month_default_event_continueleft .month_default_event_bar","display: none;"),e.add(".month_default_selected .month_default_event_inner","background: #ddd;"),e.add(".month_default_event_delete","background: url() center center no-repeat; opacity: 0.6; cursor: pointer;"),e.add(".month_default_event_delete:hover","opacity: 1;-ms-filter: none;"),e.add(".month_default_event_timeleft","color: #ccc; font-size: 8pt"),e.add(".month_default_event_timeright","color: #ccc; font-size: 8pt; text-align: right;"),e.add(".month_default_loading","background-color: orange; color: white; padding: 2px;"),e.add(".month_default_shadow_inner","box-sizing: border-box; background-color: #bbbbbb;border: 1px solid #888888;opacity: 0.5;height: 100%;"),e.add(".month_default_shadow","box-shadow: 0 2px 5px rgba(0,0,0,.2);"),e.commit(),DayPilot.Global.defaultCss=!0}}(),DayPilot.doc=function(){var e=document.documentElement;return e&&e.clientHeight?e:document.body},DayPilot.guid=function(){var e=function(){return(65536*(1+Math.random())|0).toString(16).substring(1)};return""+e()+e()+"-"+e()+"-"+e()+"-"+e()+"-"+e()+e()+e()},DayPilot.pageOffset=function(){if("undefined"!=typeof pageXOffset)return{x:pageXOffset,y:pageYOffset};var e=DayPilot.doc();return{x:e.scrollLeft,y:e.scrollTop}},DayPilot.indexOf=function(e,t){if(!e||!e.length)return-1;for(var i=0;i0)for(var o=0;o0)){e.areas=[];for(var o=0;o2&&(t=DayPilot.Util.replaceCharAt(t,2,"-")),DayPilot.Locale.all[t]},DayPilot.Locale.register=function(e){DayPilot.Locale.all[e.id]=e},DayPilot.Locale.register(new DayPilot.Locale("ca-es",{"dayNames":["diumenge","dilluns","dimarts","dimecres","dijous","divendres","dissabte"],"dayNamesShort":["dg","dl","dt","dc","dj","dv","ds"],"monthNames":["gener","febrer","març","abril","maig","juny","juliol","agost","setembre","octubre","novembre","desembre",""],"monthNamesShort":["gen.","febr.","març","abr.","maig","juny","jul.","ag.","set.","oct.","nov.","des.",""],"timePattern":"H:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("cs-cz",{"dayNames":["neděle","pondělí","úterý","středa","čtvrtek","pátek","sobota"],"dayNamesShort":["ne","po","út","st","čt","pá","so"],"monthNames":["leden","únor","březen","duben","květen","červen","červenec","srpen","září","říjen","listopad","prosinec",""],"monthNamesShort":["I","II","III","IV","V","VI","VII","VIII","IX","X","XI","XII",""],"timePattern":"H:mm","datePattern":"d. M. yyyy","dateTimePattern":"d. M. yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("da-dk",{"dayNames":["søndag","mandag","tirsdag","onsdag","torsdag","fredag","lørdag"],"dayNamesShort":["sø","ma","ti","on","to","fr","lø"],"monthNames":["januar","februar","marts","april","maj","juni","juli","august","september","oktober","november","december",""],"monthNamesShort":["jan","feb","mar","apr","maj","jun","jul","aug","sep","okt","nov","dec",""],"timePattern":"HH:mm","datePattern":"dd-MM-yyyy","dateTimePattern":"dd-MM-yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("de-at",{"dayNames":["Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag"],"dayNamesShort":["So","Mo","Di","Mi","Do","Fr","Sa"],"monthNames":["Jänner","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember",""],"monthNamesShort":["Jän","Feb","Mär","Apr","Mai","Jun","Jul","Aug","Sep","Okt","Nov","Dez",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("de-ch",{"dayNames":["Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag"],"dayNamesShort":["So","Mo","Di","Mi","Do","Fr","Sa"],"monthNames":["Januar","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember",""],"monthNamesShort":["Jan","Feb","Mrz","Apr","Mai","Jun","Jul","Aug","Sep","Okt","Nov","Dez",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("de-de",{"dayNames":["Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag"],"dayNamesShort":["So","Mo","Di","Mi","Do","Fr","Sa"],"monthNames":["Januar","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember",""],"monthNamesShort":["Jan","Feb","Mrz","Apr","Mai","Jun","Jul","Aug","Sep","Okt","Nov","Dez",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("de-lu",{"dayNames":["Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag"],"dayNamesShort":["So","Mo","Di","Mi","Do","Fr","Sa"],"monthNames":["Januar","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember",""],"monthNamesShort":["Jan","Feb","Mrz","Apr","Mai","Jun","Jul","Aug","Sep","Okt","Nov","Dez",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("en-au",{"dayNames":["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"],"dayNamesShort":["Su","Mo","Tu","We","Th","Fr","Sa"],"monthNames":["January","February","March","April","May","June","July","August","September","October","November","December",""],"monthNamesShort":["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec",""],"timePattern":"h:mm tt","datePattern":"d/MM/yyyy","dateTimePattern":"d/MM/yyyy h:mm tt","timeFormat":"Clock12Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("en-ca",{"dayNames":["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"],"dayNamesShort":["Su","Mo","Tu","We","Th","Fr","Sa"],"monthNames":["January","February","March","April","May","June","July","August","September","October","November","December",""],"monthNamesShort":["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec",""],"timePattern":"h:mm tt","datePattern":"yyyy-MM-dd","dateTimePattern":"yyyy-MM-dd h:mm tt","timeFormat":"Clock12Hours","weekStarts":0})),DayPilot.Locale.register(new DayPilot.Locale("en-gb",{"dayNames":["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"],"dayNamesShort":["Su","Mo","Tu","We","Th","Fr","Sa"],"monthNames":["January","February","March","April","May","June","July","August","September","October","November","December",""],"monthNamesShort":["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec",""],"timePattern":"HH:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("en-us",{"dayNames":["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"],"dayNamesShort":["Su","Mo","Tu","We","Th","Fr","Sa"],"monthNames":["January","February","March","April","May","June","July","August","September","October","November","December",""],"monthNamesShort":["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec",""],"timePattern":"h:mm tt","datePattern":"M/d/yyyy","dateTimePattern":"M/d/yyyy h:mm tt","timeFormat":"Clock12Hours","weekStarts":0})),DayPilot.Locale.register(new DayPilot.Locale("es-es",{"dayNames":["domingo","lunes","martes","miércoles","jueves","viernes","sábado"],"dayNamesShort":["D","L","M","X","J","V","S"],"monthNames":["enero","febrero","marzo","abril","mayo","junio","julio","agosto","septiembre","octubre","noviembre","diciembre",""],"monthNamesShort":["ene.","feb.","mar.","abr.","may.","jun.","jul.","ago.","sep.","oct.","nov.","dic.",""],"timePattern":"H:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("es-mx",{"dayNames":["domingo","lunes","martes","miércoles","jueves","viernes","sábado"],"dayNamesShort":["do.","lu.","ma.","mi.","ju.","vi.","sá."],"monthNames":["enero","febrero","marzo","abril","mayo","junio","julio","agosto","septiembre","octubre","noviembre","diciembre",""],"monthNamesShort":["ene.","feb.","mar.","abr.","may.","jun.","jul.","ago.","sep.","oct.","nov.","dic.",""],"timePattern":"hh:mm tt","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy hh:mm tt","timeFormat":"Clock12Hours","weekStarts":0})),DayPilot.Locale.register(new DayPilot.Locale("eu-es",{"dayNames":["igandea","astelehena","asteartea","asteazkena","osteguna","ostirala","larunbata"],"dayNamesShort":["ig","al","as","az","og","or","lr"],"monthNames":["urtarrila","otsaila","martxoa","apirila","maiatza","ekaina","uztaila","abuztua","iraila","urria","azaroa","abendua",""],"monthNamesShort":["urt.","ots.","mar.","api.","mai.","eka.","uzt.","abu.","ira.","urr.","aza.","abe.",""],"timePattern":"H:mm","datePattern":"yyyy/MM/dd","dateTimePattern":"yyyy/MM/dd H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("fi-fi",{"dayNames":["sunnuntai","maanantai","tiistai","keskiviikko","torstai","perjantai","lauantai"],"dayNamesShort":["su","ma","ti","ke","to","pe","la"],"monthNames":["tammikuu","helmikuu","maaliskuu","huhtikuu","toukokuu","kesäkuu","heinäkuu","elokuu","syyskuu","lokakuu","marraskuu","joulukuu",""],"monthNamesShort":["tammi","helmi","maalis","huhti","touko","kesä","heinä","elo","syys","loka","marras","joulu",""],"timePattern":"H:mm","datePattern":"d.M.yyyy","dateTimePattern":"d.M.yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("fr-be",{"dayNames":["dimanche","lundi","mardi","mercredi","jeudi","vendredi","samedi"],"dayNamesShort":["di","lu","ma","me","je","ve","sa"],"monthNames":["janvier","février","mars","avril","mai","juin","juillet","août","septembre","octobre","novembre","décembre",""],"monthNamesShort":["janv.","févr.","mars","avr.","mai","juin","juil.","août","sept.","oct.","nov.","déc.",""],"timePattern":"HH:mm","datePattern":"dd-MM-yy","dateTimePattern":"dd-MM-yy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("fr-ch",{"dayNames":["dimanche","lundi","mardi","mercredi","jeudi","vendredi","samedi"],"dayNamesShort":["di","lu","ma","me","je","ve","sa"],"monthNames":["janvier","février","mars","avril","mai","juin","juillet","août","septembre","octobre","novembre","décembre",""],"monthNamesShort":["janv.","févr.","mars","avr.","mai","juin","juil.","août","sept.","oct.","nov.","déc.",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("fr-fr",{"dayNames":["dimanche","lundi","mardi","mercredi","jeudi","vendredi","samedi"],"dayNamesShort":["di","lu","ma","me","je","ve","sa"],"monthNames":["janvier","février","mars","avril","mai","juin","juillet","août","septembre","octobre","novembre","décembre",""],"monthNamesShort":["janv.","févr.","mars","avr.","mai","juin","juil.","août","sept.","oct.","nov.","déc.",""],"timePattern":"HH:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("fr-lu",{"dayNames":["dimanche","lundi","mardi","mercredi","jeudi","vendredi","samedi"],"dayNamesShort":["di","lu","ma","me","je","ve","sa"],"monthNames":["janvier","février","mars","avril","mai","juin","juillet","août","septembre","octobre","novembre","décembre",""],"monthNamesShort":["janv.","févr.","mars","avr.","mai","juin","juil.","août","sept.","oct.","nov.","déc.",""],"timePattern":"HH:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("gl-es",{"dayNames":["domingo","luns","martes","mércores","xoves","venres","sábado"],"dayNamesShort":["do","lu","ma","mé","xo","ve","sá"],"monthNames":["xaneiro","febreiro","marzo","abril","maio","xuño","xullo","agosto","setembro","outubro","novembro","decembro",""],"monthNamesShort":["xan","feb","mar","abr","maio","xuño","xul","ago","set","out","nov","dec",""],"timePattern":"H:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("it-it",{"dayNames":["domenica","lunedì","martedì","mercoledì","giovedì","venerdì","sabato"],"dayNamesShort":["do","lu","ma","me","gi","ve","sa"],"monthNames":["gennaio","febbraio","marzo","aprile","maggio","giugno","luglio","agosto","settembre","ottobre","novembre","dicembre",""],"monthNamesShort":["gen","feb","mar","apr","mag","giu","lug","ago","set","ott","nov","dic",""],"timePattern":"HH:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("it-ch",{"dayNames":["domenica","lunedì","martedì","mercoledì","giovedì","venerdì","sabato"],"dayNamesShort":["do","lu","ma","me","gi","ve","sa"],"monthNames":["gennaio","febbraio","marzo","aprile","maggio","giugno","luglio","agosto","settembre","ottobre","novembre","dicembre",""],"monthNamesShort":["gen","feb","mar","apr","mag","giu","lug","ago","set","ott","nov","dic",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("ja-jp",{"dayNames":["日曜日","月曜日","火曜日","水曜日","木曜日","金曜日","土曜日"],"dayNamesShort":["日","月","火","水","木","金","土"],"monthNames":["1月","2月","3月","4月","5月","6月","7月","8月","9月","10月","11月","12月",""],"monthNamesShort":["1","2","3","4","5","6","7","8","9","10","11","12",""],"timePattern":"H:mm","datePattern":"yyyy/MM/dd","dateTimePattern":"yyyy/MM/dd H:mm","timeFormat":"Clock24Hours","weekStarts":0})),DayPilot.Locale.register(new DayPilot.Locale("nb-no",{"dayNames":["søndag","mandag","tirsdag","onsdag","torsdag","fredag","lørdag"],"dayNamesShort":["sø","ma","ti","on","to","fr","lø"],"monthNames":["januar","februar","mars","april","mai","juni","juli","august","september","oktober","november","desember",""],"monthNamesShort":["jan","feb","mar","apr","mai","jun","jul","aug","sep","okt","nov","des",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("nl-nl",{"dayNames":["zondag","maandag","dinsdag","woensdag","donderdag","vrijdag","zaterdag"],"dayNamesShort":["zo","ma","di","wo","do","vr","za"],"monthNames":["januari","februari","maart","april","mei","juni","juli","augustus","september","oktober","november","december",""],"monthNamesShort":["jan","feb","mrt","apr","mei","jun","jul","aug","sep","okt","nov","dec",""],"timePattern":"HH:mm","datePattern":"d-M-yyyy","dateTimePattern":"d-M-yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("nl-be",{"dayNames":["zondag","maandag","dinsdag","woensdag","donderdag","vrijdag","zaterdag"],"dayNamesShort":["zo","ma","di","wo","do","vr","za"],"monthNames":["januari","februari","maart","april","mei","juni","juli","augustus","september","oktober","november","december",""],"monthNamesShort":["jan","feb","mrt","apr","mei","jun","jul","aug","sep","okt","nov","dec",""],"timePattern":"H:mm","datePattern":"d/MM/yyyy","dateTimePattern":"d/MM/yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("nn-no",{"dayNames":["søndag","måndag","tysdag","onsdag","torsdag","fredag","laurdag"],"dayNamesShort":["sø","må","ty","on","to","fr","la"],"monthNames":["januar","februar","mars","april","mai","juni","juli","august","september","oktober","november","desember",""],"monthNamesShort":["jan","feb","mar","apr","mai","jun","jul","aug","sep","okt","nov","des",""],"timePattern":"HH:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("pt-br",{"dayNames":["domingo","segunda-feira","terça-feira","quarta-feira","quinta-feira","sexta-feira","sábado"],"dayNamesShort":["D","S","T","Q","Q","S","S"],"monthNames":["janeiro","fevereiro","março","abril","maio","junho","julho","agosto","setembro","outubro","novembro","dezembro",""],"monthNamesShort":["jan","fev","mar","abr","mai","jun","jul","ago","set","out","nov","dez",""],"timePattern":"HH:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":0})),DayPilot.Locale.register(new DayPilot.Locale("pl-pl",{"dayNames":["niedziela","poniedziałek","wtorek","środa","czwartek","piątek","sobota"],"dayNamesShort":["N","Pn","Wt","Śr","Cz","Pt","So"],"monthNames":["styczeń","luty","marzec","kwiecień","maj","czerwiec","lipiec","sierpień","wrzesień","październik","listopad","grudzień",""],"monthNamesShort":["sty","lut","mar","kwi","maj","cze","lip","sie","wrz","paź","lis","gru",""],"timePattern":"HH:mm","datePattern":"yyyy-MM-dd","dateTimePattern":"yyyy-MM-dd HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("pt-pt",{"dayNames":["domingo","segunda-feira","terça-feira","quarta-feira","quinta-feira","sexta-feira","sábado"],"dayNamesShort":["D","S","T","Q","Q","S","S"],"monthNames":["janeiro","fevereiro","março","abril","maio","junho","julho","agosto","setembro","outubro","novembro","dezembro",""],"monthNamesShort":["jan","fev","mar","abr","mai","jun","jul","ago","set","out","nov","dez",""],"timePattern":"HH:mm","datePattern":"dd/MM/yyyy","dateTimePattern":"dd/MM/yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":0})),DayPilot.Locale.register(new DayPilot.Locale("ro-ro",{"dayNames":["duminică","luni","marți","miercuri","joi","vineri","sâmbătă"],"dayNamesShort":["D","L","Ma","Mi","J","V","S"],"monthNames":["ianuarie","februarie","martie","aprilie","mai","iunie","iulie","august","septembrie","octombrie","noiembrie","decembrie",""],"monthNamesShort":["ian.","feb.","mar.","apr.","mai.","iun.","iul.","aug.","sep.","oct.","nov.","dec.",""],"timePattern":"H:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("ru-ru",{"dayNames":["воскресенье","понедельник","вторник","среда","четверг","пятница","суббота"],"dayNamesShort":["Вс","Пн","Вт","Ср","Чт","Пт","Сб"],"monthNames":["Январь","Февраль","Март","Апрель","Май","Июнь","Июль","Август","Сентябрь","Октябрь","Ноябрь","Декабрь",""],"monthNamesShort":["янв","фев","мар","апр","май","июн","июл","авг","сен","окт","ноя","дек",""],"timePattern":"H:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("sk-sk",{"dayNames":["nedeľa","pondelok","utorok","streda","štvrtok","piatok","sobota"],"dayNamesShort":["ne","po","ut","st","št","pi","so"],"monthNames":["január","február","marec","apríl","máj","jún","júl","august","september","október","november","december",""],"monthNamesShort":["1","2","3","4","5","6","7","8","9","10","11","12",""],"timePattern":"H:mm","datePattern":"d.M.yyyy","dateTimePattern":"d.M.yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("sv-se",{"dayNames":["söndag","måndag","tisdag","onsdag","torsdag","fredag","lördag"],"dayNamesShort":["sö","må","ti","on","to","fr","lö"],"monthNames":["januari","februari","mars","april","maj","juni","juli","augusti","september","oktober","november","december",""],"monthNamesShort":["jan","feb","mar","apr","maj","jun","jul","aug","sep","okt","nov","dec",""],"timePattern":"HH:mm","datePattern":"yyyy-MM-dd","dateTimePattern":"yyyy-MM-dd HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("tr-tr",{"dayNames":["Pazar","Pazartesi","Salı","Çarşamba","Perşembe","Cuma","Cumartesi"],"dayNamesShort":["Pz","Pt","Sa","Ça","Pe","Cu","Ct"],"monthNames":["Ocak","Şubat","Mart","Nisan","Mayıs","Haziran","Temmuz","Ağustos","Eylül","Ekim","Kasım","Aralık",""],"monthNamesShort":["Oca","Şub","Mar","Nis","May","Haz","Tem","Ağu","Eyl","Eki","Kas","Ara",""],"timePattern":"HH:mm","datePattern":"d.M.yyyy","dateTimePattern":"d.M.yyyy HH:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("uk-ua",{"dayNames":["неділя","понеділок","вівторок","середа","четвер","п'ятниця","субота"],"dayNamesShort":["Нд","Пн","Вт","Ср","Чт","Пт","Сб"],"monthNames":["січень","лютий","березень","квітень","травень","червень","липень","серпень","вересень","жовтень","листопад","грудень",""],"monthNamesShort":["Січ","Лют","Бер","Кві","Тра","Чер","Лип","Сер","Вер","Жов","Лис","Гру",""],"timePattern":"H:mm","datePattern":"dd.MM.yyyy","dateTimePattern":"dd.MM.yyyy H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("zh-cn",{"dayNames":["星期日","星期一","星期二","星期三","星期四","星期五","星期六"],"dayNamesShort":["日","一","二","三","四","五","六"],"monthNames":["一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月",""],"monthNamesShort":["1月","2月","3月","4月","5月","6月","7月","8月","9月","10月","11月","12月",""],"timePattern":"H:mm","datePattern":"yyyy/M/d","dateTimePattern":"yyyy/M/d H:mm","timeFormat":"Clock24Hours","weekStarts":1})),DayPilot.Locale.register(new DayPilot.Locale("zh-tw",{"dayNames":["星期日","星期一","星期二","星期三","星期四","星期五","星期六"],"dayNamesShort":["日","一","二","三","四","五","六"],"monthNames":["一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月",""],"monthNamesShort":["一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月",""],"timePattern":"tt hh:mm","datePattern":"yyyy/M/d","dateTimePattern":"yyyy/M/d tt hh:mm","timeFormat":"Clock12Hours","weekStarts":0})),DayPilot.Locale.US=DayPilot.Locale.find("en-us"),DayPilot.Switcher=function(e){function t(e,t,a){var n={};n.start=e,n.end=t,n.day=a,n.target=i.k.control,n.preventDefault=function(){this.preventDefault.value=!0};var o=i.G;o&&o.start===n.start&&o.end===n.end&&o.day===n.day&&o.target===n.target||(i.G=n,"function"==typeof i.onChange&&(i.onChange(n),n.preventDefault.value)||"function"==typeof i.onTimeRangeSelect&&(i.onTimeRangeSelect(n),n.preventDefault.value)||(i.k.q(i.l),"function"==typeof i.onChanged&&i.onChanged(n),"function"==typeof i.onTimeRangeSelected&&i.onTimeRangeSelected(n)))}var i=this;this.f=[],this.i=[],this.j={},this.selectedClass=null,this.k=null,this.l=DayPilot.Date.today(),this.onChange=null,this.onChanged=null,this.onSelect=null,this.j.updateMode=function(e){var t=i.j.control;t&&(t.selectMode=e,t.select(i.l))},this.addView=function(e,t){var a;if("string"==typeof e){if(a=document.getElementById(e),!a)throw"Element not found: "+e}else a=e;var n=a,o={};return o.m=!0,o.n=n.id,o.control=n,o.o=t||{},o.p=function(){n.hide?n.hide():n.nav&&n.nav.top?n.nav.top.style.display="none":n.style.display="none"},o.q=function(e){(function(){return!!n.backendUrl||!("function"!=typeof WebForm_DoCallback||!n.uniqueID)})()?n.commandCallBack&&n.commandCallBack("navigate",{"day":e}):(n.startDate=e,n.update())},o.s=function(){i.t(),n.show?n.show():n.nav&&n.nav.top?n.nav.top.style.display="":n.style.display=""},o.u=function(){if(o.o.navigatorSelectMode)return o.o.navigatorSelectMode;if(n.isCalendar)switch(n.viewType){case"Day":return"day";case"Week":return"week";case"WorkWeek":return"week";default:return"day"}else if(n.isMonth)switch(n.viewType){case"Month":return"month";case"Weeks":return"week";default:return"day"}return"day"},this.f.push(o),o},this.addTrigger=function(e,t){var a;if("string"==typeof e){if(a=document.getElementById(e),!a)throw"Element not found: "+e}else a=e;var n=this.z(t);n||(n=this.addView(t));var o={};return o.A=!0,o.B=a,o.n=a.id,o.C=n,o.D=function(e){i.show(o),i.E(o),e&&(e.preventDefault?e.preventDefault():e.returnValue=!1)},DayPilot.re(a,"click",o.D),this.i.push(o),o},this.addButton=this.addTrigger,this.select=function(e){var t=this.F(e);t?t.D():this.i.length>0&&this.i[0].D()},this.F=function(e){for(var t=0;t0){for(;e>=12;)e-=12,i++;e>12-a?(i++,a=e-(12-a)):a+=e}else{for(;e<=-12;)e+=12,i--;a+e<=0?(i--,a=12+a+e):a+=e}var n=new Date(t.getTime());n.setUTCDate(1),n.setUTCFullYear(i),n.setUTCMonth(a-1);var o=new DayPilot.Date(n).daysInMonth();return n.setUTCDate(Math.min(o,t.getUTCDate())),new DayPilot.Date(n)},DayPilot.Date.prototype.addSeconds=function(e){return e?this.addTime(1e3*e):this},DayPilot.Date.prototype.addTime=function(e){return e?(e instanceof DayPilot.Duration&&(e=e.ticks),new DayPilot.Date(this.ticks+e)):this},DayPilot.Date.prototype.addYears=function(e){var t=new Date(this.ticks),i=new Date(this.ticks),a=this.getYear()+e,n=this.getMonth();i.setUTCDate(1),i.setUTCFullYear(a),i.setUTCMonth(n);var o=new DayPilot.Date(i).daysInMonth();return i.setUTCDate(Math.min(o,t.getUTCDate())),new DayPilot.Date(i)},DayPilot.Date.prototype.dayOfWeek=function(){return new Date(this.ticks).getUTCDay()},DayPilot.Date.prototype.getDayOfWeek=function(){return new Date(this.ticks).getUTCDay()},DayPilot.Date.prototype.getDayOfYear=function(){var e=this.firstDayOfYear();return DayPilot.DateUtil.daysDiff(e,this)+1},DayPilot.Date.prototype.daysInMonth=function(){var e=new Date(this.ticks),t=e.getUTCMonth()+1,i=e.getUTCFullYear(),a=[31,28,31,30,31,30,31,31,30,31,30,31];return 2!==t?a[t-1]:i%4!==0?a[1]:i%100===0&&i%400!==0?a[1]:a[1]+1},DayPilot.Date.prototype.daysInYear=function(){var e=this.getYear();return e%4!==0?365:e%100===0&&e%400!==0?365:366},DayPilot.Date.prototype.dayOfYear=function(){return Math.ceil((this.getDatePart().getTime()-this.firstDayOfYear().getTime())/864e5)+1},DayPilot.Date.prototype.equals=function(e){if(null===e)return!1;if(e instanceof DayPilot.Date)return this===e;throw"The parameter must be a DayPilot.Date object (DayPilot.Date.equals())"},DayPilot.Date.prototype.firstDayOfMonth=function(){var e=new Date;return e.setUTCFullYear(this.getYear(),this.getMonth(),1),e.setUTCHours(0),e.setUTCMinutes(0),e.setUTCSeconds(0),e.setUTCMilliseconds(0),new DayPilot.Date(e)},DayPilot.Date.prototype.firstDayOfYear=function(){var e=this.getYear(),t=new Date;return t.setUTCFullYear(e,0,1),t.setUTCHours(0),t.setUTCMinutes(0),t.setUTCSeconds(0),t.setUTCMilliseconds(0),new DayPilot.Date(t)},DayPilot.Date.prototype.firstDayOfWeek=function(e){var t=this;if(e instanceof DayPilot.Locale)e=e.weekStarts;else if("string"==typeof e&&DayPilot.Locale.find(e)){var i=DayPilot.Locale.find(e);e=i.weekStarts}else e=e||0;for(var a=t.dayOfWeek();a!==e;)t=t.addDays(-1),a=t.dayOfWeek();return new DayPilot.Date(t)},DayPilot.Date.prototype.getDay=function(){return new Date(this.ticks).getUTCDate()},DayPilot.Date.prototype.getDatePart=function(){var e=new Date(this.ticks);return e.setUTCHours(0),e.setUTCMinutes(0),e.setUTCSeconds(0),e.setUTCMilliseconds(0),new DayPilot.Date(e)},DayPilot.Date.prototype.getYear=function(){return new Date(this.ticks).getUTCFullYear()},DayPilot.Date.prototype.getHours=function(){return new Date(this.ticks).getUTCHours()},DayPilot.Date.prototype.getMilliseconds=function(){return new Date(this.ticks).getUTCMilliseconds()},DayPilot.Date.prototype.getMinutes=function(){return new Date(this.ticks).getUTCMinutes()},DayPilot.Date.prototype.getMonth=function(){return new Date(this.ticks).getUTCMonth()},DayPilot.Date.prototype.getSeconds=function(){return new Date(this.ticks).getUTCSeconds()},DayPilot.Date.prototype.getTotalTicks=function(){return this.getTime()},DayPilot.Date.prototype.getTime=function(){return this.ticks},DayPilot.Date.prototype.getTimePart=function(){var e=this.getDatePart();return DayPilot.DateUtil.diff(this,e)},DayPilot.Date.prototype.lastDayOfMonth=function(){var e=new Date(this.firstDayOfMonth().getTime()),t=this.daysInMonth();return e.setUTCDate(t),new DayPilot.Date(e)},DayPilot.Date.prototype.weekNumber=function(){var e=this.firstDayOfYear(),t=(this.getTime()-e.getTime())/864e5;return Math.ceil((t+e.dayOfWeek()+1)/7)},DayPilot.Date.prototype.weekNumberISO=function(){var e=!1,t=this.dayOfYear(),i=this.firstDayOfYear().dayOfWeek(),a=this.firstDayOfYear().addYears(1).addDays(-1).dayOfWeek();0===i&&(i=7),0===a&&(a=7);var n=8-i;4!==i&&4!==a||(e=!0);var o=Math.ceil((t-n)/7),r=o;return n>=4&&(r+=1),r>52&&!e&&(r=1),0===r&&(r=this.firstDayOfYear().addDays(-1).weekNumberISO()),r},DayPilot.Date.prototype.toDateLocal=function(){var e=new Date(this.ticks),t=new Date;return t.setFullYear(e.getUTCFullYear(),e.getUTCMonth(),e.getUTCDate()),t.setHours(e.getUTCHours()),t.setMinutes(e.getUTCMinutes()),t.setSeconds(e.getUTCSeconds()),t.setMilliseconds(e.getUTCMilliseconds()),t},DayPilot.Date.prototype.toDate=function(){return new Date(this.ticks)},DayPilot.Date.prototype.toJSON=function(){return this.value},DayPilot.Date.prototype.toString=function(e,t){return e?new n(e,t).print(this):this.toStringSortable()},DayPilot.Date.prototype.toStringSortable=function(){return e(this.ticks)},DayPilot.Date.parse=function(e,t,i){return new n(t,i).parse(e)};DayPilot.Date.today=function(){return new DayPilot.Date(DayPilot.DateUtil.localToday(),!0)},DayPilot.Date.now=function(){return new DayPilot.Date},DayPilot.Date.fromYearMonthDay=function(e,t,i){t=t||1,i=i||1;var a=new Date(0);return a.setUTCFullYear(e),a.setUTCMonth(t-1),a.setUTCDate(i),new DayPilot.Date(a)},DayPilot.DateUtil={},DayPilot.DateUtil.fromStringSortable=function(e,t){if(!e)throw"Can't create DayPilot.Date from an empty string";var i=e.length,a=10===i,n=19===i,o=i>19;if(!a&&!n&&!o)throw"Invalid string format (use '2010-01-01' or '2010-01-01T00:00:00'): "+e;if(DayPilot.Date.Cache.Parsing[e]&&!t)return DayPilot.Stats.cacheHitsParsing+=1,DayPilot.Date.Cache.Parsing[e];var r=e.substring(0,4),s=e.substring(5,7),l=e.substring(8,10),d=new Date(0);if(d.setUTCFullYear(r,s-1,l),a)return DayPilot.Date.Cache.Parsing[e]=d,d;var c=e.substring(11,13),u=e.substring(14,16),h=e.substring(17,19);if(d.setUTCHours(c),d.setUTCMinutes(u),d.setUTCSeconds(h),n)return DayPilot.Date.Cache.Parsing[e]=d,d;var f=e[19],m=0;if("."===f){var p=parseInt(e.substring(20,23));d.setUTCMilliseconds(p),m=DayPilot.DateUtil.getTzOffsetMinutes(e.substring(23))}else m=DayPilot.DateUtil.getTzOffsetMinutes(e.substring(19));var v=new DayPilot.Date(d);return t||(v=v.addMinutes(-m)),d=v.toDate(),DayPilot.Date.Cache.Parsing[e]=d,d},DayPilot.DateUtil.getTzOffsetMinutes=function(e){if(DayPilot.Util.isNullOrUndefined(e)||""===e)return 0;if("Z"===e)return 0;var t=e[0],i=parseInt(e.substring(1,3)),a=parseInt(e.substring(4)),n=60*i+a;if("-"===t)return-n;if("+"===t)return n;throw"Invalid timezone spec: "+e},DayPilot.DateUtil.hasTzSpec=function(e){return!!e.indexOf("+")||!!e.indexOf("-")},DayPilot.DateUtil.daysDiff=function(e,t){if(e&&t||function(){throw"two parameters required"}(),e=new DayPilot.Date(e),t=new DayPilot.Date(t),e.getTime()>t.getTime())return null;for(var i=0,a=e.getDatePart(),n=t.getDatePart();at.getTime()?e:t},DayPilot.DateUtil.min=function(e,t){return e.getTime()0){var d=l[0],c=i(d);c?r.push(c):r.push(d),o+=d.length,n=e.length<=o}else n=!0}for(var u=0;u12||i<1)return null;var a=this.day.findValue(e),n=DayPilot.Date.fromYearMonthDay(t,i).daysInMonth();if(a<1||a>n)return null;var o=this.hours?this.hours.findValue(e):0,r=this.minutes?this.minutes.findValue(e):0,s=this.seconds?this.seconds.findValue(e):0,l=this.ampm?this.ampm.findValue(e):null;if(this.ampm&&this.hours12){var d=this.hours12.findValue(e);if(d<1||d>12)return null;o="PM"===l?12===d?12:d+12:12===d?0:d}if(o<0||o>23)return null;if(r<0||r>59)return null;if(s<0||s>59)return null;var c=new Date;return c.setUTCFullYear(t,i-1,a),c.setUTCHours(o),c.setUTCMinutes(r),c.setUTCSeconds(s),c.setUTCMilliseconds(0),new DayPilot.Date(c)},this.init()};DayPilot.ColorUtil={},DayPilot.ColorUtil.hexToRgb=function(e){if(!/^#[0-9a-f]{6}$/i.test(e))throw new DayPilot.Exception("Invalid color, only full hex color string accepted, eg. '#ffaaff'.");return e=e.replace("#",""),{r:parseInt(e.substring(0,2),16),g:parseInt(e.substring(2,4),16),b:parseInt(e.substring(4,6),16)}},DayPilot.ColorUtil.rgbToHex=function(e){return"#"+i(e.r)+i(e.g)+i(e.b)},DayPilot.ColorUtil.adjustLuminance=function(e,t){return{r:e.r+t,g:e.g+t,b:e.b+t}},DayPilot.ColorUtil.darker=function(e,t){var i=DayPilot.ColorUtil.hexToRgb(e);t=t||1;var a=17,n=t*a,o=DayPilot.ColorUtil.adjustLuminance(i,-n);return DayPilot.ColorUtil.rgbToHex(o)},DayPilot.ColorUtil.lighter=function(e,t){return"number"!=typeof t&&(t=1),DayPilot.ColorUtil.darker(e,-t)},DayPilot.ColorUtil.pl=function(e){var t=DayPilot.ColorUtil.hexToRgb(e),i=t.r/255,a=t.g/255,n=t.b/255;return Math.sqrt(.299*i*i+.587*a*a+.114*n*n)},DayPilot.ColorUtil.contrasting=function(e,t,i){var a=DayPilot.ColorUtil.pl(e);return t=t||"#ffffff",i=i||"#000000",a>.5?i:t},DayPilot.Event=function(e,t,i){var a=this;this.calendar=t,this.data=e?e:{},this.part=i?i:{},"undefined"==typeof this.data.id&&(this.data.id=this.data.value);var n={},o=["id","text","start","end","resource"];this.isEvent=!0,this.temp=function(){if(n.dirty)return n;for(var e=0;ethis.businessEndsHour?24-this.businessBeginsHour+this.businessEndsHour:this.businessEndsHour-this.businessBeginsHour},this.na=function(){return this.la()/18e5},this.la=function(){var e=0;return e="BusinessHoursNoScroll"===this.heightSpec?this.ma():24,60*e*60*1e3},this.oa=function(){return"BusinessHoursNoScroll"===this.heightSpec?this.businessBeginsHour:0},this.pa=function(){return 2===n.api},this.eventClickCallBack=function(e,t){this.R("EventClick",t,e)},this.eventClickPostBack=function(e,t){this.P("EventClick",t,e)},this.qa=function(e){var t=this,i=t.event;if(n.pa()){var a={};if(a.e=i,a.originalEvent=e,a.meta=e.metaKey,a.ctrl=e.ctrlKey,a.control=n,a.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof n.onEventClick&&(n.ra.apply(function(){n.onEventClick(a)}),a.preventDefault.value))return;switch(n.eventClickHandling){case"CallBack":n.eventClickCallBack(i);break;case"PostBack":n.eventClickPostBack(i);break;case"ContextMenu":var o=i.client.contextMenu();o?o.show(i):n.contextMenu&&n.contextMenu.show(i)}"function"==typeof n.onEventClicked&&n.ra.apply(function(){n.onEventClicked(a)})}else switch(n.eventClickHandling){case"PostBack":n.eventClickPostBack(i);break;case"CallBack":n.eventClickCallBack(i);break;case"JavaScript":n.onEventClick(i)}},this.sa=function(e){var t=this.event;if(e.stopPropagation&&e.stopPropagation(),!t.client.rightClickEnabled())return!1;var i={};if(i.e=t,i.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof n.onEventRightClick&&(n.onEventRightClick(i),i.preventDefault.value))return!1;switch(n.eventRightClickHandling){case"ContextMenu":var a=t.client.contextMenu();a?a.show(t):n.contextMenu&&n.contextMenu.show(this.event)}return"function"==typeof n.onEventRightClicked&&n.onEventRightClicked(i),e.preventDefault&&e.preventDefault(),!1},this.eventDeleteCallBack=function(e,t){this.R("EventDelete",t,e)},this.eventDeletePostBack=function(e,t){this.P("EventDelete",t,e)},this.ta=function(e){if(n.pa()){var t={};if(t.e=e,t.control=n,t.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof n.onEventDelete&&(n.ra.apply(function(){n.onEventDelete(t)}),t.preventDefault.value))return;switch(n.eventDeleteHandling){case"CallBack":n.eventDeleteCallBack(e);break;case"PostBack":n.eventDeletePostBack(e);break;case"Update":n.events.remove(e)}"function"==typeof n.onEventDeleted&&n.ra.apply(function(){n.onEventDeleted(t)})}else switch(n.eventDeleteHandling){case"PostBack":n.eventDeletePostBack(e);break;case"CallBack":n.eventDeleteCallBack(e);break;case"JavaScript":n.onEventDelete(e)}},this.eventResizeCallBack=function(e,t,i,a){if(!t)throw"newStart is null";if(!i)throw"newEnd is null";var n={};n.e=e,n.newStart=t,n.newEnd=i,this.R("EventResize",a,n)},this.eventResizePostBack=function(e,t,i,a){if(!t)throw"newStart is null";if(!i)throw"newEnd is null";var n={};n.e=e,n.newStart=t,n.newEnd=i,this.P("EventResize",a,n)},this.I=function(e,t,i,a){var o=1,r=new Date,s=new Date,l=e.start(),d=e.end();if("top"===a){var c=l.getDatePart(),u=Math.floor((i-o)/n.cellHeight),h=30*u,f=60*h*1e3,m=60*n.oa()*60*1e3;r=c.addTime(f+m),s=e.end()}else if("bottom"===a){var c=d.getDatePart(),u=Math.floor((i+t-o)/n.cellHeight),h=30*u,f=60*h*1e3,m=60*n.oa()*60*1e3;r=l,s=c.addTime(f+m)}if(n.pa()){var p={};if(p.e=e,p.control=n,p.newStart=r,p.newEnd=s,p.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof n.onEventResize&&(n.ra.apply(function(){n.onEventResize(p)}),p.preventDefault.value))return;switch(n.eventResizeHandling){case"PostBack":n.eventResizePostBack(e,r,s);break;case"CallBack":n.eventResizeCallBack(e,r,s);break;case"Update":e.start(r),e.end(s),n.events.update(e)}"function"==typeof n.onEventResized&&n.ra.apply(function(){n.onEventResized(p)})}else switch(n.eventResizeHandling){case"PostBack":n.eventResizePostBack(e,r,s);break;case"CallBack":n.eventResizeCallBack(e,r,s);break;case"JavaScript":n.onEventResize(e,r,s)}},this.eventMovePostBack=function(e,t,i,a,n){if(!t)throw"newStart is null";if(!i)throw"newEnd is null";var o={};o.e=e,o.newStart=t,o.newEnd=i,this.P("EventMove",n,o)},this.eventMoveCallBack=function(e,t,i,a,n){if(!t)throw"newStart is null";if(!i)throw"newEnd is null";var o={};o.e=e,o.newStart=t,o.newEnd=i,this.R("EventMove",n,o)},this.K=function(e,t,i,a){var o=1,r=Math.floor((i-o)/n.cellHeight),s=30*r*60*1e3,l=e.start(),d=e.end(),c=new Date;l instanceof DayPilot.Date&&(l=l.toDate()),c.setTime(Date.UTC(l.getUTCFullYear(),l.getUTCMonth(),l.getUTCDate()));var u=l.getTime()-(c.getTime()+3600*l.getUTCHours()*1e3+30*Math.floor(l.getUTCMinutes()/30)*60*1e3),h=d.getTime()-l.getTime(),f=this.ua[t],m=f.id,p=f.start.getTime(),v=new Date;v.setTime(p+s+u);var y=new DayPilot.Date(v),g=y.addTime(h);if(n.pa()){var b={};if(b.e=e,b.newStart=y,b.newEnd=g,b.newResource=m,b.ctrl=a.ctrlKey,b.shift=a.shiftKey,b.control=n,b.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof n.onEventMove&&(n.ra.apply(function(){n.onEventMove(b)}),b.preventDefault.value))return;switch(n.eventMoveHandling){case"PostBack":n.eventMovePostBack(e,y,g,f.id);break;case"CallBack":n.eventMoveCallBack(e,y,g,f.id);break;case"Update":e.start(y),e.end(g),e.resource(m),n.events.update(e)}"function"==typeof n.onEventMoved&&n.ra.apply(function(){n.onEventMoved(b)})}else switch(n.eventMoveHandling){case"PostBack":n.eventMovePostBack(e,y,g,f.id);break;case"CallBack":n.eventMoveCallBack(e,y,g,f.id);break;case"JavaScript":n.onEventMove(e,y,g,f.id,!1)}},this.timeRangeSelectedPostBack=function(e,t,i,a){var n={};n.start=e,n.end=t,this.P("TimeRangeSelected",a,n)},this.timeRangeSelectedCallBack=function(e,t,i,a){var n={};n.start=e,n.end=t,this.R("TimeRangeSelected",a,n)},this.L=function(e,t,i){if(e=new DayPilot.Date(e),t=new DayPilot.Date(t),this.pa()){var a={};if(a.start=e,a.end=t,a.resource=i,a.control=n,a.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof n.onTimeRangeSelect&&(n.ra.apply(function(){n.onTimeRangeSelect(a)}),a.preventDefault.value))return;switch(n.timeRangeSelectedHandling){case"PostBack":n.timeRangeSelectedPostBack(e,t);break;case"CallBack":n.timeRangeSelectedCallBack(e,t)}"function"==typeof n.onTimeRangeSelected&&n.ra.apply(function(){n.onTimeRangeSelected(a)})}else switch(n.timeRangeSelectedHandling){case"PostBack":n.timeRangeSelectedPostBack(e,t);break;case"CallBack":n.timeRangeSelectedCallBack(e,t);break;case"JavaScript":n.onTimeRangeSelected(e,t)}},this.va=function(e){if(!t.selecting&&"Disabled"!==n.timeRangeSelectedHandling){var i=e.which;if(1===i||0===i)return t.firstMousePos=DayPilot.mc(e),t.selecting={},t.selecting.calendar=n,t.selectedCells&&(n.clearSelection(),t.selectedCells=[]),t.column=t.getColumn(this),t.selectedCells.push(this),t.firstSelected=this,t.topSelectedCell=this,t.bottomSelectedCell=this,n.wa(),!1}},this.wa=function(){this.getSelection();!function(){var e=t.topSelectedCell,i=t.bottomSelectedCell,a=function(){if(e.data)return e.data.x;for(var t=e.parentNode.cells,i=0;ithis.max-1)},i.putIntoLine=function(e){for(var t=0;to.part.top+o.part.height-1))return!1}return!0},i.push(e),this.lines.push(i),this.lines.length-1},i.events.push(e),i.min=e.part.top,i.max=e.part.top+e.part.height,this.blocks.push(i),this.blocks.length-1},t.putIntoLine=function(e){for(var t=0;to.part.top+o.part.height-1))return!1}return!0},i.push(e),this.lines.push(i),this.lines.length-1},t},this.Ba=function(){var e=[],t=this.startDate.getDatePart(),i=this.days;switch(this.viewType){case"Day":i=1;break;case"Week":i=7;var a=o.ba();t=t.firstDayOfWeek(a);break;case"WorkWeek":i=5,t=t.firstDayOfWeek(1)}"BusinessHoursNoScroll"===this.heightSpec&&(t=t.addHours(this.businessBeginsHour));for(var r=0;r0;)this.fasterDispose||DayPilot.pu(e.rows[0]),e.deleteRow(0);this.tableCreated=!0;for(var r=o.length,s=this.nav.events;s&&s.rows&&s.rows.length>0;)this.fasterDispose||DayPilot.pu(s.rows[0]),s.deleteRow(0);for(var r=o.length,l=s.insertRow(-1),d=0;dd&&(l=d-t.originalTop),t.resizingShadow.style.height=l+"px"}else if("top"===t.resizing.dpBorder){var c=Math.floor((t.originalTop+s-r+o/2)/o)*o+r;ct.originalTop+t.originalHeight-o&&(c=t.originalTop+t.originalHeight-o);var l=t.originalHeight-(c-t.originalTop);ld&&(c=d-p),DayPilot.Util.addClass(t.moving,n.J("_event_moving_source")),t.movingShadow.parentNode.style.display="none",t.movingShadow.style.top=c+"px",t.movingShadow.parentNode.style.display="";var v=m.clientWidth/m.rows[0].cells.length,y=Math.floor((n.coords.x-45)/v);y<0&&(y=0),y=0&&t.movingShadow.column!==y&&(t.movingShadow.column=y,t.moveShadow(m.rows[0].cells[y]))}else if(t.selecting){var a=DayPilot.mc(e),g=n.za.getCellCoords(),b=t.column,D=n.nav.main.rows[g.y].cells[b];a.y=this.businessEndsHour||6===e.getDayOfWeek()||0===e.getDayOfWeek()):e.getHours()>=this.businessBeginsHour||e.getHours()0;)this.fasterDispose||DayPilot.pu(e.rows[0]),e.deleteRow(0);this.headerCreated=!0;this.Ra(t)},this.loadingStart=function(){this.loadingLabelVisible&&(this.nav.loading.innerHTML=this.loadingLabelText,this.nav.loading.style.top=this.headerHeight+5+"px",this.nav.loading.style.display="")},this.commandCallBack=function(e,t){var i={};i.command=e,this.R("Command",t,i)},this.loadingStop=function(e){this.callbackTimeout&&window.clearTimeout(this.callbackTimeout),this.nav.loading.style.display="none"},this.Va=function(){
+var e=this.nav.scroll;e.onscroll||(e.onscroll=function(){n.Wa()});var t="undefined"!=typeof this.Xa.scrollpos?this.Xa.scrollpos:this.initScrollPos;t&&("Auto"===t&&(t="BusinessHours"===this.heightSpec?2*this.cellHeight*this.businessBeginsHour:0),e.root=this,0===e.scrollTop&&(e.scrollTop=t))},this.callbackError=function(e,t){alert("Error!\r\nResult: "+e+"\r\nContext:"+t)},this.Ya=function(){var e=DayPilot.sw(this.nav.scroll),t=this.nav.cornerRight;t&&(t.style.width=e+"px")},this.Za=function(){t.globalHandlers||(t.globalHandlers=!0,DayPilot.re(document,"mouseup",t.gMouseUp),DayPilot.re(window,"unload",t.gUnload))},this.events={},this.events.add=function(e){var t=null;if(e instanceof DayPilot.Event)t=e.data;else{if("object"!=typeof e)throw"DayPilot.Calendar.events.add() expects an object or DayPilot.Event instance.";t=e}n.events.list||(n.events.list=[]),n.events.list.push(t),n.$a({"eventsOnly":!0}),n.ra.notify()},this.events.find=function(e){if(!n.events.list)return null;if("function"==typeof e){for(var t=e,i=0;i-1?"&"+s:"?"+s,DayPilot.Http.ajax({"method":"GET","url":r,"success":o,"error":a})}},this._a=function(){if(n.nav.top.className!==n.J("_main")){n.nav.top.className=n.J("_main");var e=n.nav.corner;e.className=n.J("_corner"),e.firstChild.className=n.J("_corner_inner");var t=n.nav.cornerRight;t&&(t.className=n.J("_cornerright"),t.firstChild.className=n.J("_cornerright_inner"))}},this.update=function(e){if(n.N)throw new DayPilot.Exception("You are trying to update a DayPilot.Calendar instance that has been disposed.");n.ab(e),n.$a()},this.$a=function(e){if(this.Aa){var e=e||{},t=!e.eventsOnly;n.bb(),n.V(),n.nav.top.style.cursor="auto",t&&(n.ca(),n.fa(),n.ga(),n.ha(),n.ia(),n.Oa(),n.Ya(),n._a(),n.cb()),n.da(),n.ea(),n.ja(),n.clearSelection(),this.visible?this.show():this.hide()}},this.db=null,this.ab=function(e){if(e){var t={"events":{"preInit":function(){var e=this.data||[];DayPilot.isArray(e.list)?n.events.list=e.list:n.events.list=e}},"columns":{"preInit":function(){n.columns.list=this.data}}};this.db=t;for(var i in e)if(t[i]){var a=t[i];a.data=e[i],a.preInit&&a.preInit()}else n[i]=e[i]}},this.eb=function(){var e=this.db;for(var t in e){var i=e[t];i.postInit&&i.postInit()}},this.fb=function(){if(this.id&&this.id.tagName)this.nav.top=this.id;else{if("string"!=typeof this.id)throw"DayPilot.Calendar() constructor requires the target element or its ID as a parameter";if(this.nav.top=document.getElementById(this.id),!this.nav.top)throw"DayPilot.Calendar: The placeholder element not found: '"+e+"'."}},this.gb={},this.gb.events=[],this.hb=function(e){var t=this.gb.events,i=this.events.list[e],a={};for(var o in i)a[o]=i[o];if("function"==typeof this.onBeforeEventRender){var r={};r.control=n,r.data=a,this.onBeforeEventRender(r)}t[e]=a},this.da=function(){var e=this.events.list;if(n.gb.events=[],e){if(!DayPilot.isArray(e))throw new DayPilot.Exception("DayPilot.Calendar.events.list expects an array object. You supplied: "+typeof e);var t=e.length,i=864e5;this.cache.pixels={};var a=[];this.scrollLabels=[],this.minStart=1e4,this.maxEnd=0;for(var o=0;o=f);if("Resources"===n.viewType&&(b=b&&d.id===r.resource),b){var D=new DayPilot.Event(r,n);D.part.dayIndex=o,D.part.start=ug?r.end:h;var x=this.getPixels(D.part.start,d.start),w=this.getPixels(D.part.end,d.start),C=x.top,P=w.top;if(C===P&&(x.cut||w.cut))continue;var k=w.boxBottom;D.part.top=Math.floor(C/this.cellHeight)*this.cellHeight+1,D.part.height=Math.max(Math.ceil(k/this.cellHeight)*this.cellHeight-D.part.top,this.cellHeight-1)+1,D.part.barTop=Math.max(C-D.part.top-1,0),D.part.barHeight=Math.max(P-C-2,1);var p=D.part.top,v=D.part.top+D.part.height;p>l.maxStart&&(l.maxStart=p),vthis.maxEnd&&(this.maxEnd=v),d.events.push(D),"function"==typeof this.onBeforeEventRender&&(D.cache=this.gb.events[m]),D.part.start.getTime()===y&&D.part.end.getTime()===g&&(a[m]=!0)}}}}for(var o=0;o0&&(this.nav.scroll.style.height=e+"px")},this.ra={},this.ra.scope=null,this.ra.notify=function(){n.ra.scope&&n.ra.scope["$apply"]()},this.ra.apply=function(e){e()},this.Wa=function(){if(n.nav.scroll){var e=n.nav.scroll.scrollTop,t=e/(2*n.cellHeight);n.Xa.scrollHour=t}},this.cb=function(){var e=0;"number"==typeof n.Xa.scrollHour?e=2*n.cellHeight*n.Xa.scrollHour:"Auto"===n.initScrollPos&&(e="BusinessHours"===this.heightSpec?2*this.cellHeight*this.businessBeginsHour:0),n.nav.scroll.scrollTop=e},this.kb=function(){return!(!this.backendUrl&&"function"!=typeof WebForm_DoCallback)&&("undefined"==typeof n.events.list||!n.events.list)},this.s=function(){"hidden"===this.nav.top.style.visibility&&(this.nav.top.style.visibility="visible")},this.show=function(){n.visible=!0,n.nav.top.style.display="",this.Ya()},this.hide=function(){n.visible=!1,n.nav.top.style.display="none"},this.lb=function(){this.bb(),this.ca(),this.Fa(),this.fa(),this.ga(),this.Ya(),this.Va(),this.Za(),t.register(this),this.mb(),this.R("Init")},this.Xa={},this.nb=function(){this.Xa.themes=[],this.Xa.themes.push(this.theme||this.cssClassPrefix)},this.ob=function(){for(var e=this.Xa.themes,t=0;t0&&e.offsetHeight>0)},this.mb=function(){var e=n.sb;e()||(n.U=setInterval(function(){e()&&(n.Va(),n.Ya(),clearInterval(n.U))},100))},this.Ja=function(e,t){return n._.aa()?DayPilot.Util.escapeTextHtml(e,t):DayPilot.Util.isNullOrUndefined(t)?DayPilot.Util.isNullOrUndefined(e)?"":e:t},this.internal={},this.internal.loadOptions=n.ab,this.internal.xssTextHtml=n.Ja,this.init=function(){this.fb();var e=this.kb();return this.nb(),e?void this.lb():(this.bb(),this.ca(),this.da(),this.Fa(),this.fa(),this.ga(),this.s(),this.Ya(),this.Va(),this.Za(),t.register(this),this.events&&(this.ea(),this.ja()),this.pb(),this.qb(),this.mb(),this.Aa=!0,this)},this.Init=this.init,this.ab(i)},DayPilot.CalendarColumn=function(e,t){var i=this;i.id=e.id,i.name=e.name,i.data=e.data,i.start=new DayPilot.Date(e.start),i.calendar=t,i.toJSON=function(){var e={};return e.id=this.id,this.start&&(e.start=this.start.toString()),e.name=this.name,e}},DayPilot.Calendar=t.Calendar,"undefined"!=typeof jQuery&&!function(e){e.fn.daypilotCalendar=function(e){var t=null,i=this.each(function(){if(!this.daypilot){var i=new DayPilot.Calendar(this.id);this.daypilot=i;for(name in e)i[name]=e[name];i.init(),t||(t=i)}});return 1===this.length?t:i}}(jQuery),function(){var e=DayPilot.am();e&&e.directive("daypilotCalendar",["$parse",function(e){return{"restrict":"E","template":"","replace":!0,"link":function(t,i,a){var n=new DayPilot.Calendar(i[0]);n.ra.scope=t,n.init();var o=a["id"];o&&(t[o]=n);var r=a["publishAs"];if(r){(0,e(r).assign)(t,n)}for(var s in a)0===s.indexOf("on")&&!function(i){n[i]=function(n){var o=e(a[i]);t["$apply"](function(){o(t,{"args":n})})}}(s);var l=t["$watch"],d=a["config"]||a["daypilotConfig"],c=a["events"]||a["daypilotEvents"];l.call(t,d,function(e){for(var t in e)n[t]=e[t];n.update(),n.qb()},!0),l.call(t,c,function(e){n.events.list=e,n.update()},!0)}}}])}()}}(),"undefined"==typeof DayPilot)var DayPilot={};if(function(){"undefined"!=typeof DayPilot.DatePicker&&DayPilot.DatePicker.close||(DayPilot.DatePicker=function(e){this.v="2024.3.539-lite";var t="navigator_"+(new Date).getTime(),i=this;this.onShow=null,this.onTimeRangeSelect=null,this.onTimeRangeSelected=null,this.prepare=function(){if(this.locale="en-us",this.target=null,this.targetAlignment="left",this.resetTarget=!0,this.pattern=this._.locale().datePattern,this.theme="navigator_default",this.patterns=[],this.zIndex=null,e)for(var t in e)this[t]=e[t]},this.init=function(){this.date=new DayPilot.Date(this.date);var e=this.tb();this.resetTarget&&!e?this.ub(this.date):this.resetTarget||(i.date=e);var t=this.B();return t&&t.addEventListener("input",function(){i.date=i.tb(),i.date&&i.navigator.select(i.date,{dontNotify:!0})}),document.addEventListener("mousedown",function(){i.close()}),this},this.close=function(){this.sb&&(this.sb=!1,this.navigator&&this.navigator.dispose(),this.div.innerHTML="",this.div&&this.div.parentNode===document.body&&document.body.removeChild(this.div))},this.setDate=function(e){this.date=new DayPilot.Date(e),this.ub(this.date)},this.tb=function(){var e=this.B();if(!e)return this.date;var t=null;if(t="INPUT"===e.tagName?e.value:e.innerText,!t)return null;for(var a=DayPilot.Date.parse(t,i.pattern),n=0;nA-M&&0!==A){var r=i-a-(A-M)+E;d.style.top=i-r+"px"}else d.style.top=i+"px";if("right"===o.align&&(e-=T),e-n>H-T&&0!==H){var s=e-n-(H-T)+E;d.style.left=e-s+"px"}else d.style.left=e+"px"}(),o.parentLink){var N=o.parentLink,I=parseInt(new DayPilot.StyleReader(d).get("border-top-width")),R=DayPilot.abs(o.parentLink.parentNode),O=R.x+N.offsetWidth,z=R.y-I;O+T>H&&(O=Math.max(0,R.x-T));var U=document.body.scrollTop+document.documentElement.scrollTop;z+M-U>A&&(z=Math.max(0,A-M+U)),d.style.left=O+"px",d.style.top=z+"px"}d.style.display="",this.addShadow(d),this.wb.div=d,o.submenu||(DayPilot.Menu.active=this)}},this.yb=function(e,t){var i=e,n=t.source;if((!a.wb.submenu||a.wb.submenu.item!==e)&&(a.wb.submenu&&a.wb.submenu.item!==e&&(DayPilot.Util.removeClass(a.wb.submenu.link.parentNode,a.xb("item_haschildren_active")),a.wb.submenu.menu.hide(),a.wb.submenu=null),e.items)){var o=a.cloneOptions();o.items=e.items,a.wb.submenu={},a.wb.submenu.menu=new DayPilot.Menu(o),a.wb.submenu.menu.zb=a,a.wb.submenu.menu.show(n,{"submenu":!0,"parentLink":t,"parentItem":i}),a.wb.submenu.item=e,a.wb.submenu.link=t,DayPilot.Util.addClass(t.parentNode,a.xb("item_haschildren_active"))}},this.xb=function(e){var t=this.theme||this.cssClassPrefix,i=this.cssOnly?"_":"";return t?t+i+e:""},this.cloneOptions=function(){return DayPilot.Util.copyProps(o,{},["cssClassPrefix","theme","hideAfter","hideOnMouseOut","zIndex"])},this.hide=function(e){e=e||{},this.wb.submenu&&this.wb.submenu.menu.hide();var i=t.waitingSubmenu;if(i&&(t.waitingSubmenu=null,clearTimeout(i.timeout)),this.removeShadow(),this.wb.div&&this.wb.div.parentNode===document.body&&document.body.removeChild(this.wb.div),n&&(DayPilot.de(n),n=null),a.wb.visible=!1,a.wb.source=null,a.zb&&e.hideParent&&a.zb.hide(e),DayPilot.Menu.active===a&&(DayPilot.Menu.active=null),"function"==typeof this.onHide){var o={};this.onHide(o)}},this.delayedHide=function(e){t.hideTimeout=setTimeout(function(){a.hide(e)},a.hideAfter)},this.cancelHideTimeout=function(){clearTimeout(t.hideTimeout)},this.init=function(e){return t.mouseMove(e),this},this.addShadow=function(e){},this.removeShadow=function(){};var o=DayPilot.isArray(i)?null:i;if(o)for(var r in o)this[r]=o[r]},DayPilot.MenuBar=function(e,t){var i=this;t=t||{},this.items=[],this.theme="menubar_default",this.windowMargin=0,this.nav={},this.elements={},this.elements.items=DayPilot.list(),this.k=null,this.Aa=!1;for(var a in t)this[a]=t[a];this.Ab=function(e){return this.theme+"_"+e},this.s=function(){this.nav.top=document.getElementById(e);var t=this.nav.top;t.className=this.Ab("main"),DayPilot.list(i.items).forEach(function(e){var a=document.createElement("span");a.innerHTML=DayPilot.Util.escapeTextHtml(e.text,e.html),a.className=i.Ab("item"),e.cssClass&&a.classList.add(e.cssClass),a.data=e,a.onclick=function(t){if(i.active&&i.active.item===e)i.Bb();else if(e.children)return void i.Cb(a);if("function"==typeof e.onClick){var n={};n.item=e,n.originalEvent=t,e.onClick(n)}},a.onmousedown=function(e){e.stopPropagation()},a.onmouseover=function(){i.active&&i.active.item!==e&&i.Cb(a)},t.appendChild(a),i.elements.items.push(a)})},this.Bb=function(){var e=i.Ab("item_active");i.elements.items.forEach(function(t){DayPilot.Util.removeClass(t,e)}),i.active&&i.active.menu&&i.active.menu.hide(),i.active=null},this.Db=function(e){return!!i.active&&i.active.item===e.data},this.Cb=function(e){if(!i.Db(e)){i.Bb();var t=e.data,a=i.active={};a.item=t,a.div=e;var n=i.Ab("item_active");DayPilot.Util.addClass(e,n);var o=DayPilot.abs(e);if(t.children){a.menu=new DayPilot.Menu({"items":t.children});var r=o.x;"right"===t.align&&(r+=o.w),a.menu.show(null,{"x":r,"y":o.y+o.h,"align":t.align,"windowMargin":i.windowMargin})}DayPilot.MenuBar.active=i}},this.init=function(){return this.s(),this.Aa=!0,this},this.dispose=function(){this.Aa&&(this.nav.top.innerHTML="",this.elements.items=[])}},DayPilot.MenuBar.deactivate=function(){DayPilot.MenuBar.active&&(DayPilot.MenuBar.active.Bb(),DayPilot.MenuBar.active=null)},t.menuClean=function(){"undefined"!=typeof DayPilot.Menu.active&&DayPilot.Menu.active&&(DayPilot.Menu.active.hide(),DayPilot.Menu.active=null)},t.mouseDown=function(e){"undefined"!=typeof t&&(t.menuClean(),DayPilot.MenuBar.deactivate())},t.mouseMove=function(e){"undefined"!=typeof t&&(t.mouse=t.mousePosition(e))},t.touchMove=function(e){"undefined"!=typeof t&&(t.mouse=t.touchPosition(e))},t.touchStart=function(e){"undefined"!=typeof t&&(t.mouse=t.touchPosition(e))},t.touchEnd=function(e){},t.touchPosition=function(e){if(!e||!e.touches)return null;var t=e.touches[0],i={};return i.x=t.pageX,i.y=t.pageY,i},t.mousePosition=function(e){return DayPilot.mo3(null,e)},DayPilot.Menu.touchPosition=function(e){e.touches&&(t.mouse=t.touchPosition(e))},DayPilot.Menu.hide=function(e){if(e=e||{},e.calendar){var i=DayPilot.Menu.active;if(i){var a=i.wb.source;a&&a.calendar===e.calendar&&t.menuClean()}}else t.menuClean()},t.handlersRegistered||"undefined"==typeof document||(DayPilot.re(document,"mousemove",t.mouseMove),DayPilot.re(document,"mousedown",t.mouseDown),DayPilot.re(document,"touchmove",t.touchMove),DayPilot.re(document,"touchstart",t.touchStart),DayPilot.re(document,"touchend",t.touchEnd),t.handlersRegistered=!0),DayPilot.Menu.def={}}}(DayPilot),"undefined"==typeof DayPilot&&(DayPilot={}),function(DayPilot){"use strict";function e(t,i,a){var n=i.indexOf(".");if(n===-1)return void("__proto__"!==i&&"constructor"!==i&&(t[i]=a));var o=i.substring(0,n);if("__proto__"!==o&&"constructor"!==o){var r=i.substring(n+1),s=t[o];"object"==typeof s&&null!==s||(t[o]={},s=t[o]),e(s,r,a)}}function t(e,i,a){i=i||{},a=a||"";for(var n in e){var o=e[n];"object"==typeof o?"[object Array]"===Object.prototype.toString.call(o)?i[a+n]=o:o&&o.toJSON?i[a+n]=o.toJSON():t(o,i,a+n+"."):i[a+n]=o}return i}if(!DayPilot.ModalStatic){DayPilot.ModalStatic={},DayPilot.ModalStatic.list=[],DayPilot.ModalStatic.hide=function(){if(this.list.length>0){var e=this.list.pop();e&&e.hide()}},DayPilot.ModalStatic.remove=function(e){for(var t=DayPilot.ModalStatic.list,i=0;i0&&(t[t.length-1].result=e)},DayPilot.ModalStatic.displayed=function(e){for(var t=DayPilot.ModalStatic.list,i=0;i0){var e=this.list[this.list.length-1];e&&e.stretch()}},DayPilot.ModalStatic.last=function(){var e=DayPilot.ModalStatic.list;return e.length>0?e[e.length-1]:null};var i=function(){if("undefined"==typeof window){var e={};return e.add=function(){},e.commit=function(){},e}var t=document.createElement("style");t.setAttribute("type","text/css"),t.styleSheet||t.appendChild(document.createTextNode("")),(document.head||document.getElementsByTagName("head")[0]).appendChild(t);var i=!!t.styleSheet,e={};return e.rules=[],e.commit=function(){try{i&&(t.styleSheet.cssText=this.rules.join("\n"))}catch(e){}},e.add=function(e,a,n){if(i)return void this.rules.push(e+"{"+a+"}");if(t.sheet.insertRule)"undefined"==typeof n&&(n=t.sheet.cssRules.length),t.sheet.insertRule(e+"{"+a+"}",n);else{if(!t.sheet.addRule)throw"No CSS registration method found";t.sheet.addRule(e,a,n)}},e},a="",n=new i;n.add(".modal_default_main","border: 10px solid #ccc; max-width: 90%;"),n.add(".modal_default_main:focus","outline: none;"),n.add(".modal_default_content","padding: 10px 0px;"),n.add(".modal_default_inner","padding: 20px;"),n.add(".modal_default_input","padding: 10px 0px;"),n.add(".modal_default_buttons","margin-top: 10px;"),n.add(".modal_default_buttons","padding: 10px 0px;"),n.add(".modal_default_form_item","padding: 10px 0px; position: relative;"),n.add(".modal_default_form_item_level1","border-left: 2px solid #ccc; margin-left: 10px; padding-left: 20px;"),n.add(".modal_default_form_item.modal_default_form_title","font-size: 1.5rem; font-weight: bold;"),n.add(".modal_default_form_item input[type=text]","width: 100%; box-sizing: border-box;"),n.add(".modal_default_form_item textarea","width: 100%; height: 200px; box-sizing: border-box;"),n.add(".modal_default_form_item input[type=select]","width: 100%; box-sizing: border-box;"),n.add(".modal_default_form_item label","display: block;"),n.add(".modal_default_form_item select","width: 100%; box-sizing: border-box;"),n.add(".modal_default_form_item_label","margin: 2px 0px;"),n.add(".modal_default_form_item_image img","max-width: 100%; height: auto;"),n.add(".modal_default_form_item_invalid",""),n.add(".modal_default_form_item_invalid_message","position: absolute; right: 0px; top: 9px; background-color: red; color: #ffffff; padding: 2px; border-radius: 2px;"),n.add(".modal_default_background","opacity: 0.5; background-color: #000;"),n.add(".modal_default_ok","padding: 3px; width: 80px;"),n.add(".modal_default_cancel","padding: 3px; width: 80px;"),n.add(".modal_default_form_item_date","position: relative;"),n.add(".modal_default_form_item_date:after","content: ''; position: absolute; right: 7px; top: 50%; margin-top: 3px; width: 10px; height: 15px; background-image:url("+""+")"),
+"undefined"!=typeof navigator&&navigator.userAgent.indexOf("Edge")!==-1&&n.add(".modal_default_form_item_date input::-ms-clear","display: none;"),n.add(".modal_default_form_item_scrollable_scroll","width: 100%; height: 200px; box-sizing: border-box; border: 1px solid #ccc; overflow-y: auto;"),n.add(".modal_default_form_item_scrollable_scroll_content","padding: 5px;"),n.add(".modal_default_form_item_searchable","position: relative;"),n.add(".modal_default_form_item_searchable_icon",""),n.add(".modal_default_form_item_searchable_icon:after","content:''; position: absolute; right: 5px; top: 50%; margin-top: -8px; width: 10px; height: 15px; background-image:url("+a+");"),n.add(".modal_default_form_item_searchable_list","box-sizing: border-box; border: 1px solid #999; max-height: 150px; overflow-y: auto;"),n.add(".modal_default_form_item_searchable_list_item","background: white; padding: 2px; cursor: default;"),n.add(".modal_default_form_item_searchable_list_item_highlight","background: #ccc;"),n.add(".modal_default_form_item_time","position: relative;"),n.add(".modal_default_form_item_time_icon",""),n.add(".modal_default_form_item_time_icon:after","content:''; position: absolute; right: 5px; top: 50%; margin-top: -8px; width: 10px; height: 15px; background-image:url("+a+");"),n.add(".modal_default_form_item_time_list","box-sizing: border-box; border: 1px solid #999; max-height: 150px; overflow-y: auto;"),n.add(".modal_default_form_item_time_list_item","background: white; padding: 2px; cursor: default;"),n.add(".modal_default_form_item_time_list_item_highlight","background: #ccc;"),n.add(".modal_default_form_item_datetime_parent","display: flex;"),n.add(".modal_default_form_item_datetime .modal_default_form_item_time_main","margin-left: 5px;"),n.add(".modal_default_form_item_datetime input[type='text'].modal_default_input_date ",""),n.add(".modal_default_form_item_tabular_main","margin-top: 10px;"),n.add(".modal_default_form_item_tabular_table","display: table; width: 100%; xbackground-color: #fff; border-collapse: collapse;"),n.add(".modal_default_form_item_tabular_tbody","display: table-row-group;"),n.add(".modal_default_form_item_tabular_row","display: table-row;"),n.add(".modal_default_form_item_tabular_row.modal_default_form_item_tabular_header",""),n.add(".modal_default_form_item_tabular_cell.modal_default_form_item_tabular_rowaction","padding: 0px; width: 23px;"),n.add(".modal_default_form_item_tabular_cell","display: table-cell; border: 0px; padding: 2px 2px 2px 0px; cursor: default; vertical-align: bottom;"),n.add(".modal_default_form_item_tabular_header .modal_default_form_item_tabular_cell","padding-left: 0px; padding-bottom: 0px;"),n.add(".modal_default_form_item_tabular_table input[type=text], .modal_default_form_item_tabular_table input[type=number]","width:100%; box-sizing: border-box;"),n.add(".modal_default_form_item_tabular_table select","width:100%; height:100%; box-sizing: border-box;"),n.add(".modal_default_form_item_tabular_plus","display: inline-block; background-color: #ccc; color: white; width: 20px; height: 20px; border-radius: 10px; box-sizing: border-box; position: relative; margin-left: 3px; margin-top: 3px; cursor: pointer;"),n.add(".modal_default_form_item_tabular_plus:after","content: ''; position: absolute; left: 5px; top: 5px; width: 10px; height: 10px; background-image: url(\"\")"),n.add(".modal_default_form_item_tabular_delete","display: inline-block; background-color: #ccc; color: white; width: 20px; height: 20px; border-radius: 10px; box-sizing: border-box; position: relative; margin-left: 3px; margin-top: 3px; cursor: pointer;"),n.add(".modal_default_form_item_tabular_delete:after","content: ''; position: absolute; left: 5px; top: 5px; width: 10px; height: 10px; background-image: url(\"\")"),n.add(".modal_default_form_item_tabular_disabled .modal_default_form_item_tabular_plus","display: none;"),n.add(".modal_default_form_item_tabular_plus_max.modal_default_form_item_tabular_plus","display: none;"),n.add(".modal_default_form_item_tabular_disabled .modal_default_form_item_tabular_delete","visibility: hidden;"),n.add(".modal_default_form_item_tabular_empty","height: 1px; margin: 5px 23px 5px 0px; background-color: #ccc;"),n.add(".modal_default_form_item_tabular_spacer .modal_default_form_item_tabular_cell","padding: 0px;"),n.add(".modal_min_main","border: 1px solid #ccc; max-width: 90%;"),n.add(".modal_min_background","opacity: 0.5; background-color: #000;"),n.add(".modal_min_ok","padding: 3px 10px;"),n.add(".modal_min_cancel","padding: 3px 10px;"),n.add(".navigator_modal_main","border-left: 1px solid #c0c0c0;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;background-color: white;color: #000000; box-sizing: content-box;"),n.add(".navigator_modal_main *, .navigator_modal_main *:before, .navigator_modal_main *:after","box-sizing: content-box;"),n.add(".navigator_modal_month","font-size: 11px;"),n.add(".navigator_modal_day","color: black;"),n.add(".navigator_modal_weekend","background-color: #f0f0f0;"),n.add(".navigator_modal_dayheader","color: black;"),n.add(".navigator_modal_line","border-bottom: 1px solid #c0c0c0;"),n.add(".navigator_modal_dayother","color: gray;"),n.add(".navigator_modal_todaybox","border: 1px solid red;"),n.add(".navigator_modal_title, .navigator_modal_titleleft, .navigator_modal_titleright","border-top: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;color: #333;background: #f3f3f3;"),n.add(".navigator_modal_busy","font-weight: bold;"),n.add(".navigator_modal_cell","text-align: center;"),n.add(".navigator_modal_select .navigator_modal_cell_box","background-color: #FFE794; opacity: 0.5;"),n.add(".navigator_modal_title","text-align: center;"),n.add(".navigator_modal_titleleft, .navigator_modal_titleright","text-align: center;"),n.add(".navigator_modal_dayheader","text-align: center;"),n.add(".navigator_modal_weeknumber","text-align: center;"),n.add(".navigator_modal_cell_text","cursor: pointer;"),n.commit(),DayPilot.Modal=function(e){this.autoFocus=!0,this.focus=null,this.autoStretch=!0,this.autoStretchFirstLoadOnly=!1,this.className=null,this.theme="modal_default",this.disposeOnClose=!0,this.dragDrop=!0,this.loadingHtml=null,this.maxHeight=null,this.scrollWithPage=!0,this.useIframe=!0,this.zIndex=99999,this.left=null,this.width=600,this.top=20,this.height=200,this.locale=null,this.closed=null,this.onClose=null,this.onClosed=null,this.onShow=null;var t=this;this.id="_"+(new Date).getTime()+"n"+10*Math.random(),this.Eb=!1,this.Fb=null,this.Gb=null,this.showHtml=function(e){if(DayPilot.ModalStatic.displayed(this))throw"This modal dialog is already displayed.";if(this.div||this.Hb(),this.$a(),this.useIframe){var t=function(e,t){return function(){e.setInnerHTML(e.id+"iframe",t)}};window.setTimeout(t(this,e),0)}else e.nodeType?this.div.appendChild(e):this.div.innerHTML=e;this.$a(),this.Ib(),this.Jb()},this.showUrl=function(e){if(DayPilot.ModalStatic.displayed(this))throw"This modal dialog is already displayed.";if(this.useIframe){this.div||this.Hb();var i=this.loadingHtml;i&&(this.iframe.src="about:blank",this.setInnerHTML(this.id+"iframe",i)),this.re(this.iframe,"load",this.Kb),this.iframe.src=e,this.$a(),this.Ib(),this.Jb()}else t.Lb({"url":e,"success":function(e){var i=e.request.responseText;t.showHtml(i)},"error":function(e){t.showHtml("Error loading the modal dialog")}})},this.Jb=function(){if("function"==typeof t.onShow){var e={};e.root=t.Mb(),e.modal=t,t.onShow(e)}},this.Mb=function(){return t.iframe?t.iframe.contentWindow.document:t.div},this.Lb=function(e){var t=new XMLHttpRequest;if(t){var i=e.method||"GET",a=e.success||function(){},n=e.error||function(){},o=e.data,r=e.url;t.open(i,r,!0),t.setRequestHeader("Content-type","text/plain"),t.onreadystatechange=function(){if(4===t.readyState)if(200===t.status||304===t.status){var e={};e.request=t,a(e)}else if(n){var e={};e.request=t,n(e)}else window.console&&console.log("HTTP error "+t.status)},4!==t.readyState&&("object"==typeof o&&(o=JSON.stringify(o)),t.send(o))}},this.$a=function(){delete this.result;var e=window,i=document,a=e.pageYOffset?e.pageYOffset:i.documentElement&&i.documentElement.scrollTop?i.documentElement.scrollTop:i.body.scrollTop;this.theme&&(this.hideDiv.className=this.theme+"_background"),this.zIndex&&(this.hideDiv.style.zIndex=this.zIndex),this.hideDiv.style.display="",window.setTimeout(function(){t.hideDiv&&(t.hideDiv.onclick=function(){t.hide({"backgroundClick":!0})})},500),this.theme?this.div.className=this.theme+"_main":this.div.className="",this.className&&(this.div.className+=" "+this.className),this.left?this.div.style.left=this.left+"px":this.div.style.marginLeft="-"+Math.floor(this.width/2)+"px",this.div.style.position="absolute",this.div.style.boxSizing="content-box",this.div.style.top=a+this.top+"px",this.div.style.width=this.width+"px",this.zIndex&&(this.div.style.zIndex=this.zIndex),this.height&&(this.useIframe||!this.autoStretch?this.div.style.height=this.height+"px":this.div.style.height=""),this.useIframe&&this.height&&(this.iframe.style.height=this.height+"px"),this.div.style.display="",this.Ob(),DayPilot.ModalStatic.remove(this),DayPilot.ModalStatic.list.push(this)},this.Kb=function(){t.iframe.contentWindow.modal=t,t.autoStretch&&t.stretch()},this.stretch=function(){var e=function(){return t.Nb().y},i=function(){return t.Nb().x};if(this.useIframe){for(var a=i()-40,n=this.width;nt.clientWidth},this.Qb=function(){for(var e=this.iframe.contentWindow.document,t="BackCompat"===e.compatMode?e.body:e.documentElement,i=t.scrollHeight,a=e.body.children,n=0;nt.clientHeight},this.Nb=function(){var e=document;if("CSS1Compat"===e.compatMode&&e.documentElement&&e.documentElement.clientWidth){var t=e.documentElement.clientWidth,i=e.documentElement.clientHeight;return{x:t,y:i}}var t=e.body.clientWidth,i=e.body.clientHeight;return{x:t,y:i}},this.Ib=function(){this.Eb||(this.re(window,"resize",this.Rb),this.re(window,"scroll",this.Sb),this.dragDrop&&(this.re(document,"mousemove",this.Tb),this.re(document,"mouseup",this.Ub)),this.Eb=!0)},this.Vb=function(){this.ue(window,"resize",this.Rb),this.ue(window,"scroll",this.Sb),this.dragDrop&&(this.ue(document,"mousemove",this.Tb),this.ue(document,"mouseup",this.Ub)),this.Eb=!1},this.Wb=function(e){e.target===t.div&&(e.preventDefault(),t.div.style.cursor="move",t.Xb(),t.Gb=t.mc(e||window.event),t.Fb={x:t.div.offsetLeft,y:t.div.offsetTop})},this.Tb=function(e){if(t.Gb){var e=e||window.event,i=t.mc(e),a=i.x-t.Gb.x,n=i.y-t.Gb.y;t.div.style.marginLeft="0px",t.div.style.top=t.Fb.y+n+"px",t.div.style.left=t.Fb.x+a+"px"}},this.Ub=function(e){t.Gb&&(t.Yb(),t.div.style.cursor=null,t.Gb=null)},this.Xb=function(){if(this.useIframe){var e=document.createElement("div");e.style.backgroundColor="#ffffff",e.style.filter="alpha(opacity=80)",e.style.opacity="0.80",e.style.width="100%",e.style.height=this.height+"px",e.style.position="absolute",e.style.left="0px",e.style.top="0px",this.div.appendChild(e),this.mask=e}},this.Yb=function(){this.useIframe&&(this.div.removeChild(this.mask),this.mask=null)},this.Rb=function(){t.Zb(),t.Ob()},this.Sb=function(){t.Zb()},this.Ob=function(){if(!t.left&&t.div){var e=t.div.offsetWidth;t.div.style.marginLeft="-"+Math.floor(e/2)+"px"}},this.Zb=function(){if(t.hideDiv&&t.div&&"none"!==t.hideDiv.style.display&&"none"!==t.div.style.display){var e=t.$b.scrollY();t.scrollWithPage||(t.div.style.top=e+t.top+"px")}},this.$b={},this.$b.container=function(){return t.container||document.body},this.$b.scrollY=function(){var e=t.$b.container();return e===document.body?window.pageYOffset?window.pageYOffset:document.documentElement&&document.documentElement.scrollTop?document.documentElement.scrollTop:document.body.scrollTop:e.scrollTop},this.re=function(e,t,i){e.addEventListener?e.addEventListener(t,i,!1):e.attachEvent&&e.attachEvent("on"+t,i)},this.ue=function(e,t,i){e.removeEventListener?e.removeEventListener(t,i,!1):e.detachEvent&&e.detachEvent("on"+t,i)},this.mc=function(e){return e.pageX||e.pageY?{x:e.pageX,y:e.pageY}:{x:e.clientX+document.documentElement.scrollLeft,y:e.clientY+document.documentElement.scrollTop}},this.abs=function(e){for(var t={x:e.offsetLeft,y:e.offsetTop};e.offsetParent;)e=e.offsetParent,t.x+=e.offsetLeft,t.y+=e.offsetTop;return t},this.Hb=function(){var e=t.$b.container(),i=e===document.body,a=i?"fixed":"absolute",n=document.createElement("div");n.id=this.id+"hide",n.style.position=a,n.style.left="0px",n.style.top="0px",n.style.right="0px",n.style.bottom="0px",n.oncontextmenu=function(){return!1},n.onmousedown=function(){return!1},e.appendChild(n);var o=document.createElement("div");o.id=this.id+"popup",o.style.position=a,o.style.left="50%",o.style.top="0px",o.style.backgroundColor="white",o.style.width="50px",o.style.height="50px",this.dragDrop&&(o.onmousedown=this.Wb),o.addEventListener("keydown",function(e){e.stopPropagation()});var r=null;this.useIframe&&(r=document.createElement("iframe"),r.id=this.id+"iframe",r.name=this.id+"iframe",r.frameBorder="0",r.style.width="100%",r.style.height="50px",o.appendChild(r)),e.appendChild(o),this.div=o,this.iframe=r,this.hideDiv=n},this.setInnerHTML=function(e,i){var a=window.frames[e],n=a.contentWindow||a.document||a.contentDocument;n.document&&(n=n.document),null==n.body&&n.write(""),i.nodeType?n.body.appendChild(i):n.body.innerHTML=i,t.autoStretch&&(t.autoStretchFirstLoadOnly&&t._b||(t.stretch(),t._b=!0))},this.close=function(e){this.result=e,this.hide()},this.closeSerialized=function(){for(var e=t.Mb(),i=e.querySelectorAll("input, textarea, select"),a={},n=0;n0?DayPilot.ModalStatic.list[DayPilot.ModalStatic.list.length-1]:parent&&parent.DayPilot&&parent.DayPilot.ModalStatic&&parent.DayPilot.ModalStatic.list[parent.DayPilot.ModalStatic.list.length-1]},DayPilot.Modal.Experimental={},DayPilot.Modal.Experimental.Form=r,"undefined"==typeof DayPilot.getPromise&&(DayPilot.getPromise=function(e){return"undefined"!=typeof Promise?new Promise(e):(DayPilot.Promise=function(e){var t=this;this.then=function(t,i){return t=t||function(){},i=i||function(){},e(t,i),DayPilot.getPromise(e)},this["catch"]=function(i){return t.then(null,i),DayPilot.getPromise(e)}},new DayPilot.Promise(e))});var r=function(e){this.form=[],this.data={},this.theme="form_default",this.zIndex=99999,this.locale="en-us",this.plugins={},this.onKey=null,this.dc=[],this.ec=null,this.canceling=!1,this.fc=[],this.f=[],this.gc=null,e=e||{};for(var t in e)this[t]=e[t]};r.prototype.create=function(){return this.load(),this.render(),this.gc},r.prototype.render=function(){var e=this;this.gc=document.createElement("div"),this.dc.forEach(function(t){e.createView(t)}),this.applyState()},r.prototype.createView=function(e){var t=this.theme,i=this,a=document.createElement("div");if(a.className=t+"_form_item "+t+"_form_item_level"+e.level,e.interactive||"title"!==e.type?a.className+=" "+t+"_form_item_"+e.type:a.className+=" "+t+"_form_title",e.data.cssClass&&(a.className+=" "+e.data.cssClass),!e.isValue){var n=document.createElement("div");n.className=t+"_form_item_label",n.innerText=e.text,a.appendChild(n)}var o=this.createInteractive(e);o.onInput=function(e){if(e=e||{},i.hc(o,{"debounce":!e.immediate}),"function"==typeof i.onChange){var t={};t.result=i.serialize(),i.onChange(t)}},o.onBlur=function(){i.canceling||i.hc(o)},o.apply(e),o.gc=a,o.row=e,o.element&&a.appendChild(o.element),this.f.push(o),this.gc.appendChild(a)},r.prototype.validate=function(){var e=this,t=!0;return this.f.forEach(function(i){var a=e.hc(i);t=t&&a}),t},r.prototype.hc=function(e,t){function i(){e.ic&&(e.ic.remove(),e.ic=null),e.gc.classList.add(u);var t=document.createElement("div");t.classList.add(h),t.innerText=c.message,e.ic=t,e.gc.appendChild(t)}t=t||{};var a=t.debounce,n=t.silent,o=e.row,r=!0,s="function"==typeof o.data.onValidate?o.data.onValidate:null,l="function"==typeof o.data.validate?o.data.validate:null,d=s||l;if(d){var c={};c.valid=!0,c.value=e.save()[o.field],c.message="Error",c.values=this.serialize(),c.result=this.serialize(),d(c);var u=this.theme+"_form_item_invalid",h=this.theme+"_form_item_invalid_message";if(c.valid)clearTimeout(this.fc[o.field]),e.ic&&(e.ic.remove(),e.ic=null),e.gc.classList.remove(u);else if(!n)if(a){var f=1e3;clearTimeout(this.fc[o.field]),this.fc[o.field]=setTimeout(function(){i()},f)}else i();r=c.valid}return r},r.prototype.load=function(){var e=this;this.form.forEach(function(t){e.processFormItem(t,0)});var i;try{var a=JSON.stringify(this.data);i=t(JSON.parse(a))}catch(e){throw new Error("The 'data' object is not serializable (it may contain circular dependencies): "+e)}for(var n in i)this.setValue(n,i[n])},r.prototype.setValue=function(e,t){this.dc.forEach(function(i){i.applyValue(e,t)})},r.prototype.updateDependentState=function(){var e=this,t=[!0];(this.ec?this.ec:this.dc).forEach(function(i){var a=e.updateState(i,{enabled:t[i.level]&&!i.data.disabled});a.isValue&&(t[a.level+1]=a.enabled&&a.checked)})},r.prototype.processFormItem=function(e,t){var i=this,a=this.getFieldType(e),n=[];if("radio"===a){if(e.name){var o=new s;o.field=e.id,o.data=e,o.level=t,o.type="label",o.interactive=!1,o.text=e.name,i.dc.push(o),n.push(o)}e.options.forEach(function(o){var r=new s;r.field=e.id,r.data=o,r.level=t,r.type=a,r.isValue=!0,r.text=o.name,r.resolved=o.id,i.dc.push(r),n.push(r),o.children&&o.children.forEach(function(e){var a=i.processFormItem(e,t+1);n=n.concat(a)})})}else if("title"===a){var o=new s;o.field=e.id,o.data=e,o.level=t,o.type=a,o.interactive=!1,o.text=e.name,i.dc.push(o),n.push(o)}else if("image"===a){var o=new s;o.isValue=!0,o.field=e.id,o.data=e,o.level=t,o.type=a,o.interactive=!1,o.text=null,i.dc.push(o),n.push(o)}else if("html"===a){var o=new s;o.isValue=!0,o.field=e.id,o.data=e,o.level=t,o.type=a,o.interactive=!1,o.text=null,i.dc.push(o),n.push(o)}else if("scrollable"===a){var o=new s;o.isValue=!0,o.field=e.id,o.data=e,o.level=t,o.type=a,o.interactive=!1,o.text=null,i.dc.push(o),n.push(o)}else{var o=new s;o.field=e.id,o.data=e,o.level=t,o.type=a,o.text=e.name,o.children=[],i.dc.push(o),n.push(o)}return"checkbox"===a&&(o.isValue=!0,o.resolved=!0,e.children&&e.children.forEach(function(e){var a=i.processFormItem(e,t+1);n=n.concat(a)})),n},r.prototype.doOnKey=function(e){if("function"==typeof this.onKey){var t={key:e};this.onKey(t)}},r.prototype.createInteractive=function(e){var t=this,i={"label":function(){return new l},"title":function(){return new l},"image":function(){var t=new l,i=document.createElement("img");return i.src=e.data.image,t.element=i,t},"html":function(){var t=new l,i=document.createElement("div");return"string"==typeof e.data.text?i.innerText=e.data.text:"string"==typeof e.data.html&&(i.innerHTML=e.data.html),t.element=i,t},"scrollable":function(){var i=new l,a=document.createElement("div");a.className=t.theme+"_form_item_scrollable_scroll",e.data.height&&(a.style.height=e.data.height+"px");var n=document.createElement("div");return n.className=t.theme+"_form_item_scrollable_scroll_content","string"==typeof e.data.text?n.innerText=e.data.text:"string"==typeof e.data.html&&(n.innerHTML=e.data.html),a.appendChild(n),i.element=a,i},"text":function(){var i=new l;i.apply=function(e){i.row=e;var t=i.element;t.value=e.value,t.disabled=!e.enabled};var a=document.createElement("input");return a.name=e.field,a.type="text",a.autocomplete="off",a.onkeydown=function(e){var i=!1;switch(e.keyCode){case 13:t.doOnKey("Enter");break;case 27:t.doOnKey("Escape");break;default:i=!0}i||(e.preventDefault(),e.stopPropagation())},a.oninput=function(e){i.onInput()},a.onblur=function(e){i.onBlur()},i.element=a,i.canFocus=function(){return!i.element.disabled},i.focus=function(){i.element.focus(),i.element.setSelectionRange(0,i.element.value.length)},i.save=function(){var t={};return t[e.field]=a.value,t},i},"textarea":function(){var i=new l;i.apply=function(e){i.row=e;var t=i.element;t.value=e.value,t.disabled=!e.enabled};var a=document.createElement("textarea");return a.name=e.field,e.data.height&&(a.style.height=e.data.height+"px"),a.onkeydown=function(e){var i=!1;switch(e.keyCode){case 13:(e.ctrlKey||e.metaKey)&&t.doOnKey("Enter"),i=!1;break;case 27:t.doOnKey("Escape");break;default:i=!0}i||e.stopPropagation()},a.oninput=function(e){i.onInput()},a.onblur=function(e){i.onBlur()},i.element=a,i.canFocus=function(){return!i.element.disabled},i.focus=function(){i.element.focus(),i.element.setSelectionRange(0,0)},i.save=function(){var t={};return t[e.field]=a.value,t},i},"date":function(){var i=new l;i.apply=function(e){i.row=e;var a=i.element,n=i.picker;e.data.dateFormat&&(n.pattern=e.data.dateFormat);var o=e.data.locale||t.locale;o&&(n.locale=o),a.disabled=!e.enabled,n.date=new DayPilot.Date(e.value);var r=new DayPilot.Date(e.value).toString(e.data.dateFormat||n.pattern,n.locale);a.value=r};var a=document.createElement("input");a.name=e.field;var n=new DayPilot.DatePicker({target:a,theme:"navigator_modal",zIndex:t.zIndex+1,resetTarget:!1,targetAlignment:"left",onTimeRangeSelect:function(e){i.onInput({"immediate":!0})}});return a.picker=n,a.className=t.theme+"_input_date",a.type="text",a.onkeydown=function(e){var i=!1;switch(e.keyCode){case 13:n.visible?n.close():t.doOnKey("Enter");break;case 27:n.visible?n.close():t.doOnKey("Escape");break;case 9:n.close(),i=!0;break;default:i=!0}i||(e.preventDefault(),e.stopPropagation())},a.onfocus=function(){n.show()},a.onclick=function(){n.show()},a.oninput=function(e){i.onInput()},a.onblur=function(e){i.onBlur()},i.element=a,i.picker=n,i.canFocus=function(){return!i.element.disabled},i.focus=function(){i.element.focus()},i.save=function(){var t=n.date?n.date.toString():null,i={};return i[e.field]=t,i},i},"time":function(){return t.jc(e)},"datetime":function(){return t.kc(e)},"select":function(){var t=new l;t.apply=function(e){t.row=e;var i=t.element;i.value=e.value,i.disabled=!e.enabled};var i=document.createElement("select");return i.name=e.field,e.data.options&&e.data.options.forEach&&e.data.options.forEach(function(e){var t=document.createElement("option");t.innerText=e.name||e.id,t.value=e.id,t.lc=e.id,i.appendChild(t)}),i.onchange=function(e){t.onInput({"immediate":!0})},i.onblur=function(e){t.onBlur()},t.element=i,t.canFocus=function(){return!t.element.disabled},t.focus=function(){t.element.focus()},t.save=function(){var t=null,a=i.options[i.selectedIndex];a&&"undefined"!=typeof a.lc&&(t=a.lc);var n={};return n[e.field]=t,n},t},"searchable":function(){var i=new l;i.apply=function(e){i.row=e;var t=i.searchable;t.disabled=!e.enabled,t.select(e.value)};var a=new d({data:e.data.options||[],name:e.field,theme:t.theme+"_form_item_searchable",listZIndex:t.zIndex+1,onSelect:function(e){e.ui&&i.onInput({"immediate":!0})}}),n=a.create();return i.element=n,i.searchable=a,i.canFocus=function(){return!i.searchable.disabled},i.focus=function(){i.searchable.focus()},i.save=function(){var t=a.selected&&a.selected.id,i={};return i[e.field]=t,i},i},"radio":function(){var i=new l;i.apply=function(e){i.row=e;var t=i.radio;t.checked=e.checked,t.disabled=!e.enabled};var a=document.createElement("label"),n=document.createElement("input");n.type="radio",n.name=e.field,n.lc=e.resolved,n.onchange=function(e){var a=i.row;t.findRowsByField(a.field).forEach(function(e){t.updateState(e,{checked:!1})}),t.updateState(a,{checked:!0}),t.applyState(),i.onInput({"immediate":!0})},n.onblur=function(e){i.onBlur()},a.appendChild(n);var o=document.createTextNode(e.text);return a.append(o),i.element=a,i.radio=n,i.canFocus=function(){return!1},i.focus=function(){i.radio.focus()},i.save=function(){if(!n.checked)return{};var t=n.lc,i={};return i[e.field]=t,i},i},"checkbox":function(){var i=new l;i.apply=function(e){i.row=e;var t=i.checkbox;t.checked=e.checked,t.disabled=!e.enabled};var a=document.createElement("label"),n=document.createElement("input");n.type="checkbox",n.name=e.field,n.lc=e.resolved,n.onchange=function(e){var a=i.row;t.updateState(a,{checked:this.checked}),t.applyState(),i.onInput({"immediate":!0})},n.onblur=function(e){i.onBlur()},a.appendChild(n);var o=document.createTextNode(e.text);return a.append(o),i.element=a,i.checkbox=n,i.canFocus=function(){return!1},i.focus=function(){i.checkbox.focus()},i.save=function(){var t=n.checked,i={};return i[e.field]=t,i},i},"table":function(){var i=new l;i.apply=function(e){i.row=e;var t=i.table;t.disabled=!e.enabled,t.load(e.value||[])};var a=new c({name:e.field,form:t,theme:t.theme+"_form_item_tabular",item:e.data,onInput:function(e){i.onInput()}}),n=a.create();return i.element=n,i.table=a,i.canFocus=function(){return!1},i.focus=function(){i.table.focus()},i.save=function(){var t=a.save(),i={};return i[e.field]=t,i},i}};return t.plugins&&t.plugins[e.type]?t.plugins[e.type](e):i[e.type]();
+},r.prototype.jc=function(e){var t=this,i=new l;i.apply=function(e){i.row=e;var t=i.searchable;t.disabled=!e.enabled,t.select(e.value)};var a=[],n=e.data.timeInterval||15;[1,5,10,15,20,30,60].includes(n)||(n=15);for(var o=60/n,r=e.data.locale||t.locale,s=DayPilot.Locale.find(r)||DayPilot.Locale.US,c=DayPilot.Date.today(),u=0;u<24*o;u++){var h=c.addMinutes(n*u),f={};f.name=h.toString(e.data.timeFormat||s.timePattern,s),f.id=h.toString("HH:mm"),a.push(f)}var m=new d({data:a,name:e.field,theme:t.theme+"_form_item_time",listZIndex:t.zIndex+1,strategy:"startsWith",onSelect:function(e){e.ui&&i.onInput({"immediate":!0})}}),p=m.create();return i.element=p,i.searchable=m,i.canFocus=function(){return!i.searchable.disabled},i.focus=function(){i.searchable.focus()},i.save=function(){var t=m.selected&&m.selected.id,i={};return i[e.field]=t,i},i},r.prototype.kc=function(e){var t=this,i=new l;i.apply=function(e){i.row=e;var a=i.searchable;a.disabled=!e.enabled;var n=new DayPilot.Date(e.value).toString("HH:mm");a.select(n);var o=i.dateInput,r=i.picker;e.data.dateFormat&&(r.pattern=e.data.dateFormat);var s=e.data.locale||t.locale;if(s){var l=DayPilot.Locale.find(s)||DayPilot.Locale.US;r.locale=s,r.pattern=l.datePattern}o.disabled=!e.enabled,r.date=new DayPilot.Date(e.value);var d=new DayPilot.Date(e.value).toString(e.data.dateFormat||r.pattern,r.locale);o.value=d};var a=function(){var a=document.createElement("input");a.name=e.field;var n=new DayPilot.DatePicker({target:a,theme:"navigator_modal",zIndex:t.zIndex+1,resetTarget:!1,targetAlignment:"left",onTimeRangeSelect:function(e){i.onInput({"immediate":!0})}});return a.picker=n,a.className=t.theme+"_input_date",a.type="text",a.onkeydown=function(e){var i=!1;switch(e.keyCode){case 13:n.visible?n.close():t.doOnKey("Enter");break;case 27:n.visible?n.close():t.doOnKey("Escape");break;case 9:n.close(),i=!0;break;default:i=!0}i||(e.preventDefault(),e.stopPropagation())},a.onfocus=function(){n.show()},a.onclick=function(){n.show()},a.oninput=function(e){i.onInput()},a.onblur=function(e){i.onBlur()},i.dateInput=a,i.picker=n,a}(),n=function(){var a=[],n=e.data.timeInterval||15;[1,5,10,15,20,30,60].includes(n)||(n=15);for(var o=60/n,r=e.data.locale||t.locale,s=DayPilot.Locale.find(r)||DayPilot.Locale.US,l=DayPilot.Date.today(),c=0;c<24*o;c++){var u=l.addMinutes(n*c),h={};h.name=u.toString(e.data.timeFormat||s.timePattern,s),h.id=u.toString("HH:mm"),a.push(h)}var f=new d({data:a,name:e.field,theme:t.theme+"_form_item_time",listZIndex:t.zIndex+1,strategy:"startsWith",onSelect:function(e){e.ui&&i.onInput({"immediate":!0})}});return i.searchable=f,f.create()}(),o=document.createElement("div");return o.className=t.theme+"_form_item_datetime_parent",o.appendChild(a),o.appendChild(n),i.element=o,i.canFocus=function(){return!i.searchable.disabled},i.focus=function(){i.dateInput.focus()},i.save=function(){var t=i.searchable.selected&&i.searchable.selected.id,a=i.picker.date?i.picker.date.toString():null,n=new DayPilot.Date(a).getDatePart(),o=DayPilot.Date.parse(n.toString("yyyy-dd-MM ")+t,"yyyy-dd-MM HH:mm"),r={};return r[e.field]=o,r},i},r.prototype.findRowsByField=function(e){return this.dc.filter(function(t){return t.field===e})},r.prototype.findViewById=function(e,t){return this.f.find(function(i){return i.row.field===e&&("radio"!==i.row.type||i.row.resolved===t)})},r.prototype.firstFocusable=function(){return this.f.find(function(e){return e.canFocus&&e.canFocus()})},r.prototype.updateState=function(e,t){var i=this.ec?this.ec:this.dc,a=i.indexOf(e);return this.ec=i.map(function(i){if(i!==e)return i;if(e.propsEqual(t))return e;var a=e.clone();for(var n in t)a[n]=t[n];return a}),this.ec[a]},r.prototype.updateInteractive=function(e){var t=this.ec.indexOf(e);this.f[t].apply(e)},r.prototype.applyState=function(){var e=this;if(this.updateDependentState(),this.ec){this.ec.filter(function(t,i){return e.dc[i]!==t}).forEach(function(t){e.updateInteractive(t)}),this.dc=this.ec,this.ec=null}},r.prototype.getFieldType=function(e){return["text","date","select","searchable","radio","checkbox","table","title","image","html","textarea","scrollable","time","datetime"].indexOf(e.type)!==-1?e.type:e.type&&this.plugins&&this.plugins[e.type]?e.type:e.image?"image":e.html||e.text?"html":e.id?e.options?"searchable":e.dateFormat?"date":e.columns?"table":"text":"title"},r.prototype.serialize=function(){var e={};return this.f.forEach(function(t){var i=t.save();for(var a in i)e[a]=i[a]}),e};var s=function(){this.id=this.guid(),this.field=null,this.data=null,this.type=null,this.level=0,this.enabled=!0,this.value=null,this.text=null,this.interactive=!0,this.isValue=!1,this.checked=!1,this.resolved=null};s.prototype.clone=function(){var e=new s;for(var t in this)"id"!==t&&(e[t]=this[t]);return e},s.prototype.propsEqual=function(e){for(var t in e)if(this[t]!==e[t])return!1;return!0},s.prototype.guid=function(){var e=function(){return(65536*(1+Math.random())|0).toString(16).substring(1)};return""+e()+e()+"-"+e()+"-"+e()+"-"+e()+"-"+e()+e()+e()},s.prototype.applyValue=function(e,t){this.field===e&&(this.value=t,this.isValue&&t===this.resolved&&(this.checked=!0))};var l=function(){this.element=null,this.canFocus=function(){return!1},this.apply=function(e){},this.focus=function(){},this.save=function(){return{}}},d=function(e){this.data=[],this.name=null,this.theme="searchable_default",this.nc=!1,this.listZIndex=1e5,this.onSelect=null,this.oc=null,this.pc=null,this.qc=!1,this.rc=null,this.sc=null,this.o=[],this.tc=null,e=e||{};var t=this,i={"selected":{post:function(e){"object"==typeof e&&e.id?t.oc=e:"string"!=typeof e&&"number"!=typeof e||t.select(e)}}};Object.defineProperty(this,"selected",{get:function(){return this.oc}}),Object.defineProperty(this,"disabled",{get:function(){return this.nc},set:function(e){this.nc=e,this.rc&&(this.rc.disabled=e,e&&this.uc())}});for(var a in e)i[a]||(this[a]=e[a]);for(var a in e)i[a]&&i[a].post(e[a])};d.prototype.select=function(e){return this.oc=this.data.find(function(t){return t.id===e}),this.vc(!1),this},d.prototype.create=function(){function e(e){var i=l.strategy;"includes"!==l.strategy&&"startsWith"!==l.strategy&&(i="includes"),e=e||i||"includes",h.style.display="",h.style.top=m.offsetHeight+"px",h.style.left="0px",h.style.width=m.offsetWidth+"px",h.innerHTML="",h.addEventListener("mousedown",function(e){e.preventDefault()}),l.pc=null,l.o=[];var n=null;l.data.forEach(function(i){var o=i.name||i.id;if("includes"===e){if(o.toLowerCase().indexOf(m.value.toLowerCase())===-1)return}else if("startsWith"===e&&0!==o.toLowerCase().indexOf(m.value.toLowerCase()))return;var r=document.createElement("div");r.className=l.theme+"_list_item",r.innerText=o,r.item=i,i===l.oc&&(l.pc=r),n||(n=r),r.addEventListener("mousedown",function(e){a(r),e.preventDefault()}),r.addEventListener("mousemove",function(e){l.pc!==r&&(l.pc=r,t({dontScroll:!0}))}),h.appendChild(r),l.o.push(r)}),l.pc||(l.pc=n),t()}function t(e){e=e||{};var t=!e.dontScroll;document.querySelectorAll("."+l.theme+"_list_item_highlight").forEach(function(e){e.className=e.className.replace(l.theme+"_list_item_highlight","")}),l.pc&&(l.pc.className+=" "+l.theme+"_list_item_highlight",t&&!i(l.pc,h)&&l.pc.scrollIntoView())}function i(e,t){var i=e.getBoundingClientRect(),a=t.getBoundingClientRect();return i.top>=a.top&&i.bottom<=a.bottom}function a(e){var t=e.item;l.oc=t,l.vc(!0),o(),r()}function n(){l.uc()}function o(){l.p()}function r(){l.qc=!0,m.setAttribute("readonly","readonly"),m.focus()}function s(){l.qc=!1,m.removeAttribute("readonly"),m.select(),e("all")}var l=this,d=this,c=document.createElement("div");c.className=this.theme+"_main",c.style.position="relative";var u=document.createElement("div");u.className=this.theme+"_icon",u.style.position="absolute",u.style.right="0",u.style.top="0",u.style.bottom="0",u.style.width="20px",u.addEventListener("mousedown",function(e){e.preventDefault(),l.qc?(l.focus(),s()):(n(),r())});var h=document.createElement("div");h.className=this.theme+"_list",h.style.display="none",h.style.position="absolute",h.style.zIndex=this.listZIndex;var f=document.createElement("input");f.type="hidden",f.name=this.name,f.searchable=d,this.tc=f;var m=document.createElement("input");return m.type="text",m.className=this.theme+"_input",m.disabled=this.nc,m.addEventListener("click",function(e){s()}),m.addEventListener("focus",function(t){e("all")}),m.addEventListener("input",function(t){e()}),m.addEventListener("blur",function(e){m.removeAttribute("readonly"),n()}),m.addEventListener("keydown",function(e){if(l.qc){if("Enter"===e.key)return;if("Esc"===e.key||"Escape"===e.key)return;s()}if("ArrowDown"===e.key){var i=d.o.indexOf(d.pc);i+1=0&&(d.pc=d.o[i-1]),t()}else"Enter"===e.key?l.pc?(e.stopPropagation(),a(l.pc)):(e.stopPropagation(),n(),r()):"Esc"!==e.key&&"Escape"!==e.key||(e.stopPropagation(),n(),r())}),this.rc=m,this.sc=h,this.oc||(this.oc=this.data[0],this.oc&&(m.value=this.oc.name)),c.appendChild(m),c.appendChild(u),c.appendChild(f),c.appendChild(h),c},d.prototype.uc=function(){this.p(),this.oc?this.rc.value=this.oc.name:(this.rc.value="",this.vc(!0))},d.prototype.focus=function(){this.qc=!0,this.rc.setAttribute("readonly","readonly"),this.rc.focus(),this.uc()},d.prototype.p=function(){this.sc.style.display="none"},d.prototype.vc=function(e){if(this.tc.value=this.selected?this.selected.id:null,this.oc?this.rc.value=this.oc.name:this.rc.value="","function"==typeof this.onSelect){var t={control:this,ui:e};this.onSelect(t)}};var c=function(e){this.form=null,this.item=null,this.data=null,this.name=null,this.theme="edit_table_default",this.onInput=null,this.nav={},this.wc=null,this.dc=[],e=e||{};for(var t in e)this[t]=e[t]};c.prototype.create=function(){var e=this,t=document.createElement("div");t.className=this.theme+"_main",t.style.position="relative";var i=document.createElement("input");i.type="hidden",i.name=e.name,i.table=this,t.appendChild(i);var a=document.createElement("div");a.className=this.theme+"_table";var n=this.xc();a.appendChild(n);var o=e.yc({});o.spacer=!0;var r=this.zc(o);r.classList.add(e.theme+"_spacer"),a.appendChild(r);var s=document.createElement("div");s.className=e.theme+"_tbody",a.appendChild(s),t.appendChild(a);var l=document.createElement("div");t.appendChild(l),this.nav.body=s,this.nav.table=a,this.nav.main=t,this.nav.after=l;var d=document.createElement("div"),c=document.createElement("span");return c.className=this.theme+"_plus",c.addEventListener("click",function(t){if(!e.disabled){var i=e.item.onNewRow,a={};if("function"==typeof i){var n={};n.result=e.form.serialize(),n.value={},i(n),a=n.value}var o=e.yc(a);e.dc.push(o),e.Ac(),e.Bc()}}),this.nav.plus=c,d.appendChild(c),t.appendChild(d),t},c.prototype.xc=function(){var e=this,t=document.createElement("div");return t.classList.add(this.theme+"_row"),t.classList.add(this.theme+"_header"),this.item.columns.forEach(function(i){var a=document.createElement("div");a.classList.add(e.theme+"_cell"),a.innerText=i.name,t.appendChild(a)}),t},c.prototype.Cc=function(){var e=this.item.max||0;return!!(e&&this.dc.length>=e)},c.prototype.save=function(){var e=this,t=[];return e.dc.forEach(function(e){var i={};e.cells.forEach(function(e){i[e.id]=e.value}),t.push(i)}),t},c.prototype.load=function(e){if("[object Array]"!==Object.prototype.toString.call(e))throw new Error("Array expected");this.data=e,this.Dc(),this.Ac()},c.prototype.Ec=function(){this.disabled?this.nav.main.classList.add(this.theme+"_disabled"):this.nav.main.classList.remove(this.theme+"_disabled"),this.Cc()?this.nav.plus.classList.add(this.theme+"_plus_max"):this.nav.plus.classList.remove(this.theme+"_plus_max")},c.prototype.Dc=function(){var e=this;this.dc=[],this.data.forEach(function(t){var i=e.yc(t);e.dc.push(i)})},c.prototype.Fc=function(e){var t=this,i=t.dc.indexOf(e);t.dc.splice(i,1)},c.prototype.yc=function(e){var t=this,i={};return i.data=e,i.cells=[],t.item.columns.forEach(function(a){var n=a.id,o=e[n],r=t.Gc(a);if("undefined"==typeof o)if("text"===r)o="";else if("number"===r)o=0;else if("select"===r){var s=a.options;o=s&&s[0].id}var l={};l.id=n,l.value=o,l.type=r,l.data=a,i.cells.push(l)}),i},c.prototype.Gc=function(e){var t=e.type;return t||(t=e.options?"select":"text"),t},c.prototype.Ac=function(){var e=this;if(this.nav.body.innerHTML="",this.nav.after.innerHTML="",this.dc.forEach(function(t){var i=e.zc(t);e.nav.body.appendChild(i)}),0===this.dc.length){var t=e.Hc();e.nav.after.appendChild(t)}this.Ec()},c.prototype.Hc=function(){var e=document.createElement("div");return e.className=this.theme+"_empty",e},c.prototype.zc=function(e){var t=this,i=document.createElement("div");i.className=t.theme+"_row",e.cells.forEach(function(a){var n=document.createElement("div");n.className=t.theme+"_cell";var o=t.Ic(a);if(e.spacer){var r=document.createElement("div");r.style.height="0px",r.style.overflow="hidden",r.appendChild(o),n.appendChild(r)}else n.appendChild(o);i.appendChild(n)});var a=document.createElement("div");a.classList.add(t.theme+"_cell"),a.classList.add(t.theme+"_rowaction");var n=document.createElement("span");return n.className=this.theme+"_delete",n.addEventListener("click",function(i){t.disabled||(t.Fc(e),t.Ac(),t.Bc())}),e.spacer||a.appendChild(n),i.appendChild(a),i},c.prototype.Bc=function(){var e=this;if("function"==typeof e.onInput){var t={};e.onInput(t)}},c.prototype.Ic=function(e){var t=this,i=e.type;if("text"===i||"number"===i){var a=document.createElement("input");return a.type=i,t.disabled&&(a.disabled=!0),e.value&&(a.value=e.value),a.addEventListener("keyup",function(a){"number"===i?e.value=Number(this.value):e.value=this.value,t.Bc()}),a}if("select"===i){var n=document.createElement("select");return t.disabled&&(n.disabled=!0),e.data.options.forEach(function(t){var i=document.createElement("option");i.innerText=t.name,i.value=t.id,i.lc=t.id,n.appendChild(i),e.value===t.id&&i.setAttribute("selected",!0)}),n.addEventListener("change",function(i){var a=n.options[n.selectedIndex];a&&"undefined"!=typeof a.lc&&(e.value=a.lc),t.Bc()}),n}throw new Error("Unsupported item type: "+i)},c.prototype.focus=function(){}}}(DayPilot),"undefined"==typeof DayPilot)var DayPilot={};if("undefined"==typeof DayPilot.Global&&(DayPilot.Global={}),function(){if("undefined"==typeof DayPilot.Month||!DayPilot.Month.events){var e={};e.Month=function(t,i){this.v="2024.3.539-lite",this.nav={};var a=this;this.id=t,this.isMonth=!0,this.api=2,this.backendUrl=null,this.cellHeaderHeight=24,this.cellHeight=100,this.cellMarginBottom=0,this.contextMenu=null,this.cssClassPrefix="month_default",this.eventBarVisible=!0,this.eventHeight=25,this.eventsLoadMethod="GET",this.headerHeight=30,this.hideUntilInit=!0,this.lineSpace=1,this.locale="en-us",this.showToolTip=!0,this.startDate=new DayPilot.Date,this.theme=null,this.visible=!0,this.weekStarts="Auto",this.width="100%",this.xssProtection="Enabled",this.afterRender=function(){},this.cellHeaderClickHandling="Enabled",this.eventClickHandling="Enabled",this.eventDeleteHandling="Disabled",this.eventMoveHandling="Update",this.eventResizeHandling="Update",this.eventRightClickHandling="ContextMenu",this.headerClickHandling="Enabled",this.timeRangeSelectedHandling="Enabled",this.onCellHeaderClick=null,this.onCellHeaderClicked=null,this.onEventClick=null,this.onEventClicked=null,this.onEventDelete=null,this.onEventDeleted=null,this.onEventMove=null,this.onEventMoved=null,this.onEventResize=null,this.onEventResized=null,this.onEventRightClick=null,this.onEventRightClicked=null,this.onTimeRangeSelect=null,this.onTimeRangeSelected=null,this.onBeforeEventRender=null,this.onBeforeCellRender=null,this.cellEvents=[],this.elements={},this.elements.events=[],this.cache={},this.N=!1,this.T=function(e,t){var e=JSON.parse(e);return e.CallBackRedirect?void(document.location.href=e.CallBackRedirect):"None"===e.UpdateType?void a.fireAfterRenderDetached(e.CallBackData,!0):(a.events.list=e.Events,"Full"===e.UpdateType&&(a.startDate=e.StartDate,a.timeFormat=e.TimeFormat?e.TimeFormat:a.timeFormat,"undefined"!=typeof e.WeekStarts&&(a.weekStarts=e.WeekStarts),a.hashes=e.Hashes),a.V(),a.Jc(),a.da(),"Full"===e.UpdateType&&(a.Kc(),a.Lc()),a.ia(),a.show(),a.ja(),void a.fireAfterRenderDetached(e.CallBackData,!0))},this.fireAfterRenderDetached=function(e,t){var i=function(e,t){return function(){a.afterRender&&a.afterRender(e,t)}};window.setTimeout(i(e,t),0)},this.lineHeight=function(){return this.eventHeight+this.lineSpace},this.events={},this.events.add=function(e){var t=null;if(e instanceof DayPilot.Event)t=e.data;else{if("object"!=typeof e)throw"DayPilot.Month.events.add() expects an object or DayPilot.Event instance.";t=e}a.events.list||(a.events.list=[]),a.events.list.push(t),a.update(),a.ra.notify()},this.events.find=function(e){if(!a.events.list)return null;if("function"==typeof e){for(var t=e,i=0;i-1?"&"+s:"?"+s,DayPilot.Http.ajax({"method":"GET","url":r,"success":o,"error":n})}},this.update=function(e){if(a.ab(e),this.Aa){if(a.N)throw new DayPilot.Exception("You are trying to update a DayPilot.Month instance that has been disposed.");if(this.cells){a.V(),a.Jc(),a.da(),a.Kc(),a.Lc(),a.ia(),a.s(),a.ja(),this.visible?this.show():this.hide()}}},this.db=null,this.ab=function(e){if(e){var t={"events":{"preInit":function(){var e=this.data||[];DayPilot.isArray(e.list)?a.events.list=e.list:a.events.list=e}}};this.db=t;for(var i in e)if(t[i]){var n=t[i];n.data=e[i],n.preInit&&n.preInit()}else a[i]=e[i]}},this.eb=function(){var e=this.db;for(var t in e){var i=e[t];i.postInit&&i.postInit()}},this.gb={},this.gb.events=[],this.hb=function(e){var t=this.gb.events,i=this.events.list[e],n={};for(var o in i)n[o]=i[o];if("function"==typeof this.onBeforeEventRender){var r={};r.control=a,r.data=n,this.onBeforeEventRender(r)}t[e]=n},this.da=function(){var e=this.events.list;if(e){if(!DayPilot.isArray(e))throw new DayPilot.Exception("DayPilot.Month.events.list expects an array object. You supplied: "+typeof e);if("function"==typeof this.onBeforeEventRender)for(var t=0;to.getTime()))for(var t=0;t=7;)i--,c-=7;if(c>t){c>t+(7-this.getColCount())?(i--,t=t+7-c):(l=l-c+t,t=0)}else t-=c;i<0&&(i=0,t=0);var u=null;for(e.resizingEvent?u="w-resize":e.movingEvent&&(u="move"),this.nav.top.style.cursor=u;l>0&&i=this.end.getTime())},o.getPartStart=function(e){return DayPilot.DateUtil.max(this.start,e.start())},o.getPartEnd=function(e){return DayPilot.DateUtil.min(this.end,e.end())},o.getStartColumn=function(e){var t=this.getPartStart(e);return DayPilot.DateUtil.daysDiff(this.start,t)},o.getWidth=function(e){return DayPilot.DateUtil.daysSpan(this.getPartStart(e),this.getPartEnd(e))+1},o.putIntoLine=function(e,t,i,a){for(var n=this,o=0;othis[a].part.colStart+this[a].part.colWidth-1||(i=!1);return i},r.addEvent=function(e,t,i,a,o){e.part.colStart=t,e.part.colWidth=i,e.part.row=a,e.part.line=o,e.part.startsHere=n.start.getTime()<=e.start().getTime(),e.part.endsHere=n.end.getTime()>=e.end().getTime(),this.push(e)},r.addEvent(e,t,i,a,this.lines.length),this.lines.push(r),this.lines.length-1},o.getStart=function(){for(var e=0,t=0;t=7;)i.y++,t-=7;return i.x=t,i},this.Lc=function(){var e=document.createElement("div");e.oncontextmenu=function(){return!1},this.nav.cells.appendChild(e),this.cells=[];for(var t=0;t6&&(a-=7),i.className=this.J("_header");var o=document.createElement("div");o.setAttribute("unselectable","on"),o.innerHTML=n.locale().dayNames[a],i.appendChild(o),o.style.position="absolute",o.style.top="0px",o.style.bottom="0px",
+o.style.left="0px",o.style.right="0px",o.className=this.J("_header_inner"),o.innerHTML=n.locale().dayNames[a],e.appendChild(i);for(var r=0;r","replace":!0,"link":function(t,i,a){var n=new DayPilot.Month(i[0]);n.ra.scope=t,n.init();var o=a["id"];o&&(t[o]=n);var r=a["publishAs"];if(r){(0,e(r).assign)(t,n)}for(var s in a)0===s.indexOf("on")&&!function(i){n[i]=function(n){var o=e(a[i]);t["$apply"](function(){o(t,{"args":n})})}}(s);var l=t["$watch"],d=a["config"]||a["daypilotConfig"],c=a["events"]||a["daypilotEvents"];l.call(t,d,function(e){for(var t in e)n[t]=e[t];n.update()},!0),l.call(t,c,function(e){n.events.list=e,n.update()},!0)}}}])}(),"undefined"!=typeof Sys&&Sys.Application&&Sys.Application.notifyScriptLoaded&&Sys.Application.notifyScriptLoaded()}}(),"undefined"==typeof DayPilot)var DayPilot={};"undefined"==typeof DayPilot.Global&&(DayPilot.Global={}),function(){if("undefined"==typeof DayPilot.Navigator||!DayPilot.Navigator.nav){DayPilot.Navigator=function(e,t){this.v="2024.3.539-lite";var i=this;this.id=e,this.api=2,this.isNavigator=!0,this.autoFocusOnClick=!0,this.weekStarts="Auto",this.selectMode="Day",this.titleHeight=30,this.dayHeaderHeight=30,this.bound=null,this.cellWidth=30,this.cellHeight=30,this.cssClassPrefix="navigator_default",this.freeHandSelectionEnabled=!1,this.selectionStart=(new DayPilot.Date).getDatePart(),this.selectionEnd=null,this.selectionDay=null,this.showMonths=1,this.skipMonths=1,this.command="navigate",this.year=(new DayPilot.Date).getYear(),this.month=(new DayPilot.Date).getMonth()+1,this.showWeekNumbers=!1,this.weekNumberAlgorithm="Auto",this.rowsPerMonth="Six",this.orientation="Vertical",this.locale="en-us",this.rtl=!1,this.visible=!0,this.timeRangeSelectedHandling="Bind",this.visibleRangeChangedHandling="Enabled",this.onVisibleRangeChange=null,this.onVisibleRangeChanged=null,this.onTimeRangeSelect=null,this.onTimeRangeSelected=null,this.nav={},this.gb={},this.Pc=function(){this.root.dp=this,this.root.className=this.J("_main"),"Horizontal"===this.orientation?(this.root.style.width=this.showMonths*(7*o.cellWidth()+this.Qc())+"px",this.root.style.height=6*this.cellHeight+this.titleHeight+this.dayHeaderHeight+"px"):this.root.style.width=7*o.cellWidth()+this.Qc()+"px",this.rtl&&(this.root.style.direction="rtl"),this.root.style.position="relative",this.visible||(this.root.style.display="none");var e=document.createElement("input");e.type="hidden",e.name=i.id+"_state",e.id=e.name,this.root.appendChild(e),this.state=e,this.startDate?this.startDate=new DayPilot.Date(this.startDate).firstDayOfMonth():this.startDate=DayPilot.Date.fromYearMonthDay(this.year,this.month),this.calendars=[],this.selected=[],this.months=[]},this.pa=function(){return 2===i.api},this.Kc=function(){this.root.innerHTML=""},this.J=function(e){var t=this.theme||this.cssClassPrefix;return t?t+e:""},this.Rc=function(e,t){var i=this.J("_"+t);DayPilot.Util.addClass(e,i)},this.Sc=function(e,t){var i=this.J("_"+t);DayPilot.Util.removeClass(e,i)},this.Lc=function(e,t){var a={};a.cells=[],a.days=[],a.weeks=[];var n=this.startDate.addMonths(e),r=t.before,s=t.after,l=n.firstDayOfMonth(),d=l.firstDayOfWeek(o.weekStarts()),c=l.addMonths(1),u=DayPilot.DateUtil.daysDiff(d,c),h="Auto"===this.rowsPerMonth?Math.ceil(u/7):6;a.rowCount=h;var f=(new DayPilot.Date).getDatePart(),m=7*o.cellWidth()+this.Qc();a.width=m;var p=this.cellHeight*h+this.titleHeight+this.dayHeaderHeight;a.height=p;var v=document.createElement("div");if(v.style.width=m+"px",v.style.height=p+"px","Horizontal"===this.orientation)v.style.position="absolute",v.style.left=m*e+"px",v.style.top="0px",a.top=0,a.left=m*e;else{v.style.position="relative";var y=e>0?i.months[e-1].top+i.months[e-1].height:0;a.top=y,a.left=0}v.className=this.J("_month"),v.style.cursor="default",v.style.MozUserSelect="none",v.style.KhtmlUserSelect="none",v.style.WebkitUserSelect="none",v.month=a,this.root.appendChild(v);var g=this.titleHeight+this.dayHeaderHeight,b=document.createElement("div");b.style.position="absolute",b.style.left="0px",b.style.right="0px",b.style.top="0px",b.style.width=o.cellWidth()+"px",b.style.height=this.titleHeight+"px",b.style.lineHeight=this.titleHeight+"px",b.setAttribute("unselectable","on"),b.className=this.J("_titleleft"),t.left&&(b.style.cursor="pointer",b.innerHTML="<",b.onclick=this.Tc),v.appendChild(b),this.tl=b;var D=document.createElement("div");D.style.position="absolute",D.style.left=o.cellWidth()+"px",D.style.top="0px",D.style.width=5*o.cellWidth()+this.Qc()+"px",D.style.height=this.titleHeight+"px",D.style.lineHeight=this.titleHeight+"px",D.setAttribute("unselectable","on"),D.className=this.J("_title"),D.innerHTML=o.locale().monthNames[n.getMonth()]+" "+n.getYear(),v.appendChild(D),this.ti=D;var x=document.createElement("div");x.style.position="absolute",x.style.left=6*o.cellWidth()+this.Qc()+"px",x.style.right=6*o.cellWidth()+this.Qc()+"px",x.style.top="0px",x.style.width=o.cellWidth()+"px",x.style.height=this.titleHeight+"px",x.style.lineHeight=this.titleHeight+"px",x.setAttribute("unselectable","on"),x.className=this.J("_titleright"),t.right&&(x.style.cursor="pointer",x.innerHTML=">",x.onclick=this.Uc),v.appendChild(x),this.tr=x;var w=this.Qc();if(this.showWeekNumbers)for(var C=0;C"+k+"",v.appendChild(_),a.weeks.push(_)}for(var S=0;S<7;S++){a.cells[S]=[];var _=document.createElement("div");_.style.position="absolute",_.style.left=S*o.cellWidth()+w+"px",_.style.right=S*o.cellWidth()+w+"px",_.style.top=this.titleHeight+"px",_.style.width=o.cellWidth()+"px",_.style.height=this.dayHeaderHeight+"px",_.style.lineHeight=this.dayHeaderHeight+"px",_.setAttribute("unselectable","on"),_.className=this.J("_dayheader"),_.innerHTML=""+this.Vc(S)+"",v.appendChild(_),a.days.push(_);for(var C=0;C=n.addMonths(1);if("month"===this.Xc())M=M&&T;else if("day"===this.Xc())M=M&&(T||r&&A||s&&H);else if("week"===this.Xc()){var E=P.firstDayOfMonth()===n;M=M&&(E||r&&A||s&&H)}var N=document.createElement("div");a.cells[S][C]=N;var I=i.Yc(S,C),R=I.x,O=I.y;N.day=P,N.x=S,N.y=C,N.left=R,N.top=O,N.isCurrentMonth=T,N.isNextMonth=H,N.isPrevMonth=A,N.showBefore=r,N.showAfter=s,N.className=this.J(T?"_day":"_dayother"),i.Rc(N,"cell"),P.getTime()===f.getTime()&&T&&this.Rc(N,"today"),0!==P.dayOfWeek()&&6!==P.dayOfWeek()||this.Rc(N,"weekend"),N.style.position="absolute",N.style.left=R+"px",N.style.right=R+"px",N.style.top=O+"px",N.style.width=o.cellWidth()+"px",N.style.height=this.cellHeight+"px",N.style.lineHeight=this.cellHeight+"px";var z=document.createElement("div");z.style.position="absolute",z.className=P.getTime()===f.getTime()&&T?this.J("_todaybox"):this.J("_daybox"),i.Rc(z,"cell_box"),z.style.left="0px",z.style.top="0px",z.style.right="0px",z.style.bottom="0px",N.appendChild(z);var U=null;if(this.cells&&this.cells[P.toStringSortable()]&&(U=this.cells[P.toStringSortable()]),"function"==typeof i.onBeforeCellRender){var B={};B.cell=U||{},B.cell.day=P,B.cell.isCurrentMonth=T,B.cell.isToday=P.getTime()===f.getTime()&&T,B.cell.isWeekend=0===P.dayOfWeek()||6===P.dayOfWeek(),U?(B.cell.html=U.html||P.getDay(),B.cell.cssClass=U.css):(B.cell.html=P.getDay(),B.cell.cssClass=null),i.onBeforeCellRender(B),U=B.cell}if(U&&DayPilot.Util.addClass(N,U.cssClass||U.css),T||r&&A||s&&H){var L=document.createElement("div");L.innerHTML=P.getDay(),L.style.position="absolute",L.style.left="0px",L.style.top="0px",L.style.right="0px",L.style.bottom="0px",i.Rc(L,"cell_text"),N.isClickable=!0,U&&U.html&&(L.innerHTML=U.html),N.appendChild(L)}N.setAttribute("unselectable","on"),N.onclick=this.Zc,v.appendChild(N),M&&(i.$c(v,S,C),this.selected.push(N))}}var j=document.createElement("div");j.style.position="absolute",j.style.left="0px",j.style.top=g-2+"px",j.style.width=7*o.cellWidth()+this.Qc()+"px",j.style.height="1px",j.style.fontSize="1px",j.style.lineHeight="1px",j.className=this.J("_line"),v.appendChild(j),this.months.push(a)},this.Yc=function(e,t){var i=this.titleHeight+this.dayHeaderHeight,a=this.Qc();return{"x":e*o.cellWidth()+a,"y":t*this.cellHeight+i}},this.$c=function(e,t,a){var n=e.month.cells[t][a];i.Rc(n,"select")},this._c=function(e,t,a){var n=e.month.cells[t][a];i.Sc(n,"select")},this.Qc=function(){return this.showWeekNumbers?o.cellWidth():0},this.ad=function(){if(this.items)for(var e=0;e=this.fd())&&(m=this.selectionStart.firstDayOfMonth()),m.toStringSortable()!==this.startDate.toStringSortable()&&(f=!0),this.startDate=m}s&&i.freeHandSelectionEnabled?i.selectionEnd=new DayPilot.Date(s):this.cd(),this.Kc(),this.Pc(),this.gd(),this.ad(),this.bd(),!c||u.equals(this.selectionStart)&&h.equals(this.selectionEnd)||this.L(),f&&this.hd()},this.update=function(e){if(i.ab(e),this.Aa){if(i.N)throw new DayPilot.Exception("You are trying to update a DayPilot.Navigator instance that has been disposed.");i.jd();var t={"day":i.selectionDay,"start":i.selectionStart,"end":i.selectionEnd};i.$a(),t.start===i.selectionStart&&t.end==i.selectionEnd&&t.day===i.selectionDay||i.L()}},this.$a=function(){this.Kc(),this.Pc(),this.cd(),this.gd(),this.da(),this.ad(),this.bd(),this.visible?this.show():this.hide()},this.jd=function(){i.gb={}},this.db=null,this.ab=function(e){if(e){var t={"events":{"preInit":function(){var e=this.data||[];DayPilot.isArray(e.list)?i.events.list=e.list:i.events.list=e}}};this.db=t;for(var a in e)if(t[a]){var n=t[a];n.data=e[a],n.preInit&&n.preInit()}else i[a]=e[a]}},this.eb=function(){var e=this.db;for(var t in e){var i=e[t];i.postInit&&i.postInit()}},this.R=function(e,t,i){var a={};a.action=e,a.parameters=i,a.data=t,a.header=this.Q();var n="JSON"+JSON.stringify(a);this.backendUrl?DayPilot.request(this.backendUrl,this.S,n,this.kd):WebForm_DoCallback(this.uniqueID,n,this.T,null,this.callbackError,!0)},this.kd=function(e){if("function"==typeof i.onAjaxError){var t={};t.request=e,i.onAjaxError(t)}else"function"==typeof i.ajaxError&&i.ajaxError(e)},this.S=function(e){i.T(e.responseText)},this.P=function(e,t,a){var n={};n.action=e,n.parameters=a,n.data=t,n.header=this.Q();var o="JSON"+JSON.stringify(n);__doPostBack(i.uniqueID,o)},this.Q=function(){var e={};return e.v=this.v,e.startDate=this.startDate,e.selectionStart=this.selectionStart,e.showMonths=this.showMonths,e},this.ld=function(e,t){"refresh"===e&&this.hd()},this.Vc=function(e){var t=e+o.weekStarts();return t>6&&(t-=7),o.locale().dayNamesShort[t]},this.Wc=function(e){return null!==this.selectionStart&&null!==this.selectionEnd&&(this.selectionStart.getTime()<=e.getTime()&&e.getTime()<=this.selectionEnd.getTime())},this.md=function(e){for(var t=0;tn&&(a.end=t)}a.end&&(a.clear(),a.draw())}},this.qd={};var a=this.qd;a.start=null,a.drawCell=function(e){var t=i.months[e.month],n=i.Yc(e.x,e.y),o=t.top+n.y,r=t.left+n.x,s=document.createElement("div");s.style.position="absolute",s.style.left=r+"px",s.style.top=o+"px",s.style.height=i.cellHeight+"px",s.style.width=i.cellWidth+"px",s.style.backgroundColor="#ccc",s.style.opacity=.5,s.style.cursor="default",i.nav.preselection.appendChild(s),a.cells.push(s)},a.clear=function(){if(a.cells){for(var e=0;e0){o.month-=1;var r=i.months[o.month];o.x=6,o.y=r.rowCount-1}for(a.cells=[];!t.is(o);){a.drawCell(t);var s=new n(t).next();if(!s)return;t.month=s.month,t.x=s.x,t.y=s.y}a.drawCell(t)}},a.ordered=function(){var e=a.start,t=a.end,i={};return!t||new n(e).before(t)?(i.start=e,i.end=t):(i.start=t,i.end=e),i};var n=function(e,t,a){if(e instanceof n)return e;if("object"==typeof e){var o=e;this.month=o.month,this.x=o.x,this.y=o.y}else this.month=e,this.x=t,this.y=a;this.is=function(e){return this.month===e.month&&this.x===e.x&&this.y===e.y},this.next=function(){var e=this;if(e.x<6)return{"month":e.month,"x":e.x+1,"y":e.y};var t=i.months[e.month];return e.y0)return{"month":e.month,"x":e.x-1,"y":e.y};i.months[e.month];if(e.y>0)return{"month":e.month,"x":6,"y":e.y-1};if(e.month>0){var t=i.months[e.month-1];return{"month":e.month-1,"x":6,"y":t.rowCount-1}}return null},this.previousVisible=function(){for(var e=this;!e.visible();){var t=e.previous();if(!t)return null;e=new n(t)}return e},this.cell=function(){return i.months[this.month].cells[this.x][this.y]},this.date=function(){return this.cell().day},this.before=function(e){return this.date()=i.fd())return void i.select(r)}var l=a.cells[n][o];i.$c(t,n,o),i.selected.push(l),i.selectionStart=l.day,i.selectionEnd=l.day;break;case"week":if(i.autoFocusOnClick){var s=a.cells[0][o].day,d=a.cells[6][o].day;if(s.firstDayOfMonth()===d.firstDayOfMonth()&&(s=i.fd()))return void i.select(r)}for(var c=0;c<7;c++)i.$c(t,c,o),i.selected.push(a.cells[c][o]);i.selectionStart=a.cells[0][o].day,i.selectionEnd=a.cells[6][o].day;break;case"month":if(i.autoFocusOnClick){var s=r;if(r=i.fd())return void i.select(r)}for(var s=null,d=null,o=0;o<6;o++)for(var n=0;n<7;n++){var l=a.cells[n][o];l&&l.day.getYear()===r.getYear()&&l.day.getMonth()===r.getMonth()&&(i.$c(t,n,o),i.selected.push(l),null===s&&(s=l.day),d=l.day)}i.selectionStart=s,i.selectionEnd=d;break;default:throw"unknown selectMode"}i.bd(),i.L()}},this.L=function(e){var t=i.selectionStart,a=i.selectionEnd.addDays(1),n=DayPilot.DateUtil.daysDiff(t,a),o=i.selectionDay;if(e=e||{},i.pa()){var r={};if(r.start=t,r.end=a,r.day=o,r.days=n,r.mode=e.mode||i.selectMode,r.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof i.onTimeRangeSelect&&(i.onTimeRangeSelect(r),r.preventDefault.value))return;switch(i.timeRangeSelectedHandling){case"Bind":if("object"==typeof bound){var s={};s.start=t,s.end=a,s.days=n,s.day=o,bound.commandCallBack(i.command,s)}break;case"None":break;case"PostBack":i.timeRangeSelectedPostBack(t,a,o)}"function"==typeof i.onTimeRangeSelected&&i.onTimeRangeSelected(r)}else switch(i.timeRangeSelectedHandling){case"Bind":if("object"==typeof bound){var s={};s.start=t,s.end=a,s.days=n,s.day=o,bound.commandCallBack(i.command,s)}break;case"JavaScript":i.onTimeRangeSelected(t,a,o);break;case"None":break;case"PostBack":i.timeRangeSelectedPostBack(t,a,o)}},this.timeRangeSelectedPostBack=function(e,t,i,a){var n={};n.start=e,n.end=t,n.day=a,this.P("TimeRangeSelected",i,n)},this.Uc=function(e){i.rd(i.skipMonths)},this.Tc=function(e){i.rd(-i.skipMonths)},this.rd=function(e){this.startDate=this.startDate.addMonths(e),this.Kc(),this.Pc(),this.gd(),this.bd(),this.hd(),this.ad()},this.ed=function(){return i.startDate.firstDayOfMonth()},this.fd=function(){return i.startDate.firstDayOfMonth().addMonths(this.showMonths)},this.visibleStart=function(){return i.startDate.firstDayOfMonth().firstDayOfWeek(o.weekStarts())},this.visibleEnd=function(){return i.startDate.firstDayOfMonth().addMonths(this.showMonths-1).firstDayOfWeek(o.weekStarts()).addDays(42)},this.hd=function(){var e=this.visibleStart(),t=this.visibleEnd();if(i.pa()){var a={};if(a.start=e,a.end=t,a.preventDefault=function(){this.preventDefault.value=!0},"function"==typeof i.onVisibleRangeChange&&(i.onVisibleRangeChange(a),a.preventDefault.value))return;switch(this.visibleRangeChangedHandling){case"CallBack":this.visibleRangeChangedCallBack(null);break;case"PostBack":this.visibleRangeChangedPostBack(null);break;case"Disabled":}"function"==typeof i.onVisibleRangeChanged&&i.onVisibleRangeChanged(a)}else switch(this.visibleRangeChangedHandling){case"CallBack":this.visibleRangeChangedCallBack(null);break;case"PostBack":this.visibleRangeChangedPostBack(null);break;case"JavaScript":this.onVisibleRangeChanged(e,t);break;case"Disabled":}},this.visibleRangeChangedCallBack=function(e){var t={};this.R("Visible",e,t)},this.visibleRangeChangedPostBack=function(e){var t={};this.P("Visible",e,t)},this.T=function(e,t){var e=JSON.parse(e);i.items=e.Items,i.cells=e.Cells,i.cells?i.update():i.ad()},this.gd=function(){for(var e=0;ee&&(e=i.height)}return e}for(var a=0,t=0;t","compile":function(t,i){return t.replaceWith(this["template"].replace("{{id}}",i["id"])),function(t,i,a){var n=new DayPilot.Navigator(i[0]);n.ra.scope=t,n.init();var o=a["id"];o&&(t[o]=n);var r=a["publishAs"];if(r){(0,e(r).assign)(t,n)}for(var s in a)if(0===s.indexOf("on")){var l=DayPilot.Util.shouldApply(s);l?!function(i){n[i]=function(n){var o=e(a[i]);t["$apply"](function(){o(t,{"args":n})})}}(s):!function(i){n[i]=function(n){e(a[i])(t,{"args":n})}}(s)}var d=t["$watch"],c=a["config"]||a["daypilotConfig"],u=a["events"]||a["daypilotEvents"];d.call(t,c,function(e,t){for(var i in e)n[i]=e[i];n.update()},!0),d.call(t,u,function(e){n.events.list=e,n.da(),n.ad()},!0)}}}}])}(),"undefined"!=typeof Sys&&Sys.Application&&Sys.Application.notifyScriptLoaded&&Sys.Application.notifyScriptLoaded()}}();
\ No newline at end of file
diff --git a/static/src/js/daypilot-all.src.js b/static/src/js/daypilot-all.src.js
new file mode 100644
index 0000000..2a19a54
--- /dev/null
+++ b/static/src/js/daypilot-all.src.js
@@ -0,0 +1,17290 @@
+var DayPilot = {};/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+(function() {
+
+ if (typeof DayPilot.$ !== 'undefined') {
+ return;
+ }
+
+ if (typeof DayPilot.Global === "undefined") {
+ DayPilot.Global = {};
+ }
+
+ DayPilot.$ = function(id) {
+ return document.getElementById(id);
+ };
+
+ Object.defineProperty(DayPilot, 'isKhtml', {
+ get: function() { return typeof navigator !== "undefined" && navigator.userAgent && navigator.userAgent.indexOf("KHTML") !== -1; }
+ });
+
+ DayPilot.mo2 = function (target, ev){
+ ev = ev || window.event;
+
+ // IE
+ if (typeof(ev.offsetX) !== 'undefined') {
+
+ var coords = {x: ev.offsetX + 1, y:ev.offsetY + 1};
+ //document.title = "ie/doc:" + document.documentElement.scrollTop;
+
+ if (!target) {
+ return coords;
+ }
+
+ var current = ev.srcElement;
+ while (current && current !== target) {
+ if (current.tagName !== 'SPAN') { // hack for DayPilotMonth/IE, hour info on the right side of an event
+ coords.x += current.offsetLeft;
+ if (current.offsetTop > 0) { // hack for http://forums.daypilot.org/Topic.aspx/879/move_event_bug
+ coords.y += current.offsetTop - current.scrollTop;
+ }
+ }
+
+ current = current.offsetParent;
+ }
+
+ if (current) {
+ return coords;
+ }
+ return null;
+ }
+
+ // FF
+ if (typeof(ev.layerX) !== 'undefined') {
+
+ var coords = {x:ev.layerX, y:ev.layerY, src: ev.target};
+
+ if (!target) {
+ return coords;
+ }
+ var current = ev.target;
+
+ // find the positioned offsetParent, the layerX reference
+ while (current && current.style.position !== 'absolute' && current.style.position !== 'relative') {
+ current = current.parentNode;
+ if (DayPilot.isKhtml) { // hack for KHTML (Safari and Google Chrome), used in DPC/event moving
+ coords.y += current.scrollTop;
+ }
+ }
+
+ while (current && current !== target) {
+ coords.x += current.offsetLeft;
+ coords.y += current.offsetTop - current.scrollTop;
+ current = current.offsetParent;
+ }
+ if (current) {
+ return coords;
+ }
+
+ return null;
+ }
+
+ return null;
+ };
+
+ // mouse offset relative to the specified target
+ DayPilot.mo3 = function(target, ev) {
+ var coords;
+ var page = DayPilot.page(ev);
+ if (page) {
+ if (target) {
+ var abs = DayPilot.abs(target);
+ if (!abs) {
+ throw new Error("no abs");
+ return null;
+ }
+ coords = {x: page.x - abs.x, y: page.y - abs.y};
+ }
+ else {
+ coords = {x: page.x, y: page.y};
+ }
+ }
+ else {
+ coords = DayPilot.mo2(target, ev);
+ if (!coords) {
+ // throw new Error("no coords");
+ return null;
+ }
+ }
+
+ coords.shift = ev.shiftKey;
+ coords.meta = ev.metaKey;
+ coords.ctrl = ev.ctrlKey;
+ coords.alt = ev.altKey;
+
+ return coords;
+ };
+
+ DayPilot.browser = {};
+
+ Object.defineProperty(DayPilot.browser, 'hover', {
+ get: function() { return !window.matchMedia("(any-hover: none)").matches; }
+ });
+
+ DayPilot.touch = {};
+
+ // returns pageX, pageY (calculated from clientX if pageX is not available)
+ DayPilot.page = function(ev) {
+ // required for chrome/android
+ var target = ev.changedTouches && ev.changedTouches[0] ? ev.changedTouches[0] : ev;
+
+ if (typeof target.pageX !== 'undefined') {
+ return {x: target.pageX, y: target.pageY};
+ }
+ if (typeof ev.clientX !== 'undefined' && document.body && document.documentElement) {
+ return {
+ x: ev.clientX + document.body.scrollLeft + document.documentElement.scrollLeft,
+ y: ev.clientY + document.body.scrollTop + document.documentElement.scrollTop
+ };
+ }
+ // shouldn't happen
+ return null;
+ };
+
+ // absolute element position on page
+ DayPilot.abs = function(element, visible) {
+ if (!element) {
+ return null;
+ }
+
+ if (element.getBoundingClientRect) {
+ var r = DayPilot.absBoundingClientBased(element);
+
+ if (visible) {
+ // use diff, absOffsetBased is not as accurate
+ var full = DayPilot.absOffsetBased(element, false);
+ var visible = DayPilot.absOffsetBased(element, true);
+
+ r.x += visible.x - full.x;
+ r.y += visible.y - full.y;
+ r.w = visible.w;
+ r.h = visible.h;
+ }
+
+ return r;
+ }
+ else {
+ return DayPilot.absOffsetBased(element, visible);
+ }
+
+ };
+
+ DayPilot.absBoundingClientBased = function(element) {
+ var elemRect = element.getBoundingClientRect();
+
+ return {
+ // x: elemRect.left + window.scrollX, // IE11 doesn't support this
+ x: elemRect.left + window.pageXOffset,
+ // y: elemRect.top + window.scrollY, // IE11 doesn't support this
+ y: elemRect.top + window.pageYOffset,
+ w: element.clientWidth,
+ h: element.clientHeight,
+ toString: function() {
+ return "x:" + this.x + " y:" + this.y + " w:" + this.w + " h:" + this.h;
+ }
+ };
+ };
+
+
+ // old implementation of absolute position
+ // problems with adjacent float and margin-left in IE7
+ // still the best way to calculate the visible part of the element
+ DayPilot.absOffsetBased = function(element, visible) {
+ var r = {
+ x: element.offsetLeft,
+ y: element.offsetTop,
+ w: element.clientWidth,
+ h: element.clientHeight,
+ toString: function() {
+ return "x:" + this.x + " y:" + this.y + " w:" + this.w + " h:" + this.h;
+ }
+ };
+
+ while (element.offsetParent) {
+ element = element.offsetParent;
+
+ r.x -= element.scrollLeft;
+ r.y -= element.scrollTop;
+
+ if (visible) { // calculates the visible part
+ if (r.x < 0) {
+ r.w += r.x; // decrease width
+ r.x = 0;
+ }
+
+ if (r.y < 0) {
+ r.h += r.y; // decrease height
+ r.y = 0;
+ }
+
+ if (element.scrollLeft > 0 && r.x + r.w > element.clientWidth) {
+ r.w -= r.x + r.w - element.clientWidth;
+ }
+
+ if (element.scrollTop && r.y + r.h > element.clientHeight) {
+ r.h -= r.y + r.h - element.clientHeight;
+ }
+ }
+
+ r.x += element.offsetLeft;
+ r.y += element.offsetTop;
+
+ }
+
+ var pageOffset = DayPilot.pageOffset();
+ r.x += pageOffset.x;
+ r.y += pageOffset.y;
+
+ return r;
+ };
+
+ DayPilot.isArray = function(o) {
+ return Object.prototype.toString.call(o) === '[object Array]';
+ };
+
+ // distance of two points, works with x and y
+ DayPilot.distance = function(point1, point2) {
+ return Math.sqrt(Math.pow(point1.x - point2.x, 2) + Math.pow(point1.y - point2.y, 2));
+ };
+
+ DayPilot.sheet = function() {
+
+ if (typeof window === "undefined") {
+ // next.js server-side rendering
+ var sheet = {};
+ sheet.add = function() {};
+ sheet.commit = function() {};
+ return sheet;
+ }
+
+ var style = document.createElement("style");
+ style.setAttribute("type", "text/css");
+ if (!style.styleSheet) { // ie
+ style.appendChild(document.createTextNode(""));
+ }
+
+ var h = document.head || document.getElementsByTagName('head')[0];
+ h.appendChild(style);
+
+ var oldStyle = !! style.styleSheet; // old ie
+
+ var sheet = {};
+ sheet.rules = [];
+ sheet.commit = function() {
+ if (oldStyle) {
+ style.styleSheet.cssText = this.rules.join("\n");
+ }
+ };
+
+ sheet.add = function(selector, rules, index) {
+ if (oldStyle) {
+ this.rules.push(selector + "{" + rules + "\u007d");
+ return;
+ }
+ if(style.sheet.insertRule) {
+ if (typeof index === "undefined") {
+ index = style.sheet.cssRules.length;
+ }
+ style.sheet.insertRule(selector + "{" + rules + "\u007d", index);
+ }
+ else if (style.sheet.addRule) {
+ style.sheet.addRule(selector, rules, index);
+ }
+ };
+ return sheet;
+ };
+
+ DayPilot.gs = function(el, styleProp) {
+ return window.getComputedStyle(el, null).getPropertyValue(styleProp) || "";
+ };
+
+ DayPilot.StyleReader = function(element) {
+
+ this.get = function(property) {
+ if (!element) {
+ return null;
+ }
+ return DayPilot.gs(element, property);
+ };
+
+ this.getPx = function(property) {
+ var val = this.get(property);
+ if (val.indexOf("px") === -1) {
+ return undefined;
+ }
+ else {
+ return parseInt(val, 10);
+ }
+ };
+
+ };
+
+ // register the default themes
+ (function() {
+ if (DayPilot.Global.defaultCss) {
+ return;
+ }
+
+ var sheet = DayPilot.sheet();
+
+ sheet.add(".calendar_default_main", "border: 1px solid #c0c0c0; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;");
+ sheet.add(".calendar_default_main *, .calendar_default_main *:before, .calendar_default_main *:after", "box-sizing: content-box;"); // bootstrap
+ sheet.add(".calendar_default_rowheader_inner,.calendar_default_cornerright_inner,.calendar_default_corner_inner,.calendar_default_colheader_inner,.calendar_default_alldayheader_inner", "color: #333;background: #f3f3f3;");
+ sheet.add(".calendar_default_cornerright_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px; border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_direction_rtl .calendar_default_cornerright_inner", "border-right: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_rowheader_inner", "font-size: 16pt;text-align: right; position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0; padding: 3px;");
+ sheet.add(".calendar_default_direction_rtl .calendar_default_rowheader_inner", "border-right: none;");
+ sheet.add(".calendar_default_corner_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_direction_rtl .calendar_default_corner_inner", "border-right: none;");
+ sheet.add(".calendar_default_rowheader_minutes", "font-size:10px;vertical-align: super;padding-left: 2px;padding-right: 2px;");
+ sheet.add(".calendar_default_colheader_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0; display: flex; align-items: center; justify-content: center; font-size: 13px;");
+ sheet.add(".calendar_default_cell_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #ddd;border-bottom: 1px solid #ddd; background: #f9f9f9;");
+ sheet.add(".calendar_default_cell_business .calendar_default_cell_inner", "background: #fff");
+ sheet.add(".calendar_default_alldayheader_inner", "text-align: center;position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_message", "opacity: 0.9; padding: 10px; color: #ffffff;background: #ffa216;");
+ sheet.add(".calendar_default_alldayevent_inner,.calendar_default_event_inner", 'color: #333; border: 1px solid #999;'); // border-top: 4px solid #1066a8;
+ sheet.add(".calendar_default_event_bar", "top: 0px;bottom: 0px;left: 0px;width: 6px;background-color: #9dc8e8;");
+ sheet.add(".calendar_default_event_bar_inner", "position: absolute;width: 6px;background-color: #1066a8;");
+ sheet.add(".calendar_default_alldayevent_inner,.calendar_default_event_inner", 'background: #fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".calendar_default_selected .calendar_default_event_inner", "background: #ddd;");
+ sheet.add(".calendar_default_alldayevent_inner", "position: absolute;top: 2px;bottom: 2px;left: 2px;right: 2px;overflow:hidden;padding: 2px;margin-right: 1px; font-size: 13px;");
+ sheet.add(".calendar_default_event_withheader .calendar_default_event_inner", "padding-top: 15px;");
+ sheet.add(".calendar_default_event", "cursor: default;");
+ sheet.add(".calendar_default_event_inner", "position: absolute;overflow: hidden;top: 0px;bottom: 0px;left: 0px;right: 0px;padding: 2px 2px 2px 8px; font-size: 13px;");
+ sheet.add(".calendar_default_event_delete", "background: url() center center no-repeat; opacity: 0.6; cursor: pointer;");
+ sheet.add(".calendar_default_event_delete:hover", "opacity: 1;-ms-filter: none;");
+ sheet.add(".calendar_default_scroll_up", "background: url();");
+ sheet.add(".calendar_default_scroll_down", "background: url();");
+ sheet.add(".calendar_default_now", "background-color: red;");
+ sheet.add(".calendar_default_now:before", "content: ''; top: -5px; border-width: 5px; border-color: transparent transparent transparent red; border-style: solid; width: 0px; height:0px; position: absolute; -moz-transform: scale(.9999);");
+ sheet.add(".calendar_default_shadow_top", 'box-sizing: border-box; padding:2px;border:1px solid #ccc;background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".calendar_default_shadow_bottom", 'box-sizing: border-box; padding:2px;border:1px solid #ccc;background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".calendar_default_crosshair_vertical, .calendar_default_crosshair_horizontal, .calendar_default_crosshair_left, .calendar_default_crosshair_top", "background-color: gray; opacity: 0.2;");
+ sheet.add(".calendar_default_loading", "background-color: orange; color: white; padding: 2px;");
+ sheet.add(".calendar_default_scroll", "background-color: #f3f3f3;");
+ sheet.add(".calendar_default_event_moving_source", "opacity: 0.5;");
+ sheet.add(".calendar_default_shadow_inner", "box-sizing: border-box; background-color: #bbbbbb;border: 1px solid #888888;opacity: 0.5;height: 100%;");
+ sheet.add(".calendar_default_shadow", "box-shadow: 0 2px 5px rgba(0,0,0,.2);");
+
+
+ // menu
+ sheet.add(".menu_default_main", "user-select:none; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif;font-size: 13px;border: 1px solid #dddddd;background-color: white;padding: 0px;cursor: default;background-image: url();background-repeat: repeat-y;xborder-radius: 5px;-moz-box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);-webkit-box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);");
+ sheet.add(".menu_default_main, .menu_default_main *, .menu_default_main *:before, .menu_default_main *:after", "box-sizing: content-box;");
+ sheet.add(".menu_default_title", "background-color: #f2f2f2;border-bottom: 1px solid gray;padding: 4px 4px 4px 37px;");
+ sheet.add(".menu_default_main a", "padding: 2px 2px 2px 35px;color: black;text-decoration: none;cursor: default;");
+ sheet.add(".menu_default_main.menu_default_withchildren a", "padding: 2px 35px 2px 35px;");
+ sheet.add(".menu_default_main a img", "margin-left: 6px;margin-top: 2px;");
+ sheet.add(".menu_default_item_text", "display: block;height: 20px;line-height: 20px; overflow:hidden;padding-left: 2px;padding-right: 20px; white-space: nowrap;");
+ sheet.add(".menu_default_main a:hover", "background-color: #f3f3f3;");
+ sheet.add(".menu_default_main div div", "border-top: 1px solid #dddddd;margin-top: 2px;margin-bottom: 2px;margin-left: 28px;");
+ sheet.add(".menu_default_main a.menu_default_item_disabled", "color: #ccc");
+ sheet.add(".menu_default_item_haschildren.menu_default_item_haschildren_active", 'background-color: #f3f3f3;');
+ sheet.add(".menu_default_item_haschildren a:before", "content: ''; border-width: 5px; border-color: transparent transparent transparent #666; border-style: solid; width: 0px; height:0px; position: absolute; right: 5px; margin-top: 5px;");
+ sheet.add(".menu_default_item_icon", "position: absolute; top:0px; left: 0px; padding: 2px 2px 2px 8px;");
+ sheet.add(".menu_default_item a i", "height: 20px;line-height: 20px;");
+ sheet.add(".menu_default_item .menu_default_item_symbol", "width: 18px; height: 18px; color: #999; margin-left: 6px;margin-top: 2px;");
+
+
+ // menubar
+ sheet.add(".menubar_default_main", "border-bottom: 1px solid #ccc; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px; user-select:none;");
+ sheet.add(".menubar_default_item", "display: inline-block; padding: 6px 10px; cursor: default;");
+ sheet.add(".menubar_default_item:hover", "background-color: #f2f2f2;");
+ sheet.add(".menubar_default_item_active", "background-color: #f2f2f2;");
+
+
+ sheet.add(".scheduler_default_selected .scheduler_default_event_inner", "background: #ddd;");
+ sheet.add(".scheduler_default_main", "border: 1px solid #c0c0c0;font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;");
+ sheet.add(".scheduler_default_timeheader", "cursor: default;color: #333;");
+ sheet.add(".scheduler_default_message", "opacity: 0.9;filter: alpha(opacity=90);padding: 10px; color: #ffffff;background: #ffa216;");
+ sheet.add(".scheduler_default_timeheadergroup,.scheduler_default_timeheadercol", "color: #333;background: #f3f3f3;");
+ sheet.add(".scheduler_default_rowheader,.scheduler_default_corner", "color: #333;background: #f3f3f3;");
+ sheet.add(".scheduler_default_rowheader_inner", "position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #eee;padding: 2px;");
+ sheet.add(".scheduler_default_timeheadergroup, .scheduler_default_timeheadercol", "text-align: center;");
+ sheet.add(".scheduler_default_timeheadergroup_inner", "position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".scheduler_default_timeheadercol_inner", "position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #c0c0c0;");
+ sheet.add(".scheduler_default_divider", "background-color: #c0c0c0;");
+ sheet.add(".scheduler_default_divider_horizontal", "background-color: #c0c0c0;");
+ sheet.add(".scheduler_default_matrix_vertical_line", "background-color: #eee;");
+ sheet.add(".scheduler_default_matrix_vertical_break", "background-color: #000;");
+ sheet.add(".scheduler_default_matrix_horizontal_line", "background-color: #eee;");
+ sheet.add(".scheduler_default_resourcedivider", "background-color: #c0c0c0;");
+ sheet.add(".scheduler_default_shadow_inner", "background-color: #666666;opacity: 0.5;filter: alpha(opacity=50);height: 100%;xborder-radius: 5px;");
+ sheet.add(".scheduler_default_event", "color:#333; font-size: 13px;");
+ sheet.add(".scheduler_default_event_inner", "position:absolute;top:0px;left:0px;right:0px;bottom:0px;padding:5px 2px 2px 2px;overflow:hidden;border:1px solid #ccc;");
+ sheet.add(".scheduler_default_event_bar", "top:0px;left:0px;right:0px;height:4px;background-color:#9dc8e8;");
+ sheet.add(".scheduler_default_event_bar_inner", "position:absolute;height:4px;background-color:#1066a8;");
+ sheet.add(".scheduler_default_event_inner", 'background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".scheduler_default_event_float_inner", "padding:6px 2px 2px 8px;"); // space for arrow
+ sheet.add(".scheduler_default_event_float_inner:after", 'content:"";border-color: transparent #666 transparent transparent;border-style:solid;border-width:5px;width:0;height:0;position:absolute;top:8px;left:-4px;');
+ sheet.add(".scheduler_default_columnheader_inner", "font-weight: bold;");
+ sheet.add(".scheduler_default_columnheader_splitter", "background-color: #666;opacity: 0.5;filter: alpha(opacity=50);");
+ sheet.add(".scheduler_default_columnheader_cell_inner", "padding: 2px;");
+ sheet.add(".scheduler_default_cell", "background-color: #f9f9f9;");
+ sheet.add(".scheduler_default_cell.scheduler_default_cell_business", "background-color: #fff;");
+
+ sheet.add(".navigator_default_main", "border-left: 1px solid #c0c0c0;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;background-color: white;color: #000000; box-sizing: content-box;");
+ sheet.add(".navigator_default_main *, .navigator_default_main *:before, .navigator_default_main *:after", "box-sizing: content-box;");
+ sheet.add(".navigator_default_month", "font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 12px;");
+ sheet.add(".navigator_default_day", "color: black;");
+ sheet.add(".navigator_default_weekend", "background-color: #f0f0f0;");
+ sheet.add(".navigator_default_dayheader", "color: black;");
+ sheet.add(".navigator_default_line", "border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".navigator_default_dayother", "color: gray;");
+ sheet.add(".navigator_default_todaybox", "border: 1px solid red;");
+ sheet.add(".navigator_default_title, .navigator_default_titleleft, .navigator_default_titleright", "border-top: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;color: #333;background: #f3f3f3;");
+ sheet.add(".navigator_default_busy", "font-weight: bold;");
+ sheet.add(".navigator_default_cell", "text-align: center;");
+ sheet.add(".navigator_default_select .navigator_default_cell_box", "background-color: #FFE794; opacity: 0.5;");
+ sheet.add(".navigator_default_title", "text-align: center;");
+ sheet.add(".navigator_default_titleleft, .navigator_default_titleright", "text-align: center;");
+ sheet.add(".navigator_default_dayheader", "text-align: center;");
+ sheet.add(".navigator_default_weeknumber", "text-align: center; color: #999;");
+ sheet.add(".navigator_default_cell_text", "cursor: pointer;");
+
+ sheet.add(".month_default_main", "border: 1px solid #c0c0c0;font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;color: #333;");
+ sheet.add(".month_default_main *, .month_default_main *:before, .month_default_main *:after", "box-sizing: content-box;");
+ sheet.add(".month_default_cell_inner", "border-right: 1px solid #ddd;border-bottom: 1px solid #ddd;position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;background-color: #f9f9f9;");
+ sheet.add(".month_default_cell_business .month_default_cell_inner", "background-color: #fff;");
+ sheet.add(".month_default_cell_header", "text-align: right; padding: 4px; box-sizing: border-box;");
+ sheet.add(".month_default_header_inner", 'position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;cursor: default;color: #333;background: #f3f3f3; overflow:hidden; display: flex; align-items: center; justify-content: center;');
+ sheet.add(".month_default_message", 'padding: 10px;opacity: 0.9; color: #ffffff;background: #ffa216;');
+ sheet.add(".month_default_event_inner", 'position: absolute;top: 0px;bottom: 0px;left: 1px;right: 1px;overflow:hidden;padding: 2px;padding-left: 10px;color: #333;background: #fff;background: linear-gradient(to bottom, #ffffff 0%, #eeeeee);border: 1px solid #999;border-radius: 0px;display: flex; align-items: center; font-size: 13px;');
+ sheet.add(".month_default_event_continueright .month_default_event_inner", "border-top-right-radius: 0px;border-bottom-right-radius: 0px;border-right-style: dotted;");
+ sheet.add(".month_default_event_continueleft .month_default_event_inner", "border-top-left-radius: 0px;border-bottom-left-radius: 0px;border-left-style: dotted;");
+
+ sheet.add(".month_default_event_bar", "top: 0px;bottom: 0px;left: 0px;width: 6px;");
+ sheet.add(".month_default_event_bar_inner", "position: absolute;width: 6px;background-color: #1066a8;");
+ sheet.add(".month_default_event_continueleft .month_default_event_bar", "display: none;");
+
+ sheet.add(".month_default_selected .month_default_event_inner", "background: #ddd;");
+ // sheet.add(".month_default_shadow_inner", "background-color: #666666;opacity: 0.5;height: 100%;");
+ sheet.add(".month_default_event_delete", "background: url() center center no-repeat; opacity: 0.6; cursor: pointer;");
+ sheet.add(".month_default_event_delete:hover", "opacity: 1;-ms-filter: none;");
+ sheet.add(".month_default_event_timeleft", "color: #ccc; font-size: 8pt");
+ sheet.add(".month_default_event_timeright", "color: #ccc; font-size: 8pt; text-align: right;");
+ sheet.add(".month_default_loading", "background-color: orange; color: white; padding: 2px;");
+ sheet.add(".month_default_shadow_inner", "box-sizing: border-box; background-color: #bbbbbb;border: 1px solid #888888;opacity: 0.5;height: 100%;");
+ sheet.add(".month_default_shadow", "box-shadow: 0 2px 5px rgba(0,0,0,.2);");
+
+ sheet.commit();
+
+ DayPilot.Global.defaultCss = true;
+ })();
+
+ // document element
+ DayPilot.doc = function() {
+ var de = document.documentElement;
+ return (de && de.clientHeight) ? de : document.body;
+ };
+
+ DayPilot.guid = function() {
+ var S4 = function() {
+ return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
+ };
+ return ("" + S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4());
+ };
+
+ DayPilot.pageOffset = function() {
+ if (typeof pageXOffset !== 'undefined') {
+ return { x: pageXOffset, y: pageYOffset };
+ }
+ var d = DayPilot.doc();
+ return { x: d.scrollLeft, y: d.scrollTop };
+ };
+
+ DayPilot.indexOf = function(array, object) {
+ if (!array || !array.length) {
+ return -1;
+ }
+ for (var i = 0; i < array.length; i++) {
+ if (array[i] === object) {
+ return i;
+ }
+ }
+ return -1;
+ };
+
+ // all children
+ DayPilot.ac = function(e, children) {
+ if (!children) {
+ var children = [];
+ }
+ for (var i = 0; e.children && i < e.children.length; i++) {
+ children.push(e.children[i]);
+ DayPilot.ac(e.children[i], children);
+ }
+
+ return children;
+ };
+
+ // remove from array
+ DayPilot.rfa = function(array, object) {
+ var i = DayPilot.indexOf(array, object);
+ if (i === -1) {
+ return;
+ }
+ array.splice(i, 1);
+ };
+
+ // mouse coords
+ DayPilot.mc = function(ev){
+ if(ev.pageX || ev.pageY){
+ return {x:ev.pageX, y:ev.pageY};
+ }
+ return {
+ x:ev.clientX + document.documentElement.scrollLeft,
+ y:ev.clientY + document.documentElement.scrollTop
+ };
+ };
+
+ DayPilot.Stats = {};
+ DayPilot.Stats.eventObjects = 0;
+ DayPilot.Stats.dateObjects = 0;
+ DayPilot.Stats.cacheHitsCtor = 0;
+ DayPilot.Stats.cacheHitsParsing = 0;
+ DayPilot.Stats.cacheHitsTicks = 0;
+
+ DayPilot.re = function(el, ev, func) {
+ if (!func) {
+ return;
+ }
+ if (!ev) {
+ return;
+ }
+ if (!el) {
+ return;
+ }
+ el.addEventListener(ev, func, false);
+ };
+
+ DayPilot.rePassive = function(el, ev, func) {
+ if (!func) {
+ return;
+ }
+ if (!ev) {
+ return;
+ }
+ if (!el) {
+ return;
+ }
+ el.addEventListener(ev, func, {"passive": true});
+ };
+
+ DayPilot.reNonPassive = function(el, ev, func) {
+ if (!func) {
+ return;
+ }
+ if (!ev) {
+ return;
+ }
+ if (!el) {
+ return;
+ }
+ el.addEventListener(ev, func, {"passive": false});
+ };
+
+ // purge
+ // thanks to http://javascript.crockford.com/memory/leak.html
+ DayPilot.pu = function(d) {
+ //var removed = [];
+ //var start = new Date();
+ var a = d.attributes, i, l, n;
+ if (a) {
+ l = a.length;
+ for (i = 0; i < l; i += 1) {
+ if (!a[i]) {
+ continue;
+ }
+ n = a[i].name;
+ if (typeof d[n] === 'function') {
+ d[n] = null;
+ }
+ }
+ }
+ a = d.childNodes;
+ if (a) {
+ l = a.length;
+ for (i = 0; i < l; i += 1) {
+ var children = DayPilot.pu(d.childNodes[i]);
+ }
+ }
+ };
+
+ // delete element
+ DayPilot.de = function(e) {
+ if (!e) {
+ return;
+ }
+ if (DayPilot.isArray(e)) {
+ for (var i = 0; i < e.length; i++) {
+ DayPilot.de(e[i]);
+ }
+ return;
+ }
+ e.parentNode && e.parentNode.removeChild(e);
+ };
+
+ // vertical scrollbar width
+ DayPilot.sw = function(element) {
+ if (!element) {
+ return 0;
+ }
+ return element.offsetWidth - element.clientWidth;
+ };
+
+ // angular module
+ DayPilot.am = function() {
+ if (typeof angular === "undefined") {
+ return null;
+ }
+ if (!DayPilot.am.cached) {
+ DayPilot.am.cached = angular.module("daypilot", []);
+ }
+ return DayPilot.am.cached;
+ };
+
+ DayPilot.Selection = function (start, end, resource, root) {
+ this.type = 'selection';
+ this.start = start.isDayPilotDate ? start : new DayPilot.Date(start);
+ this.end = end.isDayPilotDate ? end : new DayPilot.Date(end);
+ this.resource = resource;
+ this.root = root;
+
+ this.toJSON = function(key) {
+ var json = {};
+ json.start = this.start;
+ json.end = this.end;
+ json.resource = this.resource;
+
+ return json;
+ };
+ };
+
+ /* XMLHttpRequest */
+
+ DayPilot.request = function(url, callback, postData, errorCallback) {
+ var req = DayPilot.createXmlHttp();
+ if (!req) {
+ return;
+ }
+
+ req.open("POST", url, true);
+ req.setRequestHeader('Content-type', 'text/plain');
+ req.onreadystatechange = function() {
+ if (req.readyState !== 4)
+ return;
+ if (req.status !== 200 && req.status !== 304) {
+ if (errorCallback) {
+ errorCallback(req);
+ }
+ else {
+ if (window.console) { console.log('HTTP error ' + req.status); }
+ }
+ return;
+ }
+ callback(req);
+ };
+ if (req.readyState === 4) {
+ return;
+ }
+ if (typeof postData === 'object') {
+ postData = JSON.stringify(postData);
+ }
+ req.send(postData);
+ };
+
+ DayPilot.ajax = function(params) {
+ if (!params) {
+ throw new DayPilot.Exception("Parameter object required.");
+ }
+
+ if (typeof params.url !== "string") {
+ throw new DayPilot.Exception("The parameter object must have 'url' property.")
+ }
+
+ var req = DayPilot.createXmlHttp();
+ if (!req) {
+ throw new DayPilot.Exception("Unable to create XMLHttpRequest object");
+ }
+
+ var dataIsObject = typeof params.data === "object";
+
+ var data = params.data;
+ var method = params.method || (params.data ? "POST" : "GET");
+ var success = params.success || function() {};
+ var error = params.error || function() {};
+ var url = params.url;
+ var contentType = params.contentType || (dataIsObject ? "application/json" : "text/plain");
+ var headers = params.headers || {};
+
+ req.open(method, url, true);
+ req.setRequestHeader('Content-type', contentType);
+
+ // overriding the content-type is allowed
+ DayPilot.Util.ownPropsAsArray(headers).forEach(function(item) {
+ req.setRequestHeader(item.key, item.val);
+ });
+
+ req.onreadystatechange = function() {
+ if (req.readyState !== 4) {
+ return;
+ }
+ if (req.status !== 200 && req.status !== 201 && req.status !== 204 && req.status !== 304) {
+ if (error) {
+ var args = {};
+ args.request = req;
+ error(args);
+ }
+ else {
+ if (window.console) { console.log('HTTP error ' + req.status); }
+ }
+ return;
+ }
+ var args = {};
+ args.request = req;
+ if (req.responseText) {
+ args.data = JSON.parse(req.responseText);
+ }
+ success(args);
+ };
+ if (req.readyState === 4) {
+ return;
+ }
+ if (dataIsObject) {
+ data = JSON.stringify(data);
+ }
+ req.send(data);
+ };
+
+ DayPilot.createXmlHttp = function() {
+ return new XMLHttpRequest();
+ };
+
+ DayPilot.Http = {};
+
+ DayPilot.Http.ajax = function(params) {
+ DayPilot.ajax(params);
+ };
+
+ /*
+allowed params:
+- headers
+- contentType
+ */
+ DayPilot.Http.get = function(url, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "GET";
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Http.post = function(url, data, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "POST";
+ aparams.data = data;
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Http.put = function(url, data, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "PUT";
+ aparams.data = data;
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Http.delete = function(url, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "DELETE";
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Util = {};
+ DayPilot.Util.addClass = function(object, name) {
+ if (!object) {
+ return;
+ }
+ if (!object.className) {
+ object.className = name;
+ return;
+ }
+ var already = new RegExp("(^|\\s)" + name + "($|\\s)");
+ if (!already.test(object.className)) {
+ object.className = object.className + ' ' + name;
+ }
+ };
+
+ DayPilot.Util.removeClass = function(object, name) {
+ if (!object) {
+ return;
+ }
+ var already = new RegExp("(^|\\s)" + name + "($|\\s)");
+ object.className = object.className.replace(already, ' ').replace(/^\s\s*/, '').replace(/\s\s*$/, ''); // trim spaces
+ };
+
+ DayPilot.Util.copyProps = function(source, target, props) {
+ if (!target) {
+ target = {};
+ }
+ if (!source) {
+ return target;
+ }
+ if (typeof props === 'undefined') {
+ for (var name in source) {
+ if (source.hasOwnProperty(name) && typeof source[name] !== 'undefined') {
+ target[name] = source[name];
+ }
+ }
+ }
+ else {
+ for (var i = 0; i < props.length; i++) {
+ var name = props[i];
+ if (typeof source[name] !== 'undefined') {
+ target[name] = source[name];
+ }
+ }
+ }
+ return target;
+ };
+
+ DayPilot.Util.ownPropsAsArray = function(object) {
+ var result = [];
+
+ if (!object) {
+ return result;
+ }
+
+ for (var name in object) {
+ if (object.hasOwnProperty(name)) {
+ var item = {};
+ item.key = name;
+ item.val = object[name];
+ result.push(item);
+ }
+ }
+
+ return result;
+ };
+
+ DayPilot.Util.replaceCharAt = function(str, index, character) {
+ return str.substr(0, index) + character + str.substr(index + character.length);
+ };
+
+ DayPilot.Util.isNullOrUndefined = function(val) {
+ return val === null || typeof val === "undefined";
+ };
+
+ DayPilot.Util.escapeHtml = function(html) {
+ var div = document.createElement("div");
+ div.innerText = html;
+ return div.innerHTML;
+ };
+
+ DayPilot.Util.escapeTextHtml = function(text, html) {
+ if (!DayPilot.Util.isNullOrUndefined(html)) {
+ return html;
+ }
+ if (DayPilot.Util.isNullOrUndefined(text)) {
+ return "";
+ }
+ return DayPilot.Util.escapeHtml(text);
+ };
+
+ /* areas */
+
+ DayPilot.Areas = {};
+
+ /**
+ * Attach active areas to a target div.
+ * Works with "Visible", "TouchVisible" and "Hover" visibility
+ */
+ DayPilot.Areas.attach = function (div, e, options) {
+ var options = options || {};
+ var areas = options.areas;
+ var allowed = options.allowed || function() { return true; };
+ var offsetX = options.offsetX || 0;
+
+ // permanently visible active areas
+ areas = areasExtract(e, areas);
+
+ if (!areas) {
+ return;
+ }
+
+ if (!DayPilot.isArray(areas)) {
+ return;
+ }
+
+ if (areas.length === 0) {
+ return;
+ }
+
+ DayPilot.re(div, "mousemove", function(ev) {
+ if (!div.active && !div.areasDisabled && allowed()) {
+ DayPilot.Areas.showAreas(div, e, ev, areas, { "offsetX": offsetX, "eventDiv": options.eventDiv});
+ }
+ });
+ DayPilot.re(div, "mouseleave", function(ev) {
+ DayPilot.Areas.hideAreas(div, ev);
+ });
+
+ areas.forEach(function(area) {
+ if (!DayPilot.Areas.isVisible(area)) {
+ return;
+ }
+ var a = DayPilot.Areas.createArea(div, e, area, { "offsetX": offsetX, "eventDiv": options.eventDiv});
+ div.appendChild(a);
+ });
+ };
+
+ DayPilot.Areas.disable = function(div) {
+ div.areasDisabled = true;
+ Array.from(div.childNodes).filter(function(item) { return item.isActiveArea && !item.area.start; }).forEach(function(item) {
+ item._originalDisplay = item.style.display;
+ item.style.display = "none";
+ });
+ };
+
+ DayPilot.Areas.enable = function(div) {
+ div.areasDisabled = false;
+ Array.from(div.childNodes).filter(function(item) { return item.isActiveArea && !item.area.start; }).forEach(function(item) {
+ if (item._originalDisplay) {
+ item.style.display = item._originalDisplay;
+ }
+ else {
+ item.style.display = "";
+ }
+ });
+ };
+
+ DayPilot.Areas.remove = function(div) {
+ var divs = Array.from(div.childNodes).filter(function(item) { return item.isActiveArea; });
+
+ DayPilot.de(divs);
+ };
+
+ DayPilot.Areas.isVisible = function(area) {
+ var v = area.visibility || area.v || "Visible";
+ if (v === "Visible") {
+ return true;
+ }
+ if (v === "TouchVisible") {
+ if (!DayPilot.browser.hover) {
+ return true;
+ }
+ }
+ return false;
+ };
+
+ DayPilot.Areas.copy = function(areas) {
+ if (!DayPilot.isArray(areas)) {
+ return [];
+ }
+
+ return areas.map(function(area) {
+ return DayPilot.Util.copyProps(area, {});
+ });
+ };
+
+ /**
+ * Extracts areas array from the source object, giving priority to a standalone areas object.
+ * @param e
+ * @param areas
+ */
+ var areasExtract = function(e, areas) {
+ if (!DayPilot.isArray(areas)) {
+ areas = e.areas;
+ if (!areas) {
+ if (e.cache) {
+ areas = e.cache.areas;
+ }
+ else if (e.data) {
+ areas = e.data.areas;
+ }
+ }
+ }
+ return areas;
+ };
+
+
+ DayPilot.Areas.showAreas = function(div, e, ev, areas, options) {
+ if (DayPilot.Global.resizing) {
+ return;
+ }
+
+ if (DayPilot.Global.moving) {
+ return;
+ }
+
+ if (DayPilot.Global.selecting) {
+ return;
+ }
+
+ if (div.active) {
+ return;
+ }
+
+ if (!DayPilot.browser.hover) {
+ return;
+ }
+
+ if (DayPilot.Areas.all && DayPilot.Areas.all.length > 0) {
+ for (var i = 0; i < DayPilot.Areas.all.length; i++) {
+ var d = DayPilot.Areas.all[i];
+ if (d !== div) {
+ DayPilot.Areas.hideAreas(d, ev);
+ }
+ }
+ }
+
+ div.active = {};
+
+ //var areas;
+ if (!DayPilot.isArray(areas)) {
+ areas = e.areas;
+ if (!areas) {
+ if (e.cache) {
+ areas = e.cache.areas;
+ }
+ else if (e.data) {
+ areas = e.data.areas;
+ }
+ }
+ }
+
+ if (!areas || areas.length === 0) {
+ return;
+ }
+
+ if (div.areas && div.areas.length > 0) {
+ return;
+ }
+ div.areas = [];
+
+ for (var i = 0; i < areas.length; i++) {
+ var area = areas[i];
+
+ if (DayPilot.Areas.isVisible(area)) {
+ continue;
+ }
+
+ var a = DayPilot.Areas.createArea(div, e, area, options);
+
+ div.areas.push(a);
+ div.appendChild(a);
+
+ DayPilot.Areas.all.push(div);
+ }
+ div.active.children = DayPilot.ac(div);
+ };
+
+ /**
+ *
+ * @param div Target div element
+ * @param e Source data object
+ * @param area Area definition
+ * @returns {Element}
+ */
+ DayPilot.Areas.createArea = function(div, e, area, options) {
+
+ /*
+ actions:
+ - Default (equals not specified)
+ - None (cancels bubbling to parent element)
+ - ContextMenu
+ - HoverMenu
+ - ResizeStart
+ - ResizeEnd
+ - Move
+ - excluded: JavaScript, Bubble
+ */
+
+ var options = options || {};
+ var offsetX = options.offsetX || 0;
+ var ediv = options.eventDiv || div;
+
+ var a = document.createElement("div");
+ a.isActiveArea = true;
+ a.area = area;
+ a.setAttribute("unselectable", "on");
+ var w = area.w || area.width;
+ var h = area.h || area.height;
+ var css = area.cssClass || area.css || area.className;
+ if (typeof area.style !== "undefined") {
+ a.setAttribute("style", area.style);
+ }
+ a.style.position = "absolute";
+
+ a.style.width = resolvePosVal(w);
+ a.style.height = resolvePosVal(h);
+ a.style.right = resolvePosVal(area.right);
+ a.style.top = resolvePosVal(area.top);
+ a.style.left = resolvePosVal(area.left);
+ a.style.bottom = resolvePosVal(area.bottom);
+
+ if (typeof area.html !== 'undefined' || typeof area.text !== "undefined") {
+ // a.innerHTML = area.html || area.text;
+ a.innerHTML = DayPilot.Util.escapeTextHtml(area.text, area.html);
+ }
+ else if (area.icon) {
+ var iel = document.createElement("i");
+ iel.className = area.icon;
+ a.appendChild(iel);
+ }
+ else if (area.image) {
+ var img = document.createElement("img");
+ img.src = area.image;
+ a.appendChild(img);
+ }
+ else if (area.symbol) {
+ var ns = "http://www.w3.org/2000/svg";
+ var svg = document.createElementNS(ns,"svg");
+ svg.setAttribute("width", "100%");
+ svg.setAttribute("height", "100%");
+ var use = document.createElementNS(ns,"use");
+ use.setAttribute("href", area.symbol);
+ svg.appendChild(use);
+ a.appendChild(svg);
+ }
+
+ if (css) {
+ a.className = css;
+ }
+ if (area.toolTip) {
+ a.setAttribute("title", area.toolTip);
+ }
+ if (area.backColor) {
+ a.style.background = area.backColor;
+ }
+ if (area.background) { // alias
+ a.style.background = area.background;
+ }
+ if (area.fontColor) {
+ a.style.color = area.fontColor;
+ }
+ if (area.padding) {
+ a.style.padding = area.padding + "px";
+ a.style.boxSizing = "border-box";
+ }
+ if (area.verticalAlignment) {
+ a.style.display = "flex";
+ switch (area.verticalAlignment) {
+ case "center":
+ a.style.alignItems = "center";
+ break;
+ case "top":
+ a.style.alignItems = "flex-start";
+ break;
+ case "bottom":
+ a.style.alignItems = "flex-end";
+ break;
+ }
+ }
+ if (area.horizontalAlignment) {
+ a.style.display = "flex";
+ switch (area.horizontalAlignment) {
+ case "right":
+ a.style.justifyContent = "flex-end";
+ break;
+ case "left":
+ a.style.justifyContent = "flex-start";
+ break;
+ case "center":
+ a.style.justifyContent = "center";
+ break;
+ }
+ }
+
+ if (area.action === "ResizeEnd" || area.action === "ResizeStart" || area.action === "Move") {
+ if (e.calendar.isCalendar) {
+ switch (area.action) {
+ case "ResizeEnd":
+ area.cursor = "s-resize";
+ area.dpBorder = "bottom";
+ break;
+ case "ResizeStart":
+ area.cursor = "n-resize";
+ area.dpBorder = "top";
+ break;
+ case "Move":
+ area.cursor = "move";
+ break;
+ }
+ }
+ if (e.calendar.isScheduler || e.calendar.isMonth) {
+ switch (area.action) {
+ case "ResizeEnd":
+ area.cursor = "e-resize";
+ area.dpBorder = "right";
+ break;
+ case "ResizeStart":
+ area.cursor = "w-resize";
+ area.dpBorder = "left";
+ break;
+ case "Move":
+ area.cursor = "move";
+ break;
+ }
+ }
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+
+ if (e.calendar.internal && e.calendar.internal.dragInProgress && e.calendar.internal.dragInProgress()) { // resizing in progress
+ return;
+ }
+ ev.cancelBubble = true;
+
+ div.style.cursor = area.cursor;
+ if (area.dpBorder) {
+ div.dpBorder = area.dpBorder;
+ }
+ };
+ })(ediv, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ div.style.cursor = '';
+ };
+ })(ediv, e, area);
+ }
+ if ((area.action === "ResizeEnd" || area.action === "ResizeStart") && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ var touch = e.calendar.internal.touch;
+ var t = ev.touches ? ev.touches[0] : ev;
+ var coords = {x: t.pageX, y: t.pageY };
+ // immediately
+ e.calendar.coords = touch.relativeCoords(ev);
+ touch.preventEventTap = true;
+ if (e.calendar.isScheduler) {
+ touch.startResizing(div, area.action === "ResizeEnd" ? "right" : "left");
+ }
+ else if (e.calendar.isCalendar) {
+ touch.startResizing(div, area.action === "ResizeEnd" ? "bottom" : "top", coords);
+ }
+
+ };
+ })(ediv, e, area);
+ DayPilot.rePassive(a, DayPilot.touch.start, touchstart);
+ // a.addEventListener(DayPilot.touch.start, touchstart, {"passive": true});
+ }
+ }
+ if (area.action === "ContextMenu" && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+
+ showContextMenu(div, e, area, ev);
+ var touch = e.calendar.internal.touch;
+ touch.preventEventTap = true;
+ };
+ })(ediv, e, area);
+ var touchend = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+ };
+ })(ediv, e, area);
+ DayPilot.reNonPassive(a, DayPilot.touch.start, touchstart);
+ DayPilot.reNonPassive(a, DayPilot.touch.end, touchend);
+ }
+ }
+ if (area.action === "Bubble" && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+
+ var args = doOnClick(area, e, ev);
+ if (args.preventDefault.value) {
+ return;
+ }
+ showBubble(e, area, ev);
+ var touch = e.calendar.internal.touch;
+ touch.preventEventTap = true;
+
+ if (typeof area.onClicked === "function") {
+ area.onClicked(args);
+ }
+ };
+ })(ediv, e, area);
+ var touchend = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+ };
+ })(ediv, e, area);
+ DayPilot.reNonPassive(a, DayPilot.touch.start, touchstart);
+ DayPilot.reNonPassive(a, DayPilot.touch.end, touchend);
+ }
+ }
+ if (area.action === "Move" && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ var touch = e.calendar.internal.touch;
+ var t = ev.touches ? ev.touches[0] : ev;
+ var coords = {x: t.pageX, y: t.pageY };
+ // immediately
+ e.calendar.coords = touch.relativeCoords(ev);
+
+ // DayPilot.Global.movingAreaData = area.data;
+ if (DayPilot.Global && DayPilot.Global.touch) {
+ DayPilot.Global.touch.active = true;
+ }
+
+ touch.preventEventTap = true;
+ touch.startMoving(div, coords);
+ };
+ })(ediv, e, area);
+ DayPilot.rePassive(a, DayPilot.touch.start, touchstart);
+ // a.addEventListener(DayPilot.touch.start, touchstart, {"passive": true});
+ }
+ }
+ if (area.action === "Bubble" && e.isEvent) {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ if (area.bubble) {
+ area.bubble.showEvent(e, true);
+ }
+ else if (e.calendar.bubble) {
+ e.calendar.bubble.showEvent(e, true);
+ }
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ if (typeof DayPilot.Bubble !== "undefined") {
+ //DayPilot.Bubble.hideActive();
+ if (area.bubble) {
+ area.bubble.hideOnMouseOut();
+ }
+ else if (e.calendar.bubble) {
+ e.calendar.bubble.hideOnMouseOut();
+ }
+ }
+
+ };
+ })(div, e, area);
+ }
+ else if (area.action === "Bubble" && e.isRow) {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ if (area.bubble) {
+ area.bubble.showResource(e, true);
+ }
+ else if (e.calendar.resourceBubble) {
+ e.calendar.resourceBubble.showResource(e, true);
+ }
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ if (typeof DayPilot.Bubble !== "undefined") {
+ //DayPilot.Bubble.hideActive();
+ if (area.bubble) {
+ area.bubble.hideOnMouseOut();
+ }
+ else if (e.calendar.resourceBubble) {
+ e.calendar.resourceBubble.hideOnMouseOut();
+ }
+ }
+
+ };
+ })(div, e, area);
+ }
+ else if (area.action === "Bubble" && typeof DayPilot.Bubble !== "undefined" && area.bubble instanceof DayPilot.Bubble) {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ area.bubble.showHtml(null, null);
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ if (typeof DayPilot.Bubble !== "undefined") {
+ if (area.bubble) {
+ area.bubble.hideOnMouseOut();
+ }
+ }
+
+ };
+ })(div, e, area);
+ }
+ if (area.action === "HoverMenu") {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ var m = area.menu;
+ if (m && m.show) {
+ if (!m.visible) {
+ m.show(e);
+ }
+ else if (m.source && typeof m.source.id !== 'undefined' && m.source.id !== e.id) {
+ m.show(e);
+ }
+ m.cancelHideTimeout();
+ }
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ var m = area.menu;
+ if (!m) {
+ return;
+ }
+ if (m.hideOnMouseOver) {
+ m.delayedHide();
+ }
+ };
+ })(div, e, area);
+ }
+ if (area.action === "None") {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ var args = doOnClick(area, e, ev);
+
+ if (typeof area.onClicked === "function") {
+ area.onClicked(args);
+ }
+
+ ev.preventDefault();
+ ev.stopPropagation();
+ };
+ })(ediv, e, area);
+ DayPilot.reNonPassive(a, DayPilot.touch.start, touchstart);
+ }
+
+ // prevent event moving
+ a.onmousedown = (function(div, e, area) {
+ return function(ev) {
+ if (typeof area.onmousedown === 'function') { // obsolete, remove
+ area.onmousedown(ev);
+ }
+
+ if (typeof area.mousedown === 'function') { // internal
+ var args = {};
+ args.area = area;
+ args.div = div;
+ args.originalEvent = ev;
+ args.source = e;
+ area.mousedown(args);
+ }
+
+ if (area.action === "Move" && e.isRow) {
+ var row = e.$.row;
+ var startMoving = e.calendar.internal.rowStartMoving;
+
+ startMoving(row);
+ }
+
+ // cancel any bubble
+ if (typeof DayPilot.Bubble !== "undefined") {
+ DayPilot.Bubble.hideActive();
+ }
+
+ if (area.action === "Move") {
+ DayPilot.Global.movingAreaData = area.data;
+ }
+
+ if (area.action === "Move" && e.isEvent) {
+ if (e.calendar.internal && e.calendar.internal.startMoving) {
+ e.calendar.internal.startMoving(div, ev);
+ }
+ }
+
+ var cancel = true;
+
+ if (cancel) {
+ if (area.action === "Move" || area.action === "ResizeEnd" || area.action === "ResizeStart" || !area.action || area.action === "Default") {
+ return;
+ }
+ ev.preventDefault(); // prevents text selection on dragging
+ ev.cancelBubble = true;
+ }
+ };
+ })(div, e, area);
+ a.onclick = (function(div, e, area) {
+ return function(ev) {
+ var args = doOnClick(area, e, ev);
+
+ if (args.preventDefault.value) {
+ return;
+ }
+
+ switch (area.action) {
+ case "ContextMenu":
+ showContextMenu(div, e, area, ev);
+ ev.cancelBubble = true;
+ break;
+ case "None":
+ ev.cancelBubble = true;
+ break;
+ }
+
+ if (typeof area.onClicked === "function") {
+ area.onClicked(args);
+ }
+ };
+ })(div, e, area);
+
+ if (typeof area.onMouseEnter === "function") {
+ a.addEventListener("mouseenter", (function(div, e, area) {
+ return function(ev) {
+ var args = {};
+ args.area = area;
+ args.source = e;
+ args.originalEvent = ev;
+ area.onMouseEnter(args);
+ };
+ })(div, e, area));
+ }
+ if (typeof area.onMouseLeave === "function") {
+ a.addEventListener("mouseleave", (function(div, e, area) {
+ return function(ev) {
+ var args = {};
+ args.area = area;
+ args.source = e;
+ args.originalEvent = ev;
+ area.onMouseLeave(args);
+ };
+ })(div, e, area));
+ }
+
+ function doOnClick(area, source, originalEvent) {
+ var args = {};
+ args.area = area;
+ args.source = source;
+ args.originalEvent = originalEvent;
+ args.preventDefault = function() {
+ args.preventDefault.value = true;
+ };
+
+ if (typeof area.onClick === "function") {
+ area.onClick(args);
+ }
+ return args;
+ }
+
+ function showBubble(e, area, ev) {
+ if (DayPilot.Bubble) {
+ DayPilot.Bubble.touchPosition(ev);
+ }
+
+ if (e.calendar.bubble) {
+ e.calendar.bubble.showEvent(e, true);
+ }
+ }
+
+ function showContextMenu(div, e, area, ev) {
+ if (DayPilot.Menu) {
+ DayPilot.Menu.touchPosition(ev);
+ }
+
+ var m = area.contextMenu || area.menu;
+ if (!(m instanceof DayPilot.Menu)) {
+ if (e.isEvent && e.client.contextMenu()) {
+ m = e.client.contextMenu();
+ }
+ else if (e.isEvent && e.calendar.contextMenu) {
+ m = e.calendar.contextMenu;
+ }
+ }
+ if (m && m.show) {
+ var initiator = { "type": "area", "div": div, "e": e, "area": area, "a": a};
+ m.show(e, { "initiator": initiator});
+ }
+ }
+
+ function resolvePosVal(val) {
+ if (typeof val === "string" && isNaN(val)) {
+ return val;
+ }
+ else if (typeof val !== "undefined") {
+ return val + "px";
+ }
+ return undefined;
+ }
+
+ return a;
+ };
+
+ DayPilot.Areas.all = [];
+
+ DayPilot.Areas.hideAreas = function(div, ev) {
+ if (!div) {
+ return;
+ }
+
+ if (!div || !div.active) {
+ return;
+ }
+
+ var active = div.active;
+ var areas = div.areas;
+
+ if (active && active.children) {
+ if (ev) {
+ var target = ev.toElement || ev.relatedTarget;
+ if (~DayPilot.indexOf(active.children, target)) {
+ return;
+ }
+ }
+ }
+
+ if (!areas || areas.length === 0) {
+ div.active = null;
+ return;
+ }
+
+ DayPilot.de(areas);
+
+ div.active = null;
+ div.areas = [];
+
+ DayPilot.rfa(DayPilot.Areas.all, div);
+
+ active.children = null;
+ };
+
+ DayPilot.Areas.hideAll = function(ev) {
+ if (!DayPilot.Areas.all || DayPilot.Areas.all.length === 0) {
+ return;
+ }
+ for (var i = 0; i < DayPilot.Areas.all.length; i++) {
+ DayPilot.Areas.hideAreas(DayPilot.Areas.all[i], ev);
+ }
+
+ };
+
+ /* end of areas */
+
+ DayPilot.Exception = function(msg) {
+ return new Error(msg);
+ };
+
+ DayPilot.Locale = function(id, config) {
+ this.id = id;
+ this.dayNames = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
+ this.dayNamesShort = ["Su", "Mo", "Tu", "We", "Th", "Fr", "Sa"];
+ this.monthNames = ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"];
+ this.datePattern = "M/d/yyyy";
+ this.timePattern = "H:mm";
+ this.dateTimePattern = "M/d/yyyy H:mm";
+ this.timeFormat = "Clock12Hours";
+ this.weekStarts = 0;
+
+ if (config) {
+ for (var name in config) {
+ this[name] = config[name];
+ }
+ }
+ };
+
+ DayPilot.Locale.all = {};
+
+ DayPilot.Locale.find = function(id) {
+ if (!id) {
+ return null;
+ }
+ var normalized = id.toLowerCase();
+ if (normalized.length > 2) {
+ normalized = DayPilot.Util.replaceCharAt(normalized, 2, '-');
+ }
+ return DayPilot.Locale.all[normalized];
+ };
+
+ DayPilot.Locale.register = function(locale) {
+ DayPilot.Locale.all[locale.id] = locale;
+ };
+
+ DayPilot.Locale.register(new DayPilot.Locale('ca-es', {'dayNames':['diumenge','dilluns','dimarts','dimecres','dijous','divendres','dissabte'],'dayNamesShort':['dg','dl','dt','dc','dj','dv','ds'],'monthNames':['gener','febrer','març','abril','maig','juny','juliol','agost','setembre','octubre','novembre','desembre',''],'monthNamesShort':['gen.','febr.','març','abr.','maig','juny','jul.','ag.','set.','oct.','nov.','des.',''],'timePattern':'H:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('cs-cz', {'dayNames':['neděle','pondělí','úterý','středa','čtvrtek','pátek','sobota'],'dayNamesShort':['ne','po','út','st','čt','pá','so'],'monthNames':['leden','únor','březen','duben','květen','červen','červenec','srpen','září','říjen','listopad','prosinec',''],'monthNamesShort':['I','II','III','IV','V','VI','VII','VIII','IX','X','XI','XII',''],'timePattern':'H:mm','datePattern':'d. M. yyyy','dateTimePattern':'d. M. yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('da-dk', {'dayNames':['søndag','mandag','tirsdag','onsdag','torsdag','fredag','lørdag'],'dayNamesShort':['sø','ma','ti','on','to','fr','lø'],'monthNames':['januar','februar','marts','april','maj','juni','juli','august','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mar','apr','maj','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'HH:mm','datePattern':'dd-MM-yyyy','dateTimePattern':'dd-MM-yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-at', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Jänner','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jän','Feb','Mär','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-ch', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Januar','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jan','Feb','Mrz','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-de', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Januar','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jan','Feb','Mrz','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-lu', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Januar','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jan','Feb','Mrz','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-au', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'h:mm tt','datePattern':'d/MM/yyyy','dateTimePattern':'d/MM/yyyy h:mm tt','timeFormat':'Clock12Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-ca', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'h:mm tt','datePattern':'yyyy-MM-dd','dateTimePattern':'yyyy-MM-dd h:mm tt','timeFormat':'Clock12Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-gb', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-us', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'h:mm tt','datePattern':'M/d/yyyy','dateTimePattern':'M/d/yyyy h:mm tt','timeFormat':'Clock12Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('es-es', {'dayNames':['domingo','lunes','martes','miércoles','jueves','viernes','sábado'],'dayNamesShort':['D','L','M','X','J','V','S'],'monthNames':['enero','febrero','marzo','abril','mayo','junio','julio','agosto','septiembre','octubre','noviembre','diciembre',''],'monthNamesShort':['ene.','feb.','mar.','abr.','may.','jun.','jul.','ago.','sep.','oct.','nov.','dic.',''],'timePattern':'H:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('es-mx', {'dayNames':['domingo','lunes','martes','miércoles','jueves','viernes','sábado'],'dayNamesShort':['do.','lu.','ma.','mi.','ju.','vi.','sá.'],'monthNames':['enero','febrero','marzo','abril','mayo','junio','julio','agosto','septiembre','octubre','noviembre','diciembre',''],'monthNamesShort':['ene.','feb.','mar.','abr.','may.','jun.','jul.','ago.','sep.','oct.','nov.','dic.',''],'timePattern':'hh:mm tt','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy hh:mm tt','timeFormat':'Clock12Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('eu-es', {'dayNames':['igandea','astelehena','asteartea','asteazkena','osteguna','ostirala','larunbata'],'dayNamesShort':['ig','al','as','az','og','or','lr'],'monthNames':['urtarrila','otsaila','martxoa','apirila','maiatza','ekaina','uztaila','abuztua','iraila','urria','azaroa','abendua',''],'monthNamesShort':['urt.','ots.','mar.','api.','mai.','eka.','uzt.','abu.','ira.','urr.','aza.','abe.',''],'timePattern':'H:mm','datePattern':'yyyy/MM/dd','dateTimePattern':'yyyy/MM/dd H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fi-fi', {'dayNames':['sunnuntai','maanantai','tiistai','keskiviikko','torstai','perjantai','lauantai'],'dayNamesShort':['su','ma','ti','ke','to','pe','la'],'monthNames':['tammikuu','helmikuu','maaliskuu','huhtikuu','toukokuu','kesäkuu','heinäkuu','elokuu','syyskuu','lokakuu','marraskuu','joulukuu',''],'monthNamesShort':['tammi','helmi','maalis','huhti','touko','kesä','heinä','elo','syys','loka','marras','joulu',''],'timePattern':'H:mm','datePattern':'d.M.yyyy','dateTimePattern':'d.M.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-be', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd-MM-yy','dateTimePattern':'dd-MM-yy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-ch', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-fr', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-lu', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('gl-es', {'dayNames':['domingo','luns','martes','mércores','xoves','venres','sábado'],'dayNamesShort':['do','lu','ma','mé','xo','ve','sá'],'monthNames':['xaneiro','febreiro','marzo','abril','maio','xuño','xullo','agosto','setembro','outubro','novembro','decembro',''],'monthNamesShort':['xan','feb','mar','abr','maio','xuño','xul','ago','set','out','nov','dec',''],'timePattern':'H:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('it-it', {'dayNames':['domenica','lunedì','martedì','mercoledì','giovedì','venerdì','sabato'],'dayNamesShort':['do','lu','ma','me','gi','ve','sa'],'monthNames':['gennaio','febbraio','marzo','aprile','maggio','giugno','luglio','agosto','settembre','ottobre','novembre','dicembre',''],'monthNamesShort':['gen','feb','mar','apr','mag','giu','lug','ago','set','ott','nov','dic',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('it-ch', {'dayNames':['domenica','lunedì','martedì','mercoledì','giovedì','venerdì','sabato'],'dayNamesShort':['do','lu','ma','me','gi','ve','sa'],'monthNames':['gennaio','febbraio','marzo','aprile','maggio','giugno','luglio','agosto','settembre','ottobre','novembre','dicembre',''],'monthNamesShort':['gen','feb','mar','apr','mag','giu','lug','ago','set','ott','nov','dic',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('ja-jp', {'dayNames':['日曜日','月曜日','火曜日','水曜日','木曜日','金曜日','土曜日'],'dayNamesShort':['日','月','火','水','木','金','土'],'monthNames':['1月','2月','3月','4月','5月','6月','7月','8月','9月','10月','11月','12月',''],'monthNamesShort':['1','2','3','4','5','6','7','8','9','10','11','12',''],'timePattern':'H:mm','datePattern':'yyyy/MM/dd','dateTimePattern':'yyyy/MM/dd H:mm','timeFormat':'Clock24Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('nb-no', {'dayNames':['søndag','mandag','tirsdag','onsdag','torsdag','fredag','lørdag'],'dayNamesShort':['sø','ma','ti','on','to','fr','lø'],'monthNames':['januar','februar','mars','april','mai','juni','juli','august','september','oktober','november','desember',''],'monthNamesShort':['jan','feb','mar','apr','mai','jun','jul','aug','sep','okt','nov','des',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('nl-nl', {'dayNames':['zondag','maandag','dinsdag','woensdag','donderdag','vrijdag','zaterdag'],'dayNamesShort':['zo','ma','di','wo','do','vr','za'],'monthNames':['januari','februari','maart','april','mei','juni','juli','augustus','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mrt','apr','mei','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'HH:mm','datePattern':'d-M-yyyy','dateTimePattern':'d-M-yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('nl-be', {'dayNames':['zondag','maandag','dinsdag','woensdag','donderdag','vrijdag','zaterdag'],'dayNamesShort':['zo','ma','di','wo','do','vr','za'],'monthNames':['januari','februari','maart','april','mei','juni','juli','augustus','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mrt','apr','mei','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'H:mm','datePattern':'d/MM/yyyy','dateTimePattern':'d/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('nn-no', {'dayNames':['søndag','måndag','tysdag','onsdag','torsdag','fredag','laurdag'],'dayNamesShort':['sø','må','ty','on','to','fr','la'],'monthNames':['januar','februar','mars','april','mai','juni','juli','august','september','oktober','november','desember',''],'monthNamesShort':['jan','feb','mar','apr','mai','jun','jul','aug','sep','okt','nov','des',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('pt-br', {'dayNames':['domingo','segunda-feira','terça-feira','quarta-feira','quinta-feira','sexta-feira','sábado'],'dayNamesShort':['D','S','T','Q','Q','S','S'],'monthNames':['janeiro','fevereiro','março','abril','maio','junho','julho','agosto','setembro','outubro','novembro','dezembro',''],'monthNamesShort':['jan','fev','mar','abr','mai','jun','jul','ago','set','out','nov','dez',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('pl-pl', {'dayNames':['niedziela','poniedziałek','wtorek','środa','czwartek','piątek','sobota'],'dayNamesShort':['N','Pn','Wt','Śr','Cz','Pt','So'],'monthNames':['styczeń','luty','marzec','kwiecień','maj','czerwiec','lipiec','sierpień','wrzesień','październik','listopad','grudzień',''],'monthNamesShort':['sty','lut','mar','kwi','maj','cze','lip','sie','wrz','paź','lis','gru',''],'timePattern':'HH:mm','datePattern':'yyyy-MM-dd','dateTimePattern':'yyyy-MM-dd HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('pt-pt', {'dayNames':['domingo','segunda-feira','terça-feira','quarta-feira','quinta-feira','sexta-feira','sábado'],'dayNamesShort':['D','S','T','Q','Q','S','S'],'monthNames':['janeiro','fevereiro','março','abril','maio','junho','julho','agosto','setembro','outubro','novembro','dezembro',''],'monthNamesShort':['jan','fev','mar','abr','mai','jun','jul','ago','set','out','nov','dez',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('ro-ro', {'dayNames':['duminică','luni','marți','miercuri','joi','vineri','sâmbătă'],'dayNamesShort':['D','L','Ma','Mi','J','V','S'],'monthNames':['ianuarie','februarie','martie','aprilie','mai','iunie','iulie','august','septembrie','octombrie','noiembrie','decembrie',''],'monthNamesShort':['ian.','feb.','mar.','apr.','mai.','iun.','iul.','aug.','sep.','oct.','nov.','dec.',''],'timePattern':'H:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('ru-ru', {'dayNames':['воскресенье','понедельник','вторник','среда','четверг','пятница','суббота'],'dayNamesShort':['Вс','Пн','Вт','Ср','Чт','Пт','Сб'],'monthNames':['Январь','Февраль','Март','Апрель','Май','Июнь','Июль','Август','Сентябрь','Октябрь','Ноябрь','Декабрь',''],'monthNamesShort':['янв','фев','мар','апр','май','июн','июл','авг','сен','окт','ноя','дек',''],'timePattern':'H:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('sk-sk', {'dayNames':['nedeľa','pondelok','utorok','streda','štvrtok','piatok','sobota'],'dayNamesShort':['ne','po','ut','st','št','pi','so'],'monthNames':['január','február','marec','apríl','máj','jún','júl','august','september','október','november','december',''],'monthNamesShort':['1','2','3','4','5','6','7','8','9','10','11','12',''],'timePattern':'H:mm','datePattern':'d.M.yyyy','dateTimePattern':'d.M.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('sv-se', {'dayNames':['söndag','måndag','tisdag','onsdag','torsdag','fredag','lördag'],'dayNamesShort':['sö','må','ti','on','to','fr','lö'],'monthNames':['januari','februari','mars','april','maj','juni','juli','augusti','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mar','apr','maj','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'HH:mm','datePattern':'yyyy-MM-dd','dateTimePattern':'yyyy-MM-dd HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('tr-tr', {'dayNames':['Pazar','Pazartesi','Salı','Çarşamba','Perşembe','Cuma','Cumartesi'],'dayNamesShort':['Pz','Pt','Sa','Ça','Pe','Cu','Ct'],'monthNames':['Ocak','Şubat','Mart','Nisan','Mayıs','Haziran','Temmuz','Ağustos','Eylül','Ekim','Kasım','Aralık',''],'monthNamesShort':['Oca','Şub','Mar','Nis','May','Haz','Tem','Ağu','Eyl','Eki','Kas','Ara',''],'timePattern':'HH:mm','datePattern':'d.M.yyyy','dateTimePattern':'d.M.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('uk-ua', {'dayNames':['неділя','понеділок','вівторок','середа','четвер',"п'ятниця",'субота'],'dayNamesShort':['Нд','Пн','Вт','Ср','Чт','Пт','Сб'],'monthNames':['січень','лютий','березень','квітень','травень','червень','липень','серпень','вересень','жовтень','листопад','грудень',''],'monthNamesShort':['Січ','Лют','Бер','Кві','Тра','Чер','Лип','Сер','Вер','Жов','Лис','Гру',''],'timePattern':'H:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('zh-cn', {'dayNames':['星期日','星期一','星期二','星期三','星期四','星期五','星期六'],'dayNamesShort':['日','一','二','三','四','五','六'],'monthNames':['一月','二月','三月','四月','五月','六月','七月','八月','九月','十月','十一月','十二月',''],'monthNamesShort':['1月','2月','3月','4月','5月','6月','7月','8月','9月','10月','11月','12月',''],'timePattern':'H:mm','datePattern':'yyyy/M/d','dateTimePattern':'yyyy/M/d H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('zh-tw', {'dayNames':['星期日','星期一','星期二','星期三','星期四','星期五','星期六'],'dayNamesShort':['日','一','二','三','四','五','六'],'monthNames':['一月','二月','三月','四月','五月','六月','七月','八月','九月','十月','十一月','十二月',''],'monthNamesShort':['一月','二月','三月','四月','五月','六月','七月','八月','九月','十月','十一月','十二月',''],'timePattern':'tt hh:mm','datePattern':'yyyy/M/d','dateTimePattern':'yyyy/M/d tt hh:mm','timeFormat':'Clock12Hours','weekStarts':0}));
+
+ DayPilot.Locale.US = DayPilot.Locale.find("en-us");
+
+ /**
+ * Created on 2023-04-04.
+ */
+
+ DayPilot.Switcher = function(options) {
+
+ var This = this;
+
+ this._views = [];
+ this._triggers = [];
+ this._navigator = {};
+
+ this.selectedClass = null;
+
+ this._active = null;
+
+ this._day = DayPilot.Date.today();
+
+ this.onChange = null;
+ this.onChanged = null;
+ this.onSelect = null;
+
+ this._navigator.updateMode = function (mode) {
+ var control = This._navigator.control;
+ if (!control) {
+ return;
+ }
+ control.selectMode = mode;
+ control.select(This._day);
+ };
+
+ this.addView = function (spec, options) {
+ var element;
+ if (typeof spec === 'string') {
+ element = document.getElementById(spec);
+ if (!element) {
+ throw "Element not found: " + spec;
+ }
+ }
+ else { // DayPilot object, DOM element
+ element = spec;
+ }
+
+ var control = element;
+
+ var view = {};
+ view._isView = true;
+ view._id = control.id;
+ view.control = control;
+ view._options = options || {};
+ view._hide = function () {
+ if (control.hide) {
+ control.hide();
+ }
+ else if (control.nav && control.nav.top) {
+ control.nav.top.style.display = 'none';
+ }
+ else {
+ control.style.display = 'none';
+ }
+ };
+ view._sendNavigate = function(date) {
+ var serverBased = (function() {
+ if (control.backendUrl) { // ASP.NET MVC, Java
+ return true;
+ }
+ if (typeof WebForm_DoCallback === 'function' && control.uniqueID) { // ASP.NET WebForms
+ return true;
+ }
+ return false;
+ })();
+ if (serverBased) {
+ if (control.commandCallBack) {
+ control.commandCallBack("navigate", { "day": date });
+ }
+ }
+ else {
+ control.startDate = date;
+ control.update();
+ }
+ };
+ view._show = function () {
+ This._hideViews();
+ if (control.show) {
+ control.show();
+ }
+ else if (control.nav && control.nav.top) {
+ control.nav.top.style.display = '';
+ }
+ else {
+ control.style.display = '';
+ }
+ };
+ view._selectMode = function () { // for navigator
+ if (view._options.navigatorSelectMode) {
+ return view._options.navigatorSelectMode;
+ }
+
+ if (control.isCalendar) {
+ switch (control.viewType) {
+ case "Day":
+ return "day";
+ case "Week":
+ return "week";
+ case "WorkWeek":
+ return "week";
+ default:
+ return "day";
+ }
+ }
+ else if (control.isMonth) {
+ switch (control.viewType) {
+ case "Month":
+ return "month";
+ case "Weeks":
+ return "week";
+ default:
+ return "day";
+ }
+ }
+ return "day";
+ };
+
+ this._views.push(view);
+
+ return view;
+ };
+
+ this.addTrigger = function (id, control) {
+ var element;
+ if (typeof id === 'string') {
+ element = document.getElementById(id);
+ if (!element) {
+ throw "Element not found: " + id;
+ }
+ }
+ else {
+ element = id;
+ }
+
+ var view = this._findViewByControl(control);
+ if (!view) {
+ view = this.addView(control);
+ }
+
+ var trigger = {};
+ trigger._isTrigger = true;
+ trigger._element = element;
+ trigger._id = element.id;
+ trigger._view = view;
+ trigger._onClick = function (ev) {
+
+ This.show(trigger);
+ This._select(trigger);
+
+ if (ev) {
+ ev.preventDefault ? ev.preventDefault() : ev.returnValue = false;
+ }
+
+ };
+
+ DayPilot.re(element, 'click', trigger._onClick);
+
+ this._triggers.push(trigger);
+
+ return trigger;
+ };
+
+ // backwards compatibility
+ this.addButton = this.addTrigger;
+
+ this.select = function(id) {
+ var trigger = this._findTriggerById(id);
+ if (trigger) {
+ trigger._onClick();
+ }
+ else if (this._triggers.length > 0) {
+ this._triggers[0]._onClick();
+ }
+ };
+
+ this._findTriggerById = function(id) {
+ for (var i = 0; i < this._triggers.length; i++) {
+ var trigger = this._triggers[i];
+ if (trigger._id === id) {
+ return trigger;
+ }
+ }
+ return null;
+ };
+
+ this._select = function(trigger) {
+ if (!this.selectedClass) {
+ return;
+ }
+
+ for (var i = 0; i < this._triggers.length; i++) {
+ var s = this._triggers[i];
+ DayPilot.Util.removeClass(s._element, this.selectedClass);
+ }
+ DayPilot.Util.addClass(trigger._element, this.selectedClass);
+ };
+
+ this.addNavigator = function (control) {
+ //this.navigator = {};
+ This._navigator.control = control;
+
+ control.timeRangeSelectedHandling = "JavaScript";
+ control.onTimeRangeSelected = function() {
+ var start, end, day;
+ if (control.api === 1) {
+ start = arguments[0];
+ end = arguments[1];
+ day = arguments[2];
+ }
+ else {
+ var args = arguments[0];
+ start = args.start;
+ end = args.end;
+ day = args.day;
+ }
+ This._day = day;
+
+ navigate(start, end, day);
+
+ };
+ };
+
+ this.show = function (el) {
+ var view, trigger;
+ if (el._isTrigger) {
+ trigger = el;
+ view = trigger._view;
+ }
+ else {
+ view = el._isView ? el : this._findViewByControl(el);
+ if (this._active === view) {
+ return;
+ }
+ }
+
+ if (This.onSelect) {
+ var args = {};
+ args.source = trigger ? trigger._element : null;
+ args.target = view.control;
+
+ This.onSelect(args);
+ // TODO add preventDefault
+ }
+
+ this._active = view;
+ view._show();
+
+ var mode = view._selectMode();
+ This._navigator.updateMode(mode);
+
+ //This.navigator.select(This.day);
+
+ //This.active.sendNavigate(this.day);
+
+ // this ensures first onChange call if single day = today is displayed
+ var start = This._navigator.control.selectionStart;
+ var end = This._navigator.control.selectionEnd.addDays(1);
+ var day = This._navigator.control.selectionDay;
+ navigate(start, end, day);
+ };
+
+ this._findViewByControl = function (control) {
+ for (var i = 0; i < this._views.length; i++) {
+ if (this._views[i].control === control) {
+ return this._views[i];
+ }
+ }
+ return null;
+ };
+
+ this._hideViews = function () {
+ //var controls = [dp_day, dp_week, dp_month];
+ for (var i = 0; i < this._views.length; i++) {
+ this._views[i]._hide();
+ }
+ };
+
+ Object.defineProperty(this, "active", {
+ get: function() {
+ return This._active;
+ }
+ });
+
+ this.events = {};
+
+ this.events.load = function(url, success, error) {
+ if (This._active && This._active.control) {
+ This._active.control.events.load(url, success, error);
+ }
+ else {
+ throw "DayPilot.Switcher.events.load(): Active view not found";
+ }
+ };
+
+ this._previousArgs = null;
+
+ this._init = function() {
+ if (!options) {
+ return;
+ }
+
+ for (var name in options) {
+ if (name === "triggers") {
+ var triggers = options.triggers || [];
+ triggers.forEach(function(item) {
+ This.addTrigger(item.id, item.view);
+ });
+ }
+ else if (name === "navigator") {
+ This.addNavigator(options.navigator);
+ }
+ else {
+ This[name] = options[name];
+ }
+ }
+
+ };
+
+ this._init();
+
+ function navigate(start, end, day) {
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.day = day;
+ args.target = This._active.control;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ var previous = This._previousArgs;
+ if (previous) {
+ if (previous.start === args.start && previous.end === args.end && previous.day === args.day && previous.target === args.target) {
+ return; // duplicate, no change
+ }
+ }
+
+ This._previousArgs = args;
+
+ if (typeof This.onChange === "function") {
+ This.onChange(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // backwards compatibility
+ if (typeof This.onTimeRangeSelect === "function") {
+ This.onTimeRangeSelect(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ This._active._sendNavigate(This._day);
+
+ if (typeof This.onChanged === "function") {
+ This.onChanged(args);
+ }
+
+ if (typeof This.onTimeRangeSelected === "function") {
+ This.onTimeRangeSelected(args);
+ }
+
+ }
+
+ };
+
+ // DayPilot.Date START
+
+ DayPilot.Duration = function(ticks) {
+ var d = this;
+
+ var day = 1000*60*60*24.0;
+ var hour = 1000*60*60.0;
+ var minute = 1000*60.0;
+ var second = 1000.0;
+
+ if (arguments.length === 2) {
+ var start = arguments[0];
+ var end = arguments[1];
+
+ if (!(start instanceof DayPilot.Date) && (typeof start !== "string")) {
+ throw "DayPilot.Duration(): Invalid start argument, DayPilot.Date expected";
+ }
+ if (!(end instanceof DayPilot.Date) && (typeof end !== "string")) {
+ throw "DayPilot.Duration(): Invalid end argument, DayPilot.Date expected";
+ }
+ if (typeof start === "string") {
+ start = new DayPilot.Date(start);
+ }
+ if (typeof end === "string") {
+ end = new DayPilot.Date(end);
+ }
+ ticks = end.getTime() - start.getTime();
+ }
+
+ this.ticks = ticks;
+
+ // caching, allows direct comparison
+ if (DayPilot.Date.Cache.DurationCtor["" + ticks]) {
+ return DayPilot.Date.Cache.DurationCtor["" + ticks];
+ }
+ DayPilot.Date.Cache.DurationCtor["" + ticks] = this;
+
+ this.toString = function(pattern) {
+ if (!pattern) {
+ return d.days() + "." + d.hours() + ":" + d.minutes() + ":" + d.seconds() + "." + d.milliseconds();
+ }
+
+ var minutes = d.minutes();
+ minutes = (minutes < 10 ? "0" : "") + minutes;
+
+ // dumb replacement
+ var result = pattern;
+ result = result.replace("mm", minutes);
+ result = result.replace("m", d.minutes());
+ result = result.replace("H", d.hours());
+ result = result.replace("h", d.hours());
+ result = result.replace("d", d.days());
+ result = result.replace("s", d.seconds());
+ return result;
+ };
+
+ this.totalHours = function() {
+ return d.ticks / hour;
+ };
+
+ this.totalDays = function() {
+ return d.ticks / day;
+ };
+
+ this.totalMinutes = function() {
+ return d.ticks / minute;
+ };
+
+ this.totalSeconds = function() {
+ return d.ticks / second;
+ };
+
+ this.days = function() {
+ return Math.floor(d.totalDays());
+ };
+
+ this.hours = function() {
+ var hourPartTicks = d.ticks - d.days()*day;
+ return Math.floor(hourPartTicks/hour);
+ };
+
+ this.minutes = function() {
+ var minutePartTicks = d.ticks - Math.floor(d.totalHours()) * hour;
+ return Math.floor(minutePartTicks/minute);
+ };
+
+ this.seconds = function() {
+ var secondPartTicks = d.ticks - Math.floor(d.totalMinutes()) * minute;
+ return Math.floor(secondPartTicks/second);
+ };
+
+ this.milliseconds = function() {
+ return d.ticks % second;
+ };
+
+ };
+
+ DayPilot.Duration.weeks = function(i) {
+ return new DayPilot.Duration(i * 1000*60*60*24*7);
+ };
+
+ DayPilot.Duration.days = function(i) {
+ return new DayPilot.Duration(i * 1000*60*60*24);
+ };
+
+ DayPilot.Duration.hours = function(i) {
+ return new DayPilot.Duration(i * 1000*60*60);
+ };
+
+ DayPilot.Duration.minutes = function(i) {
+ return new DayPilot.Duration(i * 1000*60);
+ };
+
+ DayPilot.Duration.seconds = function(i) {
+ return new DayPilot.Duration(i * 1000);
+ };
+
+ // alias to DayPilot.Duration
+ // disabled, doesn't work with caching
+ DayPilot.TimeSpan = function() {
+
+ throw "Please use DayPilot.Duration class instead of DayPilot.TimeSpan.";
+ // DayPilot.Duration.apply(this, arguments);
+ };
+ try {
+ DayPilot.TimeSpan.prototype = Object.create(DayPilot.Duration.prototype); // make instanceof work
+ }
+ catch (e) {} // doesn't work in IE8
+
+ // DayPilot.TimeSpan.prototype.constructor = DayPilot.TimeSpan; // not necessary, it's an alias, not an inherited class
+
+ /* Date utils */
+
+ // DayPilot.Date class
+ /* Constructor signatures:
+
+ -- new DayPilot.Date(date, isLocal)
+ date - JavaScript Date object
+ isLocal - true if the local time should be taken from date, otherwise GMT base is used
+
+ -- new DayPilot.Date() - returns now, using local date
+
+ -- new DayPilot.Date(string)
+ string - date in ISO 8601 format, e.g. 2009-01-01T00:00:00
+
+ */
+ DayPilot.Date = function(date, readLocal) {
+
+ if (date instanceof DayPilot.Date) { // it's already a DayPilot.Date object, return it (no copy)
+ return date;
+ }
+
+ var ticks;
+
+ if (DayPilot.Util.isNullOrUndefined(date)) { // date not set, use NOW
+ ticks = DayPilot.DateUtil.fromLocal().getTime();
+ date = ticks;
+ }
+
+ var cache = DayPilot.Date.Cache.Ctor;
+ if (cache[date]) {
+ DayPilot.Stats.cacheHitsCtor += 1;
+ return cache[date];
+ }
+
+ var isString = false;
+
+ if (typeof date === "string") {
+ ticks = DayPilot.DateUtil.fromStringSortable(date, readLocal).getTime();
+ isString = true;
+ }
+ else if (typeof date === "number") {
+ if (isNaN(date)) {
+ throw "Cannot create DayPilot.Date from NaN";
+ }
+ ticks = date;
+ }
+ else if (date instanceof Date) {
+ if (readLocal) {
+ ticks = DayPilot.DateUtil.fromLocal(date).getTime();
+ }
+ else {
+ ticks = date.getTime();
+ }
+ }
+ else {
+ throw "Unrecognized parameter: use Date, number or string in ISO 8601 format";
+ }
+
+ var value = ticksToSortable(ticks); // normalized value
+
+ if (cache[value]) {
+ return cache[value];
+ }
+
+ cache[value] = this;
+ cache[ticks] = this;
+ if (isString && value !== date && DayPilot.DateUtil.hasTzSpec(date)) { // don't cache strings with TZ spec
+ cache[date] = this;
+ }
+
+ if (Object.defineProperty) {
+ Object.defineProperty(this, "ticks", {
+ get: function() { return ticks; }
+ });
+ Object.defineProperty(this, "value", {
+ "value": value,
+ "writable": false,
+ "enumerable": true
+ });
+ }
+ else {
+ this.ticks = ticks;
+ this.value = value;
+ }
+
+ if (DayPilot.Date.Config.legacyShowD) {
+ this.d = new Date(ticks);
+ }
+
+ DayPilot.Stats.dateObjects += 1;
+ };
+
+ DayPilot.Date.Config = {};
+ DayPilot.Date.Config.legacyShowD = false;
+
+ DayPilot.Date.Cache = {};
+ DayPilot.Date.Cache.Parsing = {};
+ DayPilot.Date.Cache.Ctor = {};
+ DayPilot.Date.Cache.Ticks = {};
+ DayPilot.Date.Cache.DurationCtor = {};
+
+ DayPilot.Date.Cache.clear = function() {
+ DayPilot.Date.Cache.Parsing = {};
+ DayPilot.Date.Cache.Ctor = {};
+ DayPilot.Date.Cache.Ticks = {};
+ DayPilot.Date.Cache.DurationCtor = {};
+ };
+
+
+ DayPilot.Date.prototype.addDays = function(days) {
+ if (!days) {
+ return this;
+ }
+ return new DayPilot.Date(this.ticks + days * 24 * 60 * 60 * 1000);
+ };
+
+ DayPilot.Date.prototype.addHours = function(hours) {
+ if (!hours) {
+ return this;
+ }
+ return this.addTime(hours * 60 * 60 * 1000);
+ };
+
+ DayPilot.Date.prototype.addMilliseconds = function(millis) {
+ if (!millis) {
+ return this;
+ }
+ return this.addTime(millis);
+ };
+
+ DayPilot.Date.prototype.addMinutes = function(minutes) {
+ if (!minutes) {
+ return this;
+ }
+ return this.addTime(minutes * 60 * 1000);
+ };
+
+ DayPilot.Date.prototype.addMonths = function(months) {
+ if (!months) {
+ return this;
+ }
+
+ var date = new Date(this.ticks);
+
+ var y = date.getUTCFullYear();
+ var m = date.getUTCMonth() + 1;
+
+ if (months > 0) {
+ while (months >= 12) {
+ months -= 12;
+ y++;
+ }
+ if (months > 12 - m) {
+ y++;
+ m = months - (12 - m);
+ }
+ else {
+ m += months;
+ }
+ }
+ else {
+ while (months <= -12) {
+ months += 12;
+ y--;
+ }
+ if (m + months <= 0) { //
+ y--;
+ m = 12 + m + months;
+ }
+ else {
+ m = m + months;
+ }
+ }
+
+ var d = new Date(date.getTime());
+ d.setUTCDate(1);
+ d.setUTCFullYear(y);
+ d.setUTCMonth(m - 1);
+
+ //var max = DayPilot.Date.daysInMonth(y, m);
+ var max = new DayPilot.Date(d).daysInMonth();
+ d.setUTCDate(Math.min(max, date.getUTCDate()));
+
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.addSeconds = function(seconds) {
+ if (!seconds) {
+ return this;
+ }
+ return this.addTime(seconds * 1000);
+ };
+
+ DayPilot.Date.prototype.addTime = function(ticks) {
+ if (!ticks) {
+ return this;
+ }
+ if (ticks instanceof DayPilot.Duration) {
+ ticks = ticks.ticks;
+ }
+ return new DayPilot.Date(this.ticks + ticks);
+ };
+
+ DayPilot.Date.prototype.addYears = function(years) {
+ var original = new Date(this.ticks);
+ var d = new Date(this.ticks);
+ var y = this.getYear() + years;
+ var m = this.getMonth();
+
+ d.setUTCDate(1);
+ d.setUTCFullYear(y);
+ d.setUTCMonth(m);
+
+ //var max = DayPilot.Date.daysInMonth(y, m + 1);
+ var max = new DayPilot.Date(d).daysInMonth();
+ d.setUTCDate(Math.min(max, original.getUTCDate()));
+
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.dayOfWeek = function() {
+ return new Date(this.ticks).getUTCDay();
+ };
+
+ DayPilot.Date.prototype.getDayOfWeek = function() {
+ return new Date(this.ticks).getUTCDay();
+ };
+
+ DayPilot.Date.prototype.getDayOfYear = function() {
+ var first = this.firstDayOfYear();
+ return DayPilot.DateUtil.daysDiff(first, this) + 1;
+ };
+
+ DayPilot.Date.prototype.daysInMonth = function() {
+ var date = new Date(this.ticks);
+ var month = date.getUTCMonth() + 1;
+ var year = date.getUTCFullYear();
+
+
+ var m = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];
+ if (month !== 2)
+ return m[month - 1];
+ if (year % 4 !== 0)
+ return m[1];
+ if (year % 100 === 0 && year % 400 !== 0)
+ return m[1];
+ return m[1] + 1;
+
+ };
+
+ DayPilot.Date.prototype.daysInYear = function() {
+ var year = this.getYear();
+ if (year % 4 !== 0) {
+ return 365;
+ }
+ if (year % 100 === 0 && year % 400 !== 0) {
+ return 365;
+ }
+ return 366;
+ };
+
+ DayPilot.Date.prototype.dayOfYear = function() {
+ return Math.ceil((this.getDatePart().getTime() - this.firstDayOfYear().getTime()) / 86400000) + 1;
+ };
+
+ // not required, direct comparison can be used
+ DayPilot.Date.prototype.equals = function(another) {
+ if (another === null) {
+ return false;
+ }
+ if (another instanceof DayPilot.Date) {
+ return this === another;
+ }
+ else {
+ throw "The parameter must be a DayPilot.Date object (DayPilot.Date.equals())";
+ }
+ };
+
+ DayPilot.Date.prototype.firstDayOfMonth = function() {
+ //var utc = DayPilot.Date.firstDayOfMonth(this.getYear(), this.getMonth() + 1);
+ //return new DayPilot.Date(utc);
+
+ var d = new Date();
+ d.setUTCFullYear(this.getYear(), this.getMonth(), 1);
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ return new DayPilot.Date(d);
+
+ };
+
+ DayPilot.Date.prototype.firstDayOfYear = function() {
+ var year = this.getYear();
+ var d = new Date();
+ d.setUTCFullYear(year, 0, 1);
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.firstDayOfWeek = function(weekStarts) {
+ var d = this;
+ if (weekStarts instanceof DayPilot.Locale) {
+ weekStarts = weekStarts.weekStarts;
+ }
+ else if (typeof weekStarts === "string" && DayPilot.Locale.find(weekStarts)) {
+ var locale = DayPilot.Locale.find(weekStarts);
+ weekStarts = locale.weekStarts;
+ }
+ else {
+ weekStarts = weekStarts || 0;
+ }
+
+ var day = d.dayOfWeek();
+ while (day !== weekStarts) {
+ d = d.addDays(-1);
+ day = d.dayOfWeek();
+ }
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.getDay = function() {
+ return new Date(this.ticks).getUTCDate();
+ };
+
+ DayPilot.Date.prototype.getDatePart = function() {
+ var d = new Date(this.ticks);
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.getYear = function() {
+ return new Date(this.ticks).getUTCFullYear();
+ };
+
+ DayPilot.Date.prototype.getHours = function() {
+ return new Date(this.ticks).getUTCHours();
+ };
+
+ DayPilot.Date.prototype.getMilliseconds = function() {
+ return new Date(this.ticks).getUTCMilliseconds();
+ };
+
+ DayPilot.Date.prototype.getMinutes = function() {
+ return new Date(this.ticks).getUTCMinutes();
+ };
+
+ DayPilot.Date.prototype.getMonth = function() {
+ return new Date(this.ticks).getUTCMonth();
+ };
+
+ DayPilot.Date.prototype.getSeconds = function() {
+ return new Date(this.ticks).getUTCSeconds();
+ };
+
+ DayPilot.Date.prototype.getTotalTicks = function() {
+ return this.getTime();
+ };
+
+ // undocumented
+ DayPilot.Date.prototype.getTime = function() {
+ return this.ticks;
+ };
+
+ DayPilot.Date.prototype.getTimePart = function() {
+ var datePart = this.getDatePart();
+ return DayPilot.DateUtil.diff(this, datePart);
+ };
+
+ DayPilot.Date.prototype.lastDayOfMonth = function() {
+ //var utc = DayPilot.Date.lastDayOfMonth(this.getYear(), this.getMonth() + 1);
+ //return new DayPilot.Date(utc);
+ var d = new Date(this.firstDayOfMonth().getTime());
+ var length = this.daysInMonth();
+ d.setUTCDate(length);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.weekNumber = function() {
+ var first = this.firstDayOfYear();
+ var days = (this.getTime() - first.getTime()) / 86400000;
+ return Math.ceil((days + first.dayOfWeek() + 1) / 7);
+ };
+
+ // ISO 8601
+ DayPilot.Date.prototype.weekNumberISO = function() {
+ var thursdayFlag = false;
+ var dayOfYear = this.dayOfYear();
+
+ var startWeekDayOfYear = this.firstDayOfYear().dayOfWeek();
+ var endWeekDayOfYear = this.firstDayOfYear().addYears(1).addDays(-1).dayOfWeek();
+ //int startWeekDayOfYear = new DateTime(date.getYear(), 1, 1).getDayOfWeekOrdinal();
+ //int endWeekDayOfYear = new DateTime(date.getYear(), 12, 31).getDayOfWeekOrdinal();
+
+ if (startWeekDayOfYear === 0) {
+ startWeekDayOfYear = 7;
+ }
+ if (endWeekDayOfYear === 0) {
+ endWeekDayOfYear = 7;
+ }
+
+ var daysInFirstWeek = 8 - (startWeekDayOfYear);
+
+ if (startWeekDayOfYear === 4 || endWeekDayOfYear === 4) {
+ thursdayFlag = true;
+ }
+
+ var fullWeeks = Math.ceil((dayOfYear - (daysInFirstWeek)) / 7.0);
+
+ var weekNumber = fullWeeks;
+
+ if (daysInFirstWeek >= 4) {
+ weekNumber = weekNumber + 1;
+ }
+
+ if (weekNumber > 52 && !thursdayFlag) {
+ weekNumber = 1;
+ }
+
+ if (weekNumber === 0) {
+ weekNumber = this.firstDayOfYear().addDays(-1).weekNumberISO(); //weekNrISO8601(new DateTime(date.getYear() - 1, 12, 31));
+ }
+
+ return weekNumber;
+
+ };
+
+ DayPilot.Date.prototype.toDateLocal = function() {
+ var date = new Date(this.ticks);
+
+ var d = new Date();
+ d.setFullYear(date.getUTCFullYear(), date.getUTCMonth(), date.getUTCDate());
+ d.setHours(date.getUTCHours());
+ d.setMinutes(date.getUTCMinutes());
+ d.setSeconds(date.getUTCSeconds());
+ d.setMilliseconds(date.getUTCMilliseconds());
+ return d;
+
+ };
+
+ DayPilot.Date.prototype.toDate = function() {
+ return new Date(this.ticks);
+ };
+
+ DayPilot.Date.prototype.toJSON = function() {
+ return this.value;
+ };
+
+ // formatting and languages needed here
+ DayPilot.Date.prototype.toString = function(pattern, locale) {
+ if (!pattern) {
+ return this.toStringSortable();
+ }
+ return new Pattern(pattern, locale).print(this);
+ };
+
+ DayPilot.Date.prototype.toStringSortable = function() {
+ return ticksToSortable(this.ticks);
+ };
+
+ function ticksToSortable(ticks) {
+
+ var cache = DayPilot.Date.Cache.Ticks;
+ if (cache[ticks]) {
+ DayPilot.Stats.cacheHitsTicks += 1;
+ return cache[ticks];
+ }
+
+ var d = new Date(ticks);
+
+ var millisecond;
+ var ms = d.getUTCMilliseconds();
+
+ if (ms === 0) {
+ millisecond = "";
+ }
+ else if (ms < 10) {
+ millisecond = ".00" + ms;
+ }
+ else if (ms < 100) {
+ millisecond = ".0" + ms;
+ }
+ else {
+ millisecond = "." + ms
+ }
+
+ var second = d.getUTCSeconds();
+ if (second < 10)
+ second = "0" + second;
+ var minute = d.getUTCMinutes();
+ if (minute < 10)
+ minute = "0" + minute;
+ var hour = d.getUTCHours();
+ if (hour < 10)
+ hour = "0" + hour;
+ var day = d.getUTCDate();
+ if (day < 10)
+ day = "0" + day;
+ var month = d.getUTCMonth() + 1;
+ if (month < 10)
+ month = "0" + month;
+ var year = d.getUTCFullYear();
+
+ if (year <= 0) {
+ throw "The minimum year supported is 1.";
+ }
+ if (year < 10) {
+ year = "000" + year;
+ }
+ else if (year < 100) {
+ year = "00" + year;
+ }
+ else if (year < 1000) {
+ year = "0" + year;
+ }
+
+ var result = year + "-" + month + "-" + day + 'T' + hour + ":" + minute + ":" + second + millisecond;
+ cache[ticks] = result;
+ return result;
+ }
+
+ /* static functions, return DayPilot.Date object */
+
+ // returns null if parsing was not successful
+ DayPilot.Date.parse = function(str, pattern, locale) {
+ var p = new Pattern(pattern, locale);
+ return p.parse(str);
+ };
+
+ var todayCount = 0;
+
+ DayPilot.Date.today = function() {
+ //return new DayPilot.Date().getDatePart();
+ return new DayPilot.Date(DayPilot.DateUtil.localToday(), true);
+ };
+
+ DayPilot.Date.now = function() {
+ return new DayPilot.Date();
+ };
+
+ DayPilot.Date.fromYearMonthDay = function(year, month, day) {
+ month = month || 1;
+ day = day || 1;
+
+ var d = new Date(0);
+ d.setUTCFullYear(year);
+ d.setUTCMonth(month - 1);
+ d.setUTCDate(day);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.DateUtil = {};
+
+ /* internal functions, all operate with GMT base of the date object
+ (except of DayPilot.DateUtil.fromLocal()) */
+
+ DayPilot.DateUtil.fromStringSortable = function(string, readLocal) {
+ /*
+ Supported formats:
+ 2015-01-01
+ 2015-01-01T00:00:00
+ 2015-01-01T00:00:00.000
+ 2015-01-01T00:00:00Z
+ 2015-01-01T00:00:00.000Z
+ 2015-01-01T00:00:00+01:00
+ 2015-01-01T00:00:00.000+01:00
+
+ */
+
+ if (!string) {
+ throw "Can't create DayPilot.Date from an empty string";
+ }
+
+ var len = string.length;
+ var date = len === 10;
+ var datetime = len === 19;
+ var long = len > 19;
+
+ if (!date && !datetime && !long) {
+ throw "Invalid string format (use '2010-01-01' or '2010-01-01T00:00:00'): " + string;
+ }
+
+ if (DayPilot.Date.Cache.Parsing[string] && !readLocal) {
+ DayPilot.Stats.cacheHitsParsing += 1;
+ return DayPilot.Date.Cache.Parsing[string];
+ }
+
+ var year = string.substring(0, 4);
+ var month = string.substring(5, 7);
+ var day = string.substring(8, 10);
+
+ var d = new Date(0);
+ d.setUTCFullYear(year, month - 1, day);
+
+ if (date) {
+ /*
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ */
+ //result = d;
+ DayPilot.Date.Cache.Parsing[string] = d;
+ return d;
+ }
+
+ var hours = string.substring(11, 13);
+ var minutes = string.substring(14, 16);
+ var seconds = string.substring(17, 19);
+
+ d.setUTCHours(hours);
+ d.setUTCMinutes(minutes);
+ d.setUTCSeconds(seconds);
+ //d.setUTCMilliseconds(0);
+ //result = d;
+
+ if (datetime) {
+ DayPilot.Date.Cache.Parsing[string] = d;
+ return d;
+ }
+
+ var tzdir = string[19];
+
+ var tzoffset = 0;
+
+ if (tzdir === ".") {
+ var ms = parseInt(string.substring(20, 23)); /// .000
+ d.setUTCMilliseconds(ms);
+ tzoffset = DayPilot.DateUtil.getTzOffsetMinutes(string.substring(23));
+ }
+ else {
+ tzoffset = DayPilot.DateUtil.getTzOffsetMinutes(string.substring(19));
+ }
+
+ var dd = new DayPilot.Date(d);
+ if (!readLocal) {
+ dd = dd.addMinutes(-tzoffset);
+ }
+
+ d = dd.toDate(); // get UTC base
+
+ DayPilot.Date.Cache.Parsing[string] = d;
+ return d;
+ };
+
+ DayPilot.DateUtil.getTzOffsetMinutes = function(string) {
+ if (DayPilot.Util.isNullOrUndefined(string) || string === "") {
+ return 0;
+ }
+ if (string === "Z") {
+ return 0;
+ }
+
+ var tzdir = string[0];
+
+ var tzhours = parseInt(string.substring(1, 3));
+ var tzminutes = parseInt(string.substring(4));
+ var tzoffset = tzhours * 60 + tzminutes;
+
+ if (tzdir === "-") {
+ return -tzoffset;
+ }
+ else if (tzdir === "+") {
+ return tzoffset;
+ }
+ else {
+ throw "Invalid timezone spec: " + string;
+ }
+ };
+
+ DayPilot.DateUtil.hasTzSpec = function(string) {
+ if (string.indexOf("+")) {
+ return true;
+ }
+ if (string.indexOf("-")) {
+ return true;
+ }
+ return false;
+ };
+
+
+ // rename candidate: diffDays
+ DayPilot.DateUtil.daysDiff = function(first, second) {
+ (first && second) || (function() { throw "two parameters required"; })();
+
+ first = new DayPilot.Date(first);
+ second = new DayPilot.Date(second);
+
+ if (first.getTime() > second.getTime()) {
+ return null;
+ }
+
+ var i = 0;
+ var fDay = first.getDatePart();
+ var sDay = second.getDatePart();
+
+ while (fDay < sDay) {
+ fDay = fDay.addDays(1);
+ i++;
+ }
+
+ return i;
+ };
+
+ DayPilot.DateUtil.daysSpan = function(first, second) {
+ (first && second) || (function() { throw "two parameters required"; })();
+
+ first = new DayPilot.Date(first);
+ second = new DayPilot.Date(second);
+
+ if (first.getTime() === second.getTime()) {
+ return 0;
+ }
+
+ var diff = DayPilot.DateUtil.daysDiff(first, second);
+
+ if (second.getTime() == second.getDatePart().getTime()) {
+ diff--;
+ }
+
+ return diff;
+ };
+
+ DayPilot.DateUtil.diff = function(first, second) { // = first - second
+ if (!(first && second && first.getTime && second.getTime)) {
+ throw "Both compared objects must be Date objects (DayPilot.Date.diff).";
+ }
+
+ return first.getTime() - second.getTime();
+ };
+
+ // returns Date object
+ DayPilot.DateUtil.fromLocal = function(localDate) {
+ if (!localDate) {
+ localDate = new Date();
+ }
+
+ var d = new Date();
+ d.setUTCFullYear(localDate.getFullYear(), localDate.getMonth(), localDate.getDate());
+ d.setUTCHours(localDate.getHours());
+ d.setUTCMinutes(localDate.getMinutes());
+ d.setUTCSeconds(localDate.getSeconds());
+ d.setUTCMilliseconds(localDate.getMilliseconds());
+ return d;
+ };
+
+ DayPilot.DateUtil.localToday = function() {
+ var d = new Date();
+ d.setHours(0);
+ d.setMinutes(0);
+ d.setSeconds(0);
+ d.setMilliseconds(0);
+ return d;
+ };
+
+ // rename candidate: toHourString
+ DayPilot.DateUtil.hours = function(date, use12) {
+
+ var minute = date.getUTCMinutes();
+ if (minute < 10)
+ minute = "0" + minute;
+
+
+ var hour = date.getUTCHours();
+ //if (hour < 10) hour = "0" + hour;
+
+ if (use12) {
+ var am = hour < 12;
+ var hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ var suffix = am ? "AM" : "PM";
+ return hour + ':' + minute + ' ' + suffix;
+ }
+ else {
+ return hour + ':' + minute;
+ }
+ };
+
+ DayPilot.DateUtil.max = function(first, second) {
+ if (first.getTime() > second.getTime()) {
+ return first;
+ }
+ else {
+ return second;
+ }
+ };
+
+ DayPilot.DateUtil.min = function(first, second) {
+ if (first.getTime() < second.getTime()) {
+ return first;
+ }
+ else {
+ return second;
+ }
+ };
+
+ var Pattern = function(pattern, locale) {
+ if (typeof locale === "string") {
+ locale = DayPilot.Locale.find(locale);
+ }
+ var locale = locale || DayPilot.Locale.US;
+ var all = [
+ {"seq": "yyyy", "expr": "[0-9]{4,4\u007d", "str": function(d) {
+ return d.getYear();
+ }},
+ {"seq": "yy", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ return d.getYear() % 100;
+ }},
+ {"seq": "mm", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getMinutes();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "m", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getMinutes();
+ return r;
+ }},
+ {"seq": "HH", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getHours();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "H", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getHours();
+ return r;
+ }},
+ {"seq": "hh", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var hour = d.getHours();
+ var hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ var r = hour;
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "h", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var hour = d.getHours();
+ var hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ return hour;
+ }},
+ {"seq": "ss", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getSeconds();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "s", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getSeconds();
+ return r;
+ }},
+ {"seq": "MMMM", "expr": "[^\\s0-9]*", "str": function(d) {
+ var r = locale.monthNames[d.getMonth()];
+ return r;
+ }, "transform" : function(input) {
+ var index = DayPilot.indexOf(locale.monthNames, input, equalsIgnoreCase);
+ if (index < 0) {
+ return null;
+ }
+ return index + 1;
+ }},
+ {"seq": "MMM", "expr": "[^\\s0-9]*", "str": function(d) { // \u0073 = 's'
+ var r = locale.monthNamesShort[d.getMonth()];
+ return r;
+ }, "transform" : function(input) {
+ var index = DayPilot.indexOf(locale.monthNamesShort, input, equalsIgnoreCase);
+ if (index < 0) {
+ return null;
+ }
+ return index + 1;
+ }},
+ {"seq": "MM", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getMonth() + 1;
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "M", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getMonth() + 1;
+ return r;
+ }},
+ {"seq": "dddd", "expr": "[^\\s0-9]*", "str": function(d) {
+ var r = locale.dayNames[d.getDayOfWeek()];
+ return r;
+ }},
+ {"seq": "ddd", "expr": "[^\\s0-9]*", "str": function(d) {
+ var r = locale.dayNamesShort[d.getDayOfWeek()];
+ return r;
+ }},
+ {"seq": "dd", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getDay();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "%d", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getDay();
+ return r;
+ }},
+ {"seq": "d", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getDay();
+ return r;
+ }},
+ {"seq": "tt", "expr": "(AM|PM|am|pm)", "str": function(d) {
+ var hour = d.getHours();
+ var am = hour < 12;
+ return am ? "AM" : "PM";
+ }, "transform" : function(input) {
+ return input.toUpperCase();
+ }},
+ ];
+
+ var escapeRegex = function(text) {
+ return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&");
+ };
+
+ this.init = function() {
+ this.year = this.findSequence("yyyy");
+ this.month = this.findSequence("MMMM") || this.findSequence("MMM") || this.findSequence("MM") || this.findSequence("M");
+ this.day = this.findSequence("dd") || this.findSequence("d");
+
+ this.hours = this.findSequence("HH") || this.findSequence("H");
+ this.minutes = this.findSequence("mm") || this.findSequence("m");
+ this.seconds = this.findSequence("ss") || this.findSequence("s");
+
+ this.ampm = this.findSequence("tt");
+ this.hours12 = this.findSequence("hh") || this.findSequence("h");
+
+ /*if (this.hours && this.ampm) {
+ throw new DayPilot.Exception("'HH' and 'H' specifiers cannot be used in combination with 'tt'. Use 12-hour clock specifiers: 'hh' or 'h'.");
+ }*/
+ };
+
+ this.findSequence = function(seq) {
+
+ function defaultTransform(value) {
+ return parseInt(value);
+ }
+
+ var index = pattern.indexOf(seq);
+ if (index === -1) {
+ return null;
+ }
+ return {
+ "findValue": function(input) {
+ var prepared = escapeRegex(pattern);
+ var transform = null;
+ for (var i = 0; i < all.length; i++) {
+ var len = all[i].length;
+ var pick = (seq === all[i].seq);
+ //var expr = "";
+ var expr = all[i].expr;
+ if (pick) {
+ expr = "(" + expr + ")";
+ transform = all[i].transform;
+ }
+ prepared = prepared.replace(all[i].seq, expr);
+ }
+
+ prepared = "^" + prepared + "$";
+
+ try {
+ var r = new RegExp(prepared);
+ var array = r.exec(input);
+ if (!array) {
+ return null;
+ }
+ transform = transform || defaultTransform; // parseInt is the default transform/parse function
+ return transform(array[1]);
+ }
+ catch (e) {
+ throw "unable to create regex from: " + prepared;
+ }
+ }
+ };
+ };
+
+ this.print = function(date) {
+ // always recompiles the pattern
+
+ var find = function(t) {
+ for (var i = 0; i < all.length; i++) {
+ if (all[i] && all[i].seq === t) {
+ return all[i];
+ }
+ }
+ return null;
+ };
+
+ var eos = pattern.length <= 0;
+ var pos = 0;
+ var components = [];
+
+ while (!eos) {
+ var rem = pattern.substring(pos);
+ var matches = /%?(.)\1*/.exec(rem); // matches a sequence of identical characters, with an optional '%' preceding char
+ if (matches && matches.length > 0) {
+ var match = matches[0];
+ var q = find(match);
+ if (q) {
+ components.push(q);
+ }
+ else {
+ components.push(match);
+ }
+ pos += match.length;
+ eos = pattern.length <= pos;
+ }
+ else {
+ eos = true;
+ }
+ }
+
+ // resolve placeholders
+ for (var i = 0; i < components.length; i++) {
+ var c = components[i];
+ if (typeof c !== 'string') {
+ components[i] = c.str(date);
+ }
+ }
+
+ return components.join("");
+ };
+
+
+
+ this.parse = function(input) {
+
+ var year = this.year.findValue(input);
+ if (!year) {
+ return null; // unparseable
+ }
+
+ var month = this.month.findValue(input);
+ if (DayPilot.Util.isNullOrUndefined(month)) {
+ return null;
+ }
+ if (month > 12 || month < 1) {
+ return null;
+ }
+ var day = this.day.findValue(input);
+
+ var daysInMonth = DayPilot.Date.fromYearMonthDay(year, month).daysInMonth();
+ if (day < 1 || day > daysInMonth) {
+ return null;
+ }
+
+ var hours = this.hours ? this.hours.findValue(input) : 0;
+ var minutes = this.minutes ? this.minutes.findValue(input) : 0;
+ var seconds = this.seconds ? this.seconds.findValue(input) : 0;
+
+ var ampm = this.ampm ? this.ampm.findValue(input): null;
+
+ if (this.ampm && this.hours12) {
+
+ var hours12 = this.hours12.findValue(input);
+
+ if (hours12 < 1 || hours12 > 12) {
+ return null;
+ }
+
+ if (ampm === "PM") {
+ if (hours12 === 12) {
+ hours = 12;
+ }
+ else {
+ hours = hours12 + 12;
+ }
+ }
+ else {
+ if (hours12 === 12) {
+ hours = 0;
+ }
+ else {
+ hours = hours12;
+ }
+ }
+
+ }
+
+ if (hours < 0 || hours > 23) {
+ return null;
+ }
+
+ if (minutes < 0 || minutes > 59) {
+ return null;
+ }
+
+ if (seconds < 0 || seconds > 59) {
+ return null;
+ }
+
+ var d = new Date();
+ d.setUTCFullYear(year, month - 1, day);
+ d.setUTCHours(hours);
+ d.setUTCMinutes(minutes);
+ d.setUTCSeconds(seconds);
+ d.setUTCMilliseconds(0);
+
+ return new DayPilot.Date(d);
+ };
+
+ this.init();
+
+ };
+
+ function equalsIgnoreCase(str1, str2) {
+ if (DayPilot.Util.isNullOrUndefined(str1)) {
+ return false;
+ }
+ if (DayPilot.Util.isNullOrUndefined(str2)) {
+ return false;
+ }
+ return str1.toLocaleLowerCase() === str2.toLocaleLowerCase();
+ }
+
+ // DayPilot.Date END
+
+
+ DayPilot.ColorUtil = {};
+
+ function toHex(dec) {
+ dec = Math.min(dec, 255);
+ dec = Math.max(dec, 0);
+ var str = dec.toString(16);
+ return (dec < 16) ? "0" + str : str;
+ }
+
+ DayPilot.ColorUtil.hexToRgb = function(hex) {
+ if (!/^#[0-9a-f]{6}$/i.test(hex)) {
+ throw new DayPilot.Exception("Invalid color, only full hex color string accepted, eg. '#ffaaff'.");
+ }
+ hex = hex.replace("#", "");
+ return {
+ r: parseInt(hex.substring(0, 2), 16),
+ g: parseInt(hex.substring(2, 4), 16),
+ b: parseInt(hex.substring(4, 6), 16),
+ };
+ };
+
+ DayPilot.ColorUtil.rgbToHex = function(rgb) {
+ return "#" + toHex(rgb.r) + toHex(rgb.g) + toHex(rgb.b);
+ };
+
+ // pt in 255 base
+ DayPilot.ColorUtil.adjustLuminance = function(rgb, pt) {
+ return {
+ r: rgb.r + pt,
+ g: rgb.g + pt,
+ b: rgb.b + pt
+ };
+ };
+
+ DayPilot.ColorUtil.darker = function(hexColor, steps) {
+ var src = DayPilot.ColorUtil.hexToRgb(hexColor);
+ steps = steps || 1;
+ var step = 17; // (0xbb - 0xaa = 17)
+ var pt = steps * step;
+ var target = DayPilot.ColorUtil.adjustLuminance(src, -pt);
+ return DayPilot.ColorUtil.rgbToHex(target);
+ };
+
+ DayPilot.ColorUtil.lighter = function(hexColor, steps) {
+ if (typeof steps !== "number") {
+ steps = 1;
+ }
+ return DayPilot.ColorUtil.darker(hexColor, -steps);
+ };
+
+
+ DayPilot.ColorUtil.pl = function(hexColor) {
+ var rgb = DayPilot.ColorUtil.hexToRgb(hexColor);
+ var r = rgb.r / 255;
+ var g = rgb.g / 255;
+ var b = rgb.b / 255;
+ var pl = Math.sqrt(0.299*r*r + 0.587*g*g + 0.114*b*b);
+ return pl;
+ };
+
+ DayPilot.ColorUtil.contrasting = function(hexColor, light, dark) {
+ var pl = DayPilot.ColorUtil.pl(hexColor);
+ light = light || "#ffffff";
+ dark = dark || "#000000";
+ return pl > 0.5 ? dark : light;
+ };
+
+
+ DayPilot.Event = function(data, calendar, part) {
+ var e = this;
+ this.calendar = calendar;
+ this.data = data ? data : {};
+ this.part = part ? part : {};
+
+ // backwards compatibility, still accepts id in "value"
+ if (typeof this.data.id === 'undefined') {
+ this.data.id = this.data.value;
+ }
+
+ var copy = {};
+ var synced = ["id", "text", "start", "end", "resource"];
+
+ this.isEvent = true;
+
+ // internal
+ this.temp = function() {
+ if (copy.dirty) {
+ return copy;
+ }
+ for (var i = 0; i < synced.length; i++) {
+ copy[synced[i]] = e.data[synced[i]];
+ }
+ copy.dirty = true;
+ return copy;
+
+ };
+
+ // internal
+ this.copy = function() {
+ var result = {};
+ for (var i = 0; i < synced.length; i++) {
+ result[synced[i]] = e.data[synced[i]];
+ }
+ return result;
+ };
+
+ this.commit = function() {
+ if (!copy.dirty) {
+ return;
+ }
+
+ for (var i = 0; i < synced.length; i++) {
+ e.data[synced[i]] = copy[synced[i]];
+ }
+
+ copy.dirty = false;
+ };
+
+ this.dirty = function() {
+ return copy.dirty;
+ };
+
+ this.id = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.id;
+ }
+ else {
+ this.temp().id = val;
+ }
+ };
+ // obsolete, use id() instead
+ this.value = function(val) {
+ if (typeof val === 'undefined') {
+ return e.id();
+ }
+ else {
+ e.id(val);
+ }
+ };
+ this.text = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.text;
+ }
+ else {
+ this.temp().text = val;
+ this.client.innerHTML(val); // update the HTML automatically
+ }
+ };
+ this.start = function(val) {
+ if (typeof val === 'undefined') {
+ return new DayPilot.Date(e.data.start);
+ }
+ else {
+ this.temp().start = new DayPilot.Date(val);
+ }
+ };
+ this.end = function(val) {
+ if (typeof val === 'undefined') {
+ return new DayPilot.Date(e.data.end);
+ }
+ else {
+ this.temp().end = new DayPilot.Date(val);
+ }
+ };
+ this.resource = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.resource;
+ }
+ else {
+ this.temp().resource = val;
+ }
+ };
+ this.partStart = function() {
+ return new DayPilot.Date(this.part.start);
+ };
+ this.partEnd = function() {
+ return new DayPilot.Date(this.part.end);
+ };
+
+ this.tag = function(field) {
+ var values = e.data.tag;
+ if (!values) {
+ return null;
+ }
+ if (typeof field === 'undefined') {
+ return e.data.tag;
+ }
+ var fields = e.calendar.tagFields;
+ var index = -1;
+ for (var i = 0; i < fields.length; i++) {
+ if (field === fields[i])
+ index = i;
+ }
+ if (index === -1) {
+ throw "Field name not found.";
+ }
+ //var tags = t.split('&');
+ return values[index];
+ };
+
+ this.client = {};
+ this.client.innerHTML = function(val) {
+ if (typeof val === 'undefined') {
+ var data = e.cache || e.data;
+ var xssTextHtml = e.calendar && e.calendar.internal && e.calendar.internal.xssTextHtml;
+ if (xssTextHtml) {
+ return xssTextHtml(data.text, data.html);
+ }
+ return DayPilot.Util.escapeTextHtml(data.text, data.html);
+
+ /*
+ if (e.cache && typeof e.cache.html !== "undefined") {
+ return e.cache.html;
+ }
+ if (typeof e.data.html !== "undefined") {
+ return e.data.html;
+ }
+ return e.data.text;
+ */
+ }
+ else {
+ e.data.html = val;
+ }
+ };
+
+ this.client.html = this.client.innerHTML;
+
+ this.client.header = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.header;
+ }
+ else {
+ e.data.header = val;
+ }
+ };
+
+ this.client.cssClass = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.cssClass !== "undefined") {
+ return e.cache.cssClass;
+ }
+ return e.data.cssClass;
+ }
+ else {
+ e.data.cssClass = val;
+ }
+ };
+ this.client.toolTip = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.toolTip !== "undefined") {
+ return e.cache.toolTip;
+ }
+ return typeof e.data.toolTip !== 'undefined' ? e.data.toolTip : e.data.text;
+ }
+ else {
+ e.data.toolTip = val;
+ }
+ };
+
+ this.client.barVisible = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.barHidden !== "undefined") {
+ return !e.cache.barHidden;
+ }
+ return e.calendar.durationBarVisible && !e.data.barHidden;
+ }
+ else {
+ e.data.barHidden = !val;
+ }
+ };
+
+ this.client.backColor = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.backColor !== "undefined") {
+ return e.cache.backColor;
+ }
+ return typeof e.data.backColor !== "undefined" ? e.data.backColor : e.calendar.eventBackColor;
+ }
+ else {
+ e.data.backColor = val;
+ }
+ };
+
+ this.client.borderColor = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.borderColor !== "undefined") {
+ return e.cache.borderColor;
+ }
+ return typeof e.data.borderColor !== "undefined" ? e.data.borderColor : e.calendar.eventBorderColor;
+ }
+ else {
+ e.data.borderColor = val;
+ }
+ };
+
+ this.client.contextMenu = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.oContextMenu) {
+ return e.oContextMenu;
+ }
+ var cm = e.cache ? e.cache.contextMenu : e.data.contextMenu;
+ }
+ else {
+ e.oContextMenu = val;
+ }
+ };
+
+ this.client.moveEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.moveDisabled !== "undefined") {
+ return !e.cache.moveDisabled;
+ }
+
+ return e.calendar.eventMoveHandling !== 'Disabled' && !e.data.moveDisabled;
+ }
+ else {
+ e.data.moveDisabled = !val;
+ }
+ };
+
+ this.client.resizeEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.resizeDisabled !== "undefined") {
+ return !e.cache.resizeDisabled;
+ }
+ return e.calendar.eventResizeHandling !== 'Disabled' && !e.data.resizeDisabled;
+ }
+ else {
+ e.data.resizeDisabled = !val;
+ }
+ };
+
+ this.client.clickEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.clickDisabled !== "undefined") {
+ return !e.cache.clickDisabled;
+ }
+ return e.calendar.eventClickHandling !== 'Disabled' && !e.data.clickDisabled;
+ }
+ else {
+ e.data.clickDisabled = !val;
+ }
+ };
+
+ this.client.rightClickEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.rightClickDisabled !== "undefined") {
+ return !e.cache.rightClickDisabled;
+ }
+ return e.calendar.eventRightClickHandling !== 'Disabled' && !e.data.rightClickDisabled;
+ }
+ else {
+ e.data.rightClickDisabled = !val;
+ }
+ };
+
+ this.client.deleteEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.deleteDisabled !== "undefined") {
+ return !e.cache.deleteDisabled;
+ }
+ return e.calendar.eventDeleteHandling !== 'Disabled' && !e.data.deleteDisabled;
+ }
+ else {
+ e.data.deleteDisabled = !val;
+ }
+ };
+
+ this.toJSON = function(key) {
+ var json = {};
+ json.value = this.id(); // still sending it with the old name
+ json.id = this.id();
+ json.text = this.text();
+ json.start = this.start();
+ json.end = this.end();
+ json.tag = {};
+
+ if (e.calendar && e.calendar.tagFields) {
+ var fields = e.calendar.tagFields;
+ for (var i = 0; i < fields.length; i++) {
+ json.tag[fields[i]] = this.tag(fields[i]);
+ }
+ }
+
+ return json;
+ };
+ };
+
+})();
+
+/* JSON */
+// thanks to http://www.json.org/js.html
+
+
+// declares DayPilot.JSON.stringify()
+DayPilot.JSON = {};
+
+(function () {
+ function f(n) {
+ return n < 10 ? '0' + n : n;
+ }
+
+ if (typeof Date.prototype.toJSON2 !== 'function') {
+
+ Date.prototype.toJSON2 = function (key) {
+ return this.getUTCFullYear() + '-' +
+ f(this.getUTCMonth() + 1) + '-' +
+ f(this.getUTCDate()) + 'T' +
+ f(this.getUTCHours()) + ':' +
+ f(this.getUTCMinutes()) + ':' +
+ f(this.getUTCSeconds()) + '';
+ };
+
+ String.prototype.toJSON =
+ Number.prototype.toJSON =
+ Boolean.prototype.toJSON = function (key) {
+ return this.valueOf();
+ };
+ }
+
+ var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
+ escapeable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
+ gap,
+ indent,
+ meta = {
+ '\b': '\\b',
+ '\t': '\\t',
+ '\n': '\\n',
+ '\f': '\\f',
+ '\r': '\\r',
+ '"' : '\\"',
+ '\\': '\\\\'
+ },
+ rep;
+
+ function quote(string) {
+ escapeable.lastIndex = 0;
+ return escapeable.test(string) ?
+ '"' + string.replace(escapeable, function (a) {
+ var c = meta[a];
+ if (typeof c === 'string') {
+ return c;
+ }
+ return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
+ }) + '"' :
+ '"' + string + '"';
+ }
+
+ function str(key, holder) {
+ var i,
+ k,
+ v,
+ length,
+ mind = gap,
+ partial,
+ value = holder[key];
+ if (value && typeof value === 'object' && typeof value.toJSON2 === 'function') {
+ value = value.toJSON2(key);
+ }
+ else if (value && typeof value === 'object' && typeof value.toJSON === 'function' && !value.ignoreToJSON) {
+ value = value.toJSON(key);
+ }
+ if (typeof rep === 'function') {
+ value = rep.call(holder, key, value);
+ }
+ switch (typeof value) {
+ case 'string':
+ return quote(value);
+ case 'number':
+ return isFinite(value) ? String(value) : 'null';
+ case 'boolean':
+ case 'null':
+ return String(value);
+ case 'object':
+ if (!value) {
+ return 'null';
+ }
+ gap += indent;
+ partial = [];
+ if (typeof value.length === 'number' &&
+ !value.propertyIsEnumerable('length')) {
+ length = value.length;
+ for (i = 0; i < length; i += 1) {
+ partial[i] = str(i, value) || 'null';
+ }
+ v = partial.length === 0 ? '[]' :
+ gap ? '[\n' + gap +
+ partial.join(',\n' + gap) + '\n' +
+ mind + ']' :
+ '[' + partial.join(',') + ']';
+ gap = mind;
+ return v;
+ }
+ if (rep && typeof rep === 'object') {
+ length = rep.length;
+ for (i = 0; i < length; i += 1) {
+ k = rep[i];
+ if (typeof k === 'string') {
+ v = str(k, value);
+ if (v) {
+ partial.push(quote(k) + (gap ? ': ' : ':') + v);
+ }
+ }
+ }
+ } else {
+ for (k in value) {
+ if (Object.hasOwnProperty.call(value, k)) {
+ v = str(k, value);
+ if (v) {
+ partial.push(quote(k) + (gap ? ': ' : ':') + v);
+ }
+ }
+ }
+ }
+ v = (partial.length === 0) ? '{\u007D' :
+ gap ? '{\n' + gap + partial.join(',\n' + gap) + '\n' +
+ mind + '\u007D' : '{' + partial.join(',') + '\u007D';
+ gap = mind;
+ return v;
+ }
+ }
+
+ if (typeof DayPilot.JSON.stringify !== 'function') {
+ DayPilot.JSON.stringify = function (value, replacer, space) {
+ var i;
+ gap = '';
+ indent = '';
+ if (typeof space === 'number') {
+ for (i = 0; i < space; i += 1) {
+ indent += ' ';
+ }
+ } else if (typeof space === 'string') {
+ indent = space;
+ }
+ rep = replacer;
+ if (replacer && typeof replacer !== 'function' && (typeof replacer !== 'object' || typeof replacer.length !== 'number')) {
+ throw new Error('JSON.stringify');
+ }
+ return str('', {'': value});
+ };
+ }
+
+})();
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function() {
+
+ var doNothing = function() {};
+
+ if (typeof DayPilot.Calendar !== 'undefined' && DayPilot.Calendar.events) {
+ return;
+ }
+
+ var DayPilotCalendar = {};
+
+ // internal selecting
+ DayPilotCalendar.selectedCells = [];
+ DayPilotCalendar.topSelectedCell = null;
+ DayPilotCalendar.bottomSelectedCell = null;
+ DayPilotCalendar.selecting = false;
+ DayPilotCalendar.column = null;
+ DayPilotCalendar.firstSelected = null;
+ DayPilotCalendar.firstMousePos = null;
+
+ // internal resizing
+ DayPilotCalendar.originalMouse = null;
+ DayPilotCalendar.originalHeight = null;
+ DayPilotCalendar.originalTop = null;
+ DayPilotCalendar.resizing = null;
+ DayPilotCalendar.globalHandlers = false;
+
+ // internal moving
+ DayPilotCalendar.moving = null;
+
+ // helpers
+ DayPilotCalendar.register = function(calendar) {
+ if (!DayPilotCalendar.registered) {
+ DayPilotCalendar.registered = [];
+ }
+ var r = DayPilotCalendar.registered;
+
+ for (var i = 0; i < r.length; i++) {
+ if (r[i] === calendar) {
+ return;
+ }
+ }
+ r.push(calendar);
+ };
+
+ DayPilotCalendar.unregister = function (calendar) {
+ var a = DayPilotCalendar.registered;
+ if (!a) {
+ return;
+ }
+
+ var i = DayPilot.indexOf(a, calendar);
+ if (i === -1) {
+ return;
+ }
+ a.splice(i, 1);
+ };
+
+ DayPilotCalendar.getCellsAbove = function(cell) {
+ var array = [];
+ var c = DayPilotCalendar.getColumn(cell);
+
+ var tr = cell.parentNode;
+
+ var select = null;
+ while (tr && select !== DayPilotCalendar.firstSelected) {
+ select = tr.getElementsByTagName("td")[c];
+ array.push(select);
+ tr = tr.previousSibling;
+ while (tr && tr.tagName !== "TR") {
+ tr = tr.previousSibling;
+ }
+ }
+ return array;
+ };
+
+ DayPilotCalendar.getCellsBelow = function(cell) {
+ var array = [];
+ var c = DayPilotCalendar.getColumn(cell);
+ var tr = cell.parentNode;
+
+ var select = null;
+ while (tr && select !== DayPilotCalendar.firstSelected) {
+ select = tr.getElementsByTagName("td")[c];
+ array.push(select);
+ tr = tr.nextSibling;
+ while (tr && tr.tagName !== "TR") {
+ tr = tr.nextSibling;
+ }
+ }
+ return array;
+ };
+
+ DayPilotCalendar.getColumn = function(cell) {
+ var i = 0;
+ while (cell.previousSibling) {
+ cell = cell.previousSibling;
+ if (cell.tagName === "TD") {
+ i++;
+ }
+ }
+ return i;
+ };
+
+ DayPilotCalendar.gUnload = function (ev) {
+
+ if (!DayPilotCalendar.registered) {
+ return;
+ }
+ var r = DayPilotCalendar.registered;
+
+ for (var i = 0; i < r.length; i++) {
+ var c = r[i];
+ c.dispose();
+
+ DayPilotCalendar.unregister(c);
+ }
+
+ };
+
+ DayPilotCalendar.gMouseUp = function (ev){
+
+ if (DayPilotCalendar.resizing) {
+ if (!DayPilotCalendar.resizingShadow) {
+ DayPilotCalendar.resizing.style.cursor = 'default';
+ document.body.style.cursor = 'default';
+ DayPilotCalendar.resizing = null;
+ DayPilot.Global.resizing = null;
+ return;
+ }
+
+ var dpEvent = DayPilotCalendar.resizing.event;
+ var height = DayPilotCalendar.resizingShadow.clientHeight + 4;
+ var top = DayPilotCalendar.resizingShadow.offsetTop;
+ var border = DayPilotCalendar.resizing.dpBorder;
+
+ // stop resizing on the client
+ DayPilotCalendar.deleteShadow(DayPilotCalendar.resizingShadow);
+ DayPilotCalendar.resizingShadow = null;
+ DayPilotCalendar.resizing.style.cursor = 'default';
+ dpEvent.calendar.nav.top.style.cursor = 'auto';
+ //document.body.style.cursor = 'default';
+ DayPilotCalendar.resizing.onclick = null; // will be re-created anyway
+
+ DayPilotCalendar.resizing = null;
+ DayPilot.Global.resizing = null;
+
+ dpEvent.calendar._eventResizeDispatch(dpEvent, height, top, border);
+ }
+ else if (DayPilotCalendar.moving) {
+ if (!DayPilotCalendar.movingShadow) {
+ DayPilotCalendar.moving = null;
+ DayPilot.Global.moving = null;
+ document.body.style.cursor = 'default';
+ return;
+ }
+
+ var top = DayPilotCalendar.movingShadow.offsetTop;
+ var dpEvent = DayPilotCalendar.moving.event;
+
+ DayPilotCalendar.deleteShadow(DayPilotCalendar.movingShadow);
+ DayPilot.Util.removeClass(DayPilotCalendar.moving, dpEvent.calendar._prefixCssClass("_event_moving_source"));
+
+ var newColumnIndex = DayPilotCalendar.movingShadow.column;
+
+ // stop moving on the client
+ DayPilotCalendar.moving = null;
+ DayPilot.Global.moving = null;
+ DayPilotCalendar.movingShadow = null;
+ //document.body.style.cursor = 'default';
+ dpEvent.calendar.nav.top.style.cursor = 'auto';
+
+ dpEvent.calendar._eventMoveDispatch(dpEvent, newColumnIndex, top, ev);
+ }
+ else if (DayPilotCalendar.selecting && DayPilotCalendar.topSelectedCell !== null) {
+ var calendar = DayPilotCalendar.selecting.calendar;
+ DayPilotCalendar.selecting = false;
+
+ var sel = calendar.getSelection();
+
+ calendar._timeRangeSelectedDispatch(sel.start, sel.end, sel.resource);
+ if (calendar.timeRangeSelectedHandling !== "Hold" && calendar.timeRangeSelectedHandling !== "HoldForever") {
+ doNothing();
+ }
+ }
+ else {
+ DayPilotCalendar.selecting = false;
+ }
+
+
+ };
+
+ DayPilotCalendar.deleteShadow = function(shadow) {
+ if (!shadow) {
+ return;
+ }
+ if (!shadow.parentNode) {
+ return;
+ }
+
+ shadow.parentNode.removeChild(shadow);
+ };
+
+ DayPilotCalendar.moveShadow = function(column) {
+ var shadow = DayPilotCalendar.movingShadow;
+ var parent = shadow.parentNode;
+
+ parent.style.display = 'none';
+
+ shadow.parentNode.removeChild(shadow);
+ column.firstChild.appendChild(shadow);
+ shadow.style.left = '0px';
+
+ parent.style.display = '';
+
+ shadow.style.width = (DayPilotCalendar.movingShadow.parentNode.offsetWidth + 1) + 'px';
+ };
+
+ DayPilotCalendar.Calendar = function(id, options) {
+
+ var isConstructor = false;
+ if (this instanceof DayPilotCalendar.Calendar && !this.__constructor) {
+ isConstructor = true;
+ this.__constructor = true;
+ }
+
+ if (!isConstructor) {
+ throw "DayPilot.Calendar() is a constructor and must be called as 'var c = new DayPilot.Calendar(id);'";
+ }
+
+ var calendar = this;
+ this.uniqueID = null;
+ this.isCalendar = true;
+
+ this.v = '2024.3.539-lite';
+ this.id = id;
+ this.clientName = id;
+
+ this.cache = {};
+ this.cache.pixels = {};
+
+ this.elements = {};
+ this.elements.events = [];
+ this.elements.selection = [];
+
+ this.nav = {};
+
+ this.afterRender = function() {};
+
+ this.fasterDispose = true;
+
+ this.angularAutoApply = false;
+ this.api = 2;
+ this.businessBeginsHour = 9;
+ this.businessEndsHour = 18;
+ this.cellHeight = 30;
+ this.columnMarginRight = 5;
+ this.columnsLoadMethod = "GET";
+ this.contextMenu = null;
+ this.days = 1;
+ this.durationBarVisible = true;
+ this.eventsLoadMethod = "GET";
+ this.headerDateFormat = null; // uses locale.dateFormat by default
+ this.headerHeight = 30;
+ this.headerTextWrappingEnabled = false;
+ this.height = 300;
+ this.heightSpec = 'BusinessHours';
+ this.hideUntilInit = true;
+ this.hourWidth = 60;
+ this.initScrollPos = 'Auto';
+ this.loadingLabelHtml = null;
+ this.loadingLabelText = "Loading...";
+ this.loadingLabelVisible = true;
+ this.locale = "en-us";
+ this.showToolTip = true;
+ this.startDate = new DayPilot.Date().getDatePart();
+ this.cssClassPrefix = "calendar_default";
+ this.theme = null;
+ this.timeFormat = 'Auto';
+ this.viewType = "Days";
+ this.visible = true;
+ this.xssProtection = "Enabled";
+
+ this.headerClickHandling = "Enabled";
+ this.eventClickHandling = 'Enabled';
+ this.eventResizeHandling = 'Update';
+ this.eventRightClickHandling = 'ContextMenu';
+ this.eventMoveHandling = 'Update';
+ this.eventDeleteHandling = "Disabled";
+ this.timeRangeSelectedHandling = 'Enabled';
+
+ this.onBeforeCellRender = null;
+ this.onBeforeEventRender = null;
+
+ this.onEventClick = null;
+ this.onEventClicked = null;
+ this.onEventDelete = null;
+ this.onEventDeleted = null;
+ this.onEventMove = null;
+ this.onEventMoved = null;
+ this.onEventResize = null;
+ this.onEventResized = null;
+ this.onEventRightClick = null;
+ this.onEventRightClicked = null;
+ this.onHeaderClick = null;
+ this.onHeaderClicked = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+
+ this._disposed = false;
+
+ this.clearSelection = function() {
+ DayPilotCalendar.topSelectedCell = null;
+ DayPilotCalendar.bottomSelectedCell = null;
+ this._hideSelection();
+ };
+
+ this._hideSelection = function() {
+ DayPilot.de(calendar.elements.selection);
+ calendar.elements.selection = [];
+ calendar.nav.activeSelection = null;
+ };
+
+ this.cleanSelection = this.clearSelection;
+
+ this._postBack2 = function(action, data, parameters) {
+ var envelope = {};
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + DayPilot.JSON.stringify(envelope);
+ __doPostBack(calendar.uniqueID, commandstring);
+ };
+
+ this._callBack2 = function(action, data, parameters) {
+
+ if (this.callbackTimeout) {
+ window.clearTimeout(this.callbackTimeout);
+ }
+
+ this.callbackTimeout = window.setTimeout(function() {
+ calendar.loadingStart();
+ }, 100);
+
+ var envelope = {};
+
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + DayPilot.JSON.stringify(envelope);
+ if (this.backendUrl) {
+ DayPilot.request(this.backendUrl, this._callBackResponse, commandstring, this.ajaxError);
+ }
+ else if (typeof WebForm_DoCallback === 'function') {
+ WebForm_DoCallback(this.uniqueID, commandstring, this._updateView, this.clientName, this.onCallbackError, true);
+ }
+ };
+
+ this.onCallbackError = function(result, context) {
+ alert("Error!\r\nResult: " + result + "\r\nContext:" + context);
+ };
+
+ this.dispose = function() {
+
+
+ var c = calendar;
+
+ if (c._disposed) {
+ return;
+ }
+ c._disposed = true;
+
+ clearInterval(c._visibilityInterval);
+
+ c._deleteEvents();
+ c.nav.scroll.root = null;
+
+ DayPilot.pu(c.nav.loading);
+
+ c._disposeMain();
+ c._disposeHeader();
+
+ c.nav.select = null;
+ c.nav.cornerRight = null;
+ c.nav.scrollable = null;
+ c.nav.zoom = null;
+ c.nav.loading = null;
+ c.nav.header = null;
+ c.nav.hourTable = null;
+ c.nav.scrolltop = null;
+ c.nav.scroll.onscroll = null;
+ c.nav.scroll = null;
+ c.nav.main = null;
+ c.nav.message = null;
+ c.nav.messageClose = null;
+ c.nav.top = null;
+
+ DayPilotCalendar.unregister(c);
+ };
+
+ this.disposed = function() {
+ return this._disposed;
+ };
+
+ this._registerDispose = function() {
+ this.nav.top.dispose = this.dispose;
+ };
+
+ this._callBackResponse = function(response) {
+ calendar._updateView(response.responseText);
+ };
+
+ this._getCallBackHeader = function() {
+ var h = {};
+
+ h.control = "dpc";
+ h.id = this.id;
+ h.v = this.v;
+
+ h.days = calendar.days;
+ h.startDate = calendar.startDate;
+ h.heightSpec = calendar.heightSpec;
+ h.businessBeginsHour = calendar.businessBeginsHour;
+ h.businessEndsHour = calendar.businessEndsHour;
+ h.hashes = calendar.hashes;
+
+ // h.backColor = calendar.cellBackColor;
+ h.timeFormat = calendar.timeFormat;
+ h.viewType = calendar.viewType;
+ h.locale = calendar.locale;
+
+ return h;
+ };
+
+ this._createShadow = function(object, copyText) {
+ var parentTd = object.parentNode;
+ while (parentTd && parentTd.tagName !== "TD") {
+ parentTd = parentTd.parentNode;
+ }
+
+ var shadow = document.createElement('div');
+ shadow.setAttribute('unselectable', 'on');
+ shadow.style.position = 'absolute';
+ shadow.style.width = (object.offsetWidth) + 'px';
+ shadow.style.height = (object.offsetHeight) + 'px';
+ shadow.style.left = (object.offsetLeft) + 'px';
+ shadow.style.top = (object.offsetTop) + 'px';
+ shadow.style.boxSizing = "border-box";
+ shadow.style.zIndex = 101;
+
+ // shadow.style.backgroundColor = "#aaaaaa";
+ // shadow.style.opacity = 0.5;
+ // shadow.style.filter = "alpha(opacity=50)";
+ // shadow.style.border = '2px solid #aaaaaa';
+
+ shadow.className = calendar._prefixCssClass("_shadow");
+
+ var inner = document.createElement("div");
+ inner.className = calendar._prefixCssClass("_shadow_inner");
+ shadow.appendChild(inner);
+
+ if (copyText && false) { // disabled
+ shadow.style.overflow = 'hidden';
+ shadow.style.fontSize = object.style.fontSize;
+ shadow.style.fontFamily = object.style.fontFamily;
+ shadow.style.color = object.style.color;
+ shadow.innerHTML = object.data.client.html();
+ }
+
+ /*
+ shadow.style.MozBorderRadius = "5px";
+ shadow.style.webkitBorderRadius = "5px";
+ shadow.style.borderRadius = "5px";
+ */
+
+ parentTd.firstChild.appendChild(shadow);
+
+ return shadow;
+ };
+
+
+ this._resolved = {};
+ this._resolved.locale = function() {
+ var found = DayPilot.Locale.find(calendar.locale);
+ if (!found) {
+ return DayPilot.Locale.US;
+ }
+ return found;
+ };
+
+ this._resolved.timeFormat = function() {
+ if (calendar.timeFormat !== 'Auto') {
+ return calendar.timeFormat;
+ }
+ return this.locale().timeFormat;
+ };
+
+ this._resolved._xssProtectionEnabled = function() {
+ return calendar.xssProtection !== "Disabled";
+ };
+
+ this._resolved._weekStarts = function() {
+ if (calendar.weekStarts === 'Auto') {
+ var locale = resolved.locale();
+ if (locale) {
+ return locale.weekStarts;
+ }
+ else {
+ return 0; // Sunday
+ }
+ }
+ else {
+ return calendar.weekStarts || 0;
+ }
+ };
+
+ var resolved = this._resolved;
+
+ this._updateView = function(result, context) {
+
+ if (result && result.indexOf("$$$") === 0) {
+ if (window.console) {
+ console.log("Error received from the server side: " + result);
+ }
+ else {
+ throw "Error received from the server side: " + result;
+ }
+ return;
+ }
+
+ var result = JSON.parse(result);
+
+ if (result.CallBackRedirect) {
+ document.location.href = result.CallBackRedirect;
+ return;
+ }
+
+ if (result.UpdateType === "None") {
+ calendar.loadingStop();
+ calendar._show();
+ return;
+ }
+
+ calendar._deleteEvents();
+
+ if (result.UpdateType === "Full") {
+
+ calendar.columns = result.Columns;
+
+ // properties
+ calendar.days = result.Days;
+ calendar.startDate = new DayPilot.Date(result.StartDate);
+ calendar.heightSpec = result.HeightSpec ? result.HeightSpec : calendar.heightSpec;
+ calendar.businessBeginsHour = result.BusinessBeginsHour ? result.BusinessBeginsHour : calendar.businessBeginsHour;
+ calendar.businessEndsHour = result.BusinessEndsHour ? result.BusinessEndsHour : calendar.businessEndsHour;
+ calendar.headerDateFormat = result.HeaderDateFormat ? result.HeaderDateFormat : calendar.headerDateFormat;
+ calendar.viewType = result.ViewType; //
+ calendar.backColor = result.BackColor ? result.BackColor : calendar.backColor;
+ calendar.eventHeaderVisible = result.EventHeaderVisible ? result.EventHeaderVisible : calendar.eventHeaderVisible;
+ calendar.timeFormat = result.TimeFormat ? result.TimeFormat : calendar.timeFormat;
+ calendar.locale = result.Locale ? result.Locale : calendar.locale;
+
+ calendar._prepareColumns();
+ }
+
+ // hashes
+ if (result.Hashes) {
+ for (var key in result.Hashes) {
+ calendar.hashes[key] = result.Hashes[key];
+ }
+ }
+
+ calendar.events.list = result.Events;
+ calendar._loadEvents();
+ calendar._updateHeaderHeight();
+
+ if (result.UpdateType === "Full") {
+ calendar._drawHeader();
+ calendar._drawMain();
+ calendar._drawHourTable();
+ calendar._updateHeight();
+ }
+
+ calendar._show();
+
+ calendar._drawEvents();
+ calendar.clearSelection();
+
+ calendar.afterRender(result.CallBackData, true);
+
+ calendar.loadingStop();
+
+ };
+
+ this._durationHours = function() {
+ return this._duration() / (3600 * 1000);
+ };
+
+ this._businessHoursSpan = function() {
+ if (this.businessBeginsHour > this.businessEndsHour) {
+ return 24 - this.businessBeginsHour + this.businessEndsHour;
+ }
+ else {
+ return this.businessEndsHour - this.businessBeginsHour;
+ }
+ };
+
+ this._rowCount = function() {
+ return this._duration() / (60 * 60 * 1000 / 2);
+ };
+
+ // in ticks
+ this._duration = function() {
+ var dHours = 0;
+
+ if (this.heightSpec === 'BusinessHoursNoScroll') {
+ dHours = this._businessHoursSpan();
+ }
+ else {
+ dHours = 24;
+ }
+ return dHours * 60 * 60 * 1000; // return ticks
+ };
+
+ this._visibleStart = function() {
+ if (this.heightSpec === 'BusinessHoursNoScroll') {
+ return this.businessBeginsHour;
+ }
+ return 0;
+ };
+
+
+
+ this._api2 = function() {
+ return calendar.api === 2;
+ };
+
+ this.eventClickCallBack = function(e, data) {
+ this._callBack2('EventClick', data, e);
+ };
+
+ this.eventClickPostBack = function(e, data) {
+ this._postBack2('EventClick', data, e);
+ };
+
+ this._eventClickDispatch = function (ev) {
+ var thisDiv = this;
+
+ var e = thisDiv.event;
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.originalEvent = ev;
+ args.meta = ev.metaKey;
+ args.ctrl = ev.ctrlKey;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventClick === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClick(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventClickHandling) {
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'PostBack':
+ calendar.eventClickPostBack(e);
+ break;
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(e);
+ }
+ }
+ break;
+ }
+
+ if (typeof calendar.onEventClicked === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClicked(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventClickHandling) {
+ case 'PostBack':
+ calendar.eventClickPostBack(e);
+ break;
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventClick(e);
+ break;
+ }
+ }
+
+ };
+
+ this._eventRightClickDispatch = function(ev) {
+
+ var e = this.event;
+
+ if (ev.stopPropagation) {
+ ev.stopPropagation();
+ }
+
+ if (!e.client.rightClickEnabled()) {
+ return false;
+ }
+
+ var args = {};
+ args.e = e;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventRightClick === 'function') {
+ calendar.onEventRightClick(args);
+ if (args.preventDefault.value) {
+ return false;
+ }
+ }
+
+ switch (calendar.eventRightClickHandling) {
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(this.event);
+ }
+ }
+ break;
+ }
+
+ if (typeof calendar.onEventRightClicked === 'function') {
+ calendar.onEventRightClicked(args);
+ }
+
+
+ if (ev.preventDefault) {
+ ev.preventDefault();
+ }
+ return false;
+ };
+
+
+ this.eventDeleteCallBack = function(e, data) {
+ this._callBack2('EventDelete', data, e);
+ };
+
+ this.eventDeletePostBack = function(e, data) {
+ this._postBack2('EventDelete', data, e);
+ };
+
+ this._eventDeleteDispatch = function (e) {
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventDelete === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDelete(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventDeleteHandling) {
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'Update':
+ calendar.events.remove(e);
+ break;
+ }
+
+ if (typeof calendar.onEventDeleted === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDeleted(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventDeleteHandling) {
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventDelete(e);
+ break;
+ }
+ }
+
+ };
+
+ this.eventResizeCallBack = function(e, newStart, newEnd, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._callBack2('EventResize', data, params);
+ };
+
+ this.eventResizePostBack = function(e, newStart, newEnd, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._postBack2('EventResize', data, params);
+ };
+
+ this._eventResizeDispatch = function (e, shadowHeight, shadowTop, border ) {
+ var _startOffset = 1;
+
+ var newStart = new Date();
+ var newEnd = new Date();
+
+ var start = e.start();
+ var end = e.end();
+
+ if (border === 'top') {
+ var day = start.getDatePart();
+ var step = Math.floor((shadowTop - _startOffset) / calendar.cellHeight);
+ var minutes = step * 30;
+ var ts = minutes * 60 * 1000;
+ var visibleStartOffset = calendar._visibleStart() * 60 * 60 * 1000;
+
+ newStart = day.addTime(ts + visibleStartOffset);
+ newEnd = e.end();
+
+ }
+ else if (border === 'bottom') {
+ var day = end.getDatePart();
+ var step = Math.floor((shadowTop + shadowHeight - _startOffset) / calendar.cellHeight);
+ var minutes = step * 30;
+ var ts = minutes * 60 * 1000;
+ var visibleStartOffset = calendar._visibleStart() * 60 * 60 * 1000;
+
+ newStart = start;
+ newEnd = day.addTime(ts + visibleStartOffset);
+ }
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.control = calendar;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventResize === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResize(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventResizeHandling) {
+ case 'PostBack':
+ calendar.eventResizePostBack(e, newStart, newEnd);
+ break;
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventResized === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResized(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventResizeHandling) {
+ case 'PostBack':
+ calendar.eventResizePostBack(e, newStart, newEnd);
+ break;
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'JavaScript':
+ calendar.onEventResize(e, newStart, newEnd);
+ break;
+ }
+ }
+ };
+
+ this.eventMovePostBack = function(e, newStart, newEnd, newResource, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._postBack2('EventMove', data, params);
+ };
+
+ this.eventMoveCallBack = function(e, newStart, newEnd, newResource, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._callBack2('EventMove', data, params);
+ };
+
+ this._eventMoveDispatch = function (e, newColumnIndex, shadowTop, ev) {
+ var _startOffset = 1;
+ var step = Math.floor((shadowTop - _startOffset) / calendar.cellHeight);
+
+ var boxStart = step * 30 * 60 * 1000;
+ var start = e.start();
+ var end = e.end();
+ var day = new Date();
+
+ if (start instanceof DayPilot.Date) {
+ start = start.toDate();
+ }
+ day.setTime(Date.UTC(start.getUTCFullYear(), start.getUTCMonth(), start.getUTCDate()));
+
+ var startOffset = start.getTime() - (day.getTime() + start.getUTCHours() * 3600 *1000 + Math.floor(start.getUTCMinutes()/30)*30*60*1000 );
+ var length = end.getTime() - start.getTime();
+
+ var newColumn = this._columns[newColumnIndex];
+ var newResource = newColumn.id;
+
+ var date = newColumn.start.getTime();
+ var newStartUTC = new Date();
+ newStartUTC.setTime(date + boxStart + startOffset);
+
+ var newStart = new DayPilot.Date(newStartUTC);
+
+ var newEnd = newStart.addTime(length);
+
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.newResource = newResource;
+ args.ctrl = ev.ctrlKey;
+ args.shift = ev.shiftKey;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventMove === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMove(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventMoveHandling) {
+ case 'PostBack':
+ calendar.eventMovePostBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ e.resource(newResource);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventMoved === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMoved(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventMoveHandling) {
+ case 'PostBack':
+ calendar.eventMovePostBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'JavaScript':
+ calendar.onEventMove(e, newStart, newEnd, newColumn.id, false);
+ break;
+ }
+ }
+
+ };
+
+ this.timeRangeSelectedPostBack = function(start, end, resource, data) {
+
+ var range = {};
+ range.start = start;
+ range.end = end;
+
+ this._postBack2('TimeRangeSelected', data, range);
+ };
+
+ this.timeRangeSelectedCallBack = function(start, end, resource, data) {
+
+ var range = {};
+ range.start = start;
+ range.end = end;
+
+ this._callBack2('TimeRangeSelected', data, range);
+ };
+
+ this._timeRangeSelectedDispatch = function (start, end, resource) {
+
+ // make sure it's DayPilot.Date
+ start = new DayPilot.Date(start);
+ end = new DayPilot.Date(end);
+
+ if (this._api2()) {
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.resource = resource;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onTimeRangeSelect === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelect(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end);
+ break;
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ }
+
+ if (typeof calendar.onTimeRangeSelected === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelected(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end);
+ break;
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ case 'JavaScript':
+ calendar.onTimeRangeSelected(start, end);
+ break;
+ }
+ }
+ };
+
+ this._onCellMousedown = function(ev) {
+
+ if (DayPilotCalendar.selecting) {
+ return;
+ }
+
+ if (calendar.timeRangeSelectedHandling === "Disabled") {
+ return;
+ }
+
+ var button = ev.which;
+ if (button !== 1 && button !== 0) { // Khtml says first button is 0
+ return;
+ }
+
+ DayPilotCalendar.firstMousePos = DayPilot.mc(ev);
+ DayPilotCalendar.selecting = {};
+ DayPilotCalendar.selecting.calendar = calendar;
+ if (DayPilotCalendar.selectedCells) {
+ calendar.clearSelection();
+ DayPilotCalendar.selectedCells = [];
+ }
+ DayPilotCalendar.column = DayPilotCalendar.getColumn(this);
+ DayPilotCalendar.selectedCells.push(this);
+ DayPilotCalendar.firstSelected = this;
+
+ DayPilotCalendar.topSelectedCell = this;
+ DayPilotCalendar.bottomSelectedCell = this;
+
+ calendar._activateSelection();
+
+ return false;
+ };
+
+ this._activateSelection = function() {
+ var selection = this.getSelection();
+
+ (function activateSelectionNew() {
+
+ var first = DayPilotCalendar.topSelectedCell;
+ var last = DayPilotCalendar.bottomSelectedCell;
+
+ // var columnIndex = first.parentNode.cells.indexOf(first);
+
+ var columnIndex = (function() {
+ // it's the new div cell
+ if (first.data) {
+ return first.data.x;
+ }
+
+ var cells = first.parentNode.cells;
+ for (var i = 0; i < cells.length; i++) {
+ if (cells[i] === first) {
+ return i;
+ }
+ }
+ return -1;
+ })();
+
+ var col = calendar._columns[columnIndex];
+
+ if (!col) {
+ return;
+ }
+
+ var colStart = col.start;
+
+ var top = calendar.getPixels(first.start, colStart).boxTop;
+ var bottom = calendar.getPixels(last.end, colStart).boxBottom;
+ var height = bottom - top;
+
+ var div = (function() {
+ if (calendar.nav.activeSelection) {
+ return calendar.nav.activeSelection;
+ }
+ var div = document.createElement("div");
+ div.setAttribute("unselectable", "on");
+ // div.style.zIndex = calendar._shadowZindex;
+ div.style.position = "absolute";
+ div.style.left = "0px";
+ div.style.width = "100%";
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = calendar._prefixCssClass("_shadow_inner");
+
+ div.appendChild(inner);
+
+ calendar.nav.events.rows[0].cells[columnIndex].selection.appendChild(div);
+ calendar.elements.selection.push(div);
+ calendar.nav.activeSelection = div;
+
+ return div;
+ })();
+
+ // reset
+ div.className = calendar._prefixCssClass("_shadow");
+ div.firstChild.innerHTML = "";
+
+ // position
+ div.style.top = top + "px";
+ div.style.height = height + "px";
+
+ })();
+ };
+
+ this._mousemove = function(ev) {
+
+ if (typeof (DayPilotCalendar) === 'undefined') {
+ return;
+ }
+
+ if (!DayPilotCalendar.selecting) {
+ return;
+ }
+
+ var mousePos = DayPilot.mc(ev);
+
+ var thisColumn = DayPilotCalendar.getColumn(this);
+ if (thisColumn !== DayPilotCalendar.column) {
+ return;
+ }
+
+ // clean
+ calendar.clearSelection();
+
+ // new selected cells
+ if (mousePos.y < DayPilotCalendar.firstMousePos.y) {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsBelow(this);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.selectedCells[0];
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.firstSelected;
+ }
+ else {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsAbove(this);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.firstSelected;
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.selectedCells[0];
+ }
+
+ calendar._activateSelection();
+ };
+
+ this.getSelection = function() {
+ var start = DayPilotCalendar.topSelectedCell.start;
+ var end = DayPilotCalendar.bottomSelectedCell.end;
+ var resource = DayPilotCalendar.topSelectedCell.resource;
+
+ return new DayPilot.Selection(start, end, resource, calendar);
+ };
+
+ this._getColumnForPixels = function(x) {
+
+ if (x < 0) {
+ return null;
+ }
+
+ var i = 0;
+ var cells = calendar.nav.events.rows[0].cells;
+
+ for (var j = 0; j < cells.length; j++) {
+ var cell = cells[j];
+ var width = cell.offsetWidth;
+ i += width;
+ if (x < i) {
+ return j;
+ }
+ }
+ return null;
+ };
+
+ this._table = {};
+ this._table.getCellCoords = function() {
+ var result = {};
+ result.x = 0;
+ result.y = 0;
+
+ if (!calendar.coords) {
+ return null;
+ }
+
+ result.x = calendar._getColumnForPixels(calendar.coords.x);
+
+ var _startOffset = 0;
+ var row = Math.floor((calendar.coords.y - _startOffset) / calendar.cellHeight);
+ result.y = row;
+
+ if (result.x < 0) {
+ return null;
+ }
+
+ return result;
+ };
+
+ this.columns = {};
+ this.columns.list = [];
+
+ this.columns.load = function(url, success, error) {
+
+ if (!url) {
+ throw new DayPilot.Exception("columns.load(): 'url' parameter required");
+ }
+
+ var onError = function(args) {
+ var largs = {};
+ largs.exception = args.exception;
+ largs.request = args.request;
+
+ if (typeof error === 'function') {
+ error(largs);
+ }
+ };
+
+ var onSuccess = function(args) {
+ var r = args.request;
+ var data;
+
+ // it's supposed to be JSON
+ try {
+ data = JSON.parse(r.responseText);
+ }
+ catch (e) {
+ var fargs = {};
+ fargs.exception = e;
+ onError(fargs);
+ return;
+ }
+
+ if (DayPilot.isArray(data)) {
+ var sargs = {};
+ sargs.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+ sargs.data = data;
+ if (typeof success === "function") {
+ success(sargs);
+ }
+
+ if (sargs.preventDefault.value) {
+ return;
+ }
+
+ calendar.columns.list = data;
+ if (calendar._initialized) {
+ calendar.update();
+ }
+ }
+ };
+
+ var usePost = calendar.columnsLoadMethod && calendar.columnsLoadMethod.toUpperCase() === "POST";
+
+ if (usePost) {
+ DayPilot.ajax({
+ "method": "POST",
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ else {
+ DayPilot.ajax({
+ "method": "GET",
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ };
+
+
+ this._prepareColumns = function() {
+ var columns;
+ if (calendar.viewType !== "Resources") {
+ columns = this._createDaysViewColumns();
+ }
+ else {
+ columns = calendar.columns.list;
+ }
+
+ this._columns = [];
+ for (var i = 0; i < columns.length; i++) {
+ var c = this._activateColumn(columns[i]);
+ this._columns.push(c);
+ }
+ };
+
+ this._activateColumn = function(column) {
+
+/*
+ if (column.Start) {
+ column.start = column.Start;
+ column.html = column.InnerHTML;
+ column.name = column.Name;
+ }
+*/
+ var result = {};
+ result.name = column.name;
+ result.html = column.html;
+ result.id = column.id;
+ result.toolTip = column.toolTip;
+ result.data = column;
+
+ if (column.start) {
+ result.start = new DayPilot.Date(column.start);
+ }
+ else {
+ result.start = new DayPilot.Date(calendar.startDate);
+ }
+
+ result.putIntoBlock = function(ep) {
+
+ for (var i = 0; i < this.blocks.length; i++) {
+ var block = this.blocks[i];
+ if (block.overlapsWith(ep.part.top, ep.part.height)) {
+ block.events.push(ep);
+ block.min = Math.min(block.min, ep.part.top);
+ block.max = Math.max(block.max, ep.part.top + ep.part.height);
+ return i;
+ }
+ }
+
+ // no suitable block found, create a new one
+ var block = [];
+ block.lines = [];
+ block.events = [];
+
+ block.overlapsWith = function(start, width) {
+ var end = start + width - 1;
+
+ if (!(end < this.min || start > this.max - 1)) {
+ return true;
+ }
+
+ return false;
+ };
+ block.putIntoLine = function (ep) {
+ var thisCol = this;
+
+ for (var i = 0; i < this.lines.length; i++) {
+ var line = this.lines[i];
+ if (line.isFree(ep.part.top, ep.part.height)) {
+ line.push(ep);
+ return i;
+ }
+ }
+
+ var line = [];
+ line.isFree = function(start, width) {
+ var end = start + width - 1;
+ var max = this.length;
+
+ for (var i = 0; i < max; i++) {
+ var e = this[i];
+ if (!(end < e.part.top || start > e.part.top + e.part.height - 1)) {
+ return false;
+ }
+ }
+
+ return true;
+ };
+
+ line.push(ep);
+
+ this.lines.push(line);
+
+ return this.lines.length - 1;
+
+ };
+
+ block.events.push(ep);
+ block.min = ep.part.top;
+ block.max = ep.part.top + ep.part.height;
+
+ this.blocks.push(block);
+
+ return this.blocks.length - 1;
+
+ };
+
+ result.putIntoLine = function(ep) {
+ var thisCol = this;
+
+ for (var i = 0; i < this.lines.length; i++) {
+ var line = this.lines[i];
+ if (line.isFree(ep.part.top, ep.part.height)) {
+ line.push(ep);
+ return i;
+ }
+ }
+
+ var line = [];
+ line.isFree = function(start, width) {
+ var end = start + width - 1;
+ var max = this.length;
+
+ for (var i = 0; i < max; i++) {
+ var e = this[i];
+ if (!(end < e.part.top || start > e.part.top + e.part.height - 1)) {
+ return false;
+ }
+ }
+
+ return true;
+ };
+
+ line.push(ep);
+
+ this.lines.push(line);
+
+ return this.lines.length - 1;
+ };
+
+ return result;
+
+ };
+
+ this._createDaysViewColumns = function() {
+ var columns = [];
+
+ var start = this.startDate.getDatePart();
+
+ var days = this.days;
+ switch (this.viewType) {
+ case "Day":
+ days = 1;
+ break;
+ case "Week":
+ days = 7;
+ var weekStarts = resolved._weekStarts();
+ start = start.firstDayOfWeek(weekStarts);
+ break;
+ case "WorkWeek":
+ days = 5;
+ start = start.firstDayOfWeek(1); // Monday
+ break;
+ }
+
+ if (this.heightSpec === 'BusinessHoursNoScroll') {
+ start = start.addHours(this.businessBeginsHour);
+ }
+
+ for (var i = 0; i < days; i++) {
+
+ var format = calendar.headerDateFormat ? calendar.headerDateFormat : resolved.locale().datePattern;
+
+ var column = {};
+ column.start = start.addDays(i);
+ column.name = column.start.toString(format, resolved.locale());
+
+ columns.push(column);
+ }
+
+ return columns;
+ };
+
+ this.visibleStart = function() {
+ if (calendar.viewType === "Resources") {
+ if (calendar._columns.length === 0) {
+ return DayPilot.Date.today();
+ }
+ var dates = calendar._columns.map(function(column) {
+ return column.start.getTime();
+ });
+ var min = Math.min.apply(null, dates);
+ return new DayPilot.Date(min);
+ }
+ return this._columns[0].start;
+ };
+
+ this.visibleEnd = function() {
+ if (calendar.viewType === "Resources") {
+ if (calendar._columns.length === 0) {
+ return DayPilot.Date.today().addDays(1);
+ }
+ var dates = calendar._columns.map(function(column) {
+ return column.start.getTime();
+ });
+ var max = Math.max.apply(null, dates);
+ return new DayPilot.Date(max).addDays(1);
+ }
+ var max = this._columns.length - 1;
+ return this._columns[max].start.addDays(1);
+ };
+
+
+ this._prefixCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ if (prefix) {
+ return prefix + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this._deleteEvents = function() {
+
+ if (this.elements.events) {
+
+ for (var i = 0; i < this.elements.events.length; i++) {
+ var div = this.elements.events[i];
+
+ var object = div.event;
+ if (object) {
+ object.calendar = null;
+ }
+
+ div.onclick = null;
+ div.onclickSave = null;
+ div.onmouseover = null;
+ div.onmouseout = null;
+ div.onmousemove = null;
+ div.onmousedown = null;
+
+ if (div.firstChild && div.firstChild.firstChild && div.firstChild.firstChild.tagName && div.firstChild.firstChild.tagName.toUpperCase() === 'IMG') {
+ var img = div.firstChild.firstChild;
+ img.onmousedown = null;
+ img.onmousemove = null;
+ img.onclick = null;
+
+ }
+
+ div.helper = null;
+ div.data = null;
+ div.event = null;
+
+ DayPilot.de(div);
+ }
+ }
+ this.elements.events = [];
+ };
+
+ this._drawEvent = function(e) {
+ /*
+
+ supported e.data properties:
+ * text
+ * html
+ * cssClass
+ * backColor
+ * barColor
+ * barBackColor
+ * toolTip
+ * barHidden
+ * borderColor
+ * fontColor
+
+ */
+
+
+ var data = e.cache || e.data;
+
+ var main = this.nav.events;
+
+ var rounded = true;
+ var radius = true;
+ var pixels = false;
+
+ // var borderColor = this.eventBorderColor;
+
+ var div = document.createElement("div");
+ div.style.position = 'absolute';
+ div.style.left = e.part.left + '%';
+ div.style.top = (e.part.top) + 'px';
+ div.style.width = e.part.width + '%';
+ div.style.height = Math.max(e.part.height, 2) + 'px';
+ div.style.overflow = 'hidden';
+
+ // TODO unify
+ div.data = e;
+ div.event = e;
+
+ div.unselectable = 'on';
+ div.style.MozUserSelect = 'none';
+ div.style.KhtmlUserSelect = 'none';
+
+ div.className = this._prefixCssClass("_event");
+
+ if (data.cssClass) {
+ DayPilot.Util.addClass(div, data.cssClass);
+ }
+
+ if (calendar.showToolTip && e.client.toolTip()) {
+ div.title = e.client.toolTip();
+ }
+
+ div.isFirst = e.part.start.getTime() === e.start().getTime();
+ div.isLast = e.part.end.getTime() === e.end().getTime();
+
+ div.onclick = this._eventClickDispatch;
+ DayPilot.re(div, "contextmenu", this._eventRightClickDispatch);
+
+ div.onmouseout = function(ev) {
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "none";
+ }
+ };
+
+ div.onmousemove = function (ev) {
+ // const
+ var resizeMargin = 5;
+
+ if (typeof(DayPilotCalendar) === 'undefined') {
+ return;
+ }
+
+ // position
+ var offset = DayPilot.mo3(div, ev);
+ if (!offset) {
+ return;
+ }
+
+ if (DayPilotCalendar.resizing || DayPilotCalendar.moving) {
+ return;
+ }
+
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "";
+ }
+
+ var isLastPart = this.isLast;
+
+ if (offset.y <= resizeMargin && e.client.resizeEnabled()) {
+ this.style.cursor = "n-resize";
+ this.dpBorder = 'top';
+ }
+ else if (this.offsetHeight - offset.y <= resizeMargin && e.client.resizeEnabled()) {
+ if (isLastPart) {
+ this.style.cursor = "s-resize";
+ this.dpBorder = 'bottom';
+ }
+ else {
+ this.style.cursor = 'not-allowed';
+ }
+ }
+ else if (!DayPilotCalendar.resizing && !DayPilotCalendar.moving) {
+ if (calendar.eventClickHandling !== 'Disabled') {
+ this.style.cursor = 'pointer';
+ }
+ else {
+ this.style.cursor = 'default';
+ }
+ }
+
+ };
+
+ div.onmousedown = function (ev) {
+ var button = ev.which || ev.button;
+
+ if ((this.style.cursor === 'n-resize' || this.style.cursor === 's-resize') && button === 1) {
+ // set
+ DayPilotCalendar.resizing = this;
+ DayPilot.Global.resizing = this;
+ DayPilotCalendar.originalMouse = DayPilot.mc(ev);
+ DayPilotCalendar.originalHeight = this.offsetHeight;
+ DayPilotCalendar.originalTop = this.offsetTop;
+
+ calendar.nav.top.style.cursor = this.style.cursor;
+
+ }
+ else if (button === 1 && e.client.moveEnabled()) {
+ DayPilotCalendar.moving = this;
+ DayPilot.Global.moving = this;
+ DayPilotCalendar.moving.event = this.event;
+ var helper = DayPilotCalendar.moving.helper = {};
+ helper.oldColumn = calendar._columns[this.data.part.dayIndex].id;
+ DayPilotCalendar.originalMouse = DayPilot.mc(ev);
+ DayPilotCalendar.originalTop = this.offsetTop;
+
+ var offset = DayPilot.mo3(this, ev);
+ if (offset) {
+ DayPilotCalendar.moveOffsetY = offset.y;
+ }
+ else {
+ DayPilotCalendar.moveOffsetY = 0;
+ }
+
+ // cursor
+ //document.body.style.cursor = 'move';
+ calendar.nav.top.style.cursor = 'move';
+ }
+
+ return false;
+ };
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = calendar._prefixCssClass("_event_inner");
+ inner.innerHTML = e.client.html();
+
+ if (data.borderColor === "darker" && data.backColor) {
+ inner.style.borderColor = DayPilot.ColorUtil.darker(data.backColor, 2);
+ }
+ else {
+ inner.style.borderColor = data.borderColor;
+ }
+
+ if (data.backColor) {
+ inner.style.background = data.backColor;
+ }
+
+ if (data.fontColor) {
+ inner.style.color = data.fontColor;
+ }
+
+ div.appendChild(inner);
+
+ // TODO
+ if (e.client.barVisible()) {
+ var height = e.part.height - 2;
+ var barTop = 100 * e.part.barTop / height; // %
+ var barHeight = Math.ceil(100 * e.part.barHeight / height); // %
+
+ var bar = document.createElement("div");
+ bar.setAttribute("unselectable", "on");
+ bar.className = this._prefixCssClass("_event_bar");
+ bar.style.position = "absolute";
+
+ if (data.barBackColor) {
+ bar.style.backgroundColor = data.barBackColor;
+ }
+
+ var barInner = document.createElement("div");
+ barInner.setAttribute("unselectable", "on");
+ barInner.className = this._prefixCssClass("_event_bar_inner");
+ barInner.style.top = barTop + "%";
+ if (0 < barHeight && barHeight <= 1) {
+ barInner.style.height = "1px";
+ }
+ else {
+ barInner.style.height = barHeight + "%";
+ }
+
+ if (data.barColor) {
+ barInner.style.backgroundColor = data.barColor;
+ }
+
+ bar.appendChild(barInner);
+ div.appendChild(bar);
+ }
+
+ if (e.client.deleteEnabled()) {
+ var del = document.createElement("div");
+ del.style.position = "absolute";
+ del.style.right = "2px";
+ del.style.top = "2px";
+ del.style.width = "17px";
+ del.style.height = "17px";
+ del.className = calendar._prefixCssClass("_event_delete");
+ del.onmousedown = function(ev) {
+ ev.stopPropagation();
+ };
+ del.onclick = function(ev) {
+ ev.stopPropagation();
+ var e = this.parentNode.event;
+ if (e) {
+ calendar._eventDeleteDispatch(e);
+ }
+ };
+ del.style.display = "none";
+ div.deleteIcon = del;
+ div.appendChild(del);
+ }
+
+ // areas
+ var areas = data.areas ? DayPilot.Areas.copy(data.areas) : [];
+
+/* var col = calendar._columnsBottom[e.part.dayIndex];
+
+ areas.forEach(function(area) {
+ if (area.start) {
+ area.top = calendar._getPixels(new DayPilot.Date(area.start), col.start).top - e.part.top - calendar._autoHiddenPixels();
+ }
+ if (area.end) {
+ area.bottom = e.part.top + e.part.height - calendar._getPixels(new DayPilot.Date(area.end), col.start).top - calendar._autoHiddenPixels();
+ }
+ });
+
+ if (e.client.deleteEnabled()) {
+ areas.push({"action":"JavaScript","v":"Hover","w":17,"h":17,"top":2,"right":2, "css": calendar._prefixCssClass("_event_delete"),"js":function(e) { calendar._eventDeleteDispatch(e); } });
+ }*/
+
+ DayPilot.Areas.attach(div, e, {"areas": areas});
+
+ if (main.rows[0].cells[e.part.dayIndex]) { // temporary fix for multirow header, but won't hurt later
+ var wrapper = main.rows[0].cells[e.part.dayIndex].firstChild;
+ wrapper.appendChild(div);
+
+ calendar._makeChildrenUnselectable(div);
+
+ //var e = new DayPilotCalendar.Event(div, calendar);
+ }
+
+ if (typeof calendar.onAfterEventRender === 'function') {
+ var args = {};
+ args.e = div.event;
+ args.div = div;
+
+ calendar.onAfterEventRender(args);
+ }
+
+ calendar.elements.events.push(div);
+ };
+
+ this._makeChildrenUnselectable = function(el) {
+ var c = (el && el.childNodes) ? el.childNodes.length : 0;
+ for (var i = 0; i < c; i++) {
+ try {
+ var child = el.childNodes[i];
+ if (child.nodeType === 1) {
+ child.unselectable = 'on';
+ this._makeChildrenUnselectable(child);
+ }
+ }
+ catch (e) {
+ }
+ }
+ };
+
+ this._drawEvents = function() {
+
+ //var start = new Date();
+
+ for (var i = 0; i < this._columns.length; i++) {
+ var col = this._columns[i];
+ if (!col.blocks) {
+ continue;
+ }
+
+ for (var m = 0; m < col.blocks.length; m++) {
+ var block = col.blocks[m];
+ for (var j = 0; j < block.lines.length; j++) {
+ var line = block.lines[j];
+
+ for(var k = 0; k < line.length; k++) {
+ var e = line[k];
+
+ e.part.width = 100 / block.lines.length;
+ e.part.left = e.part.width * j;
+
+ var isLastBlock = (j === block.lines.length - 1);
+ if (!isLastBlock) {
+ e.part.width = e.part.width * 1.5;
+ }
+
+ this._drawEvent(e);
+ }
+ }
+ }
+ }
+
+ //var end = new Date();
+ //var diff = end.getTime() - start.getTime();
+ };
+
+ this._drawTop = function() {
+
+ //this.nav.top = document.getElementById(this.id);
+ this.nav.top.innerHTML = '';
+
+ DayPilot.Util.addClass(this.nav.top, this._prefixCssClass("_main"));
+
+ this.nav.top.style.MozUserSelect = 'none';
+ this.nav.top.style.KhtmlUserSelect = 'none';
+ this.nav.top.style.position = 'relative';
+ this.nav.top.style.width = this.width ? this.width : '100%';
+
+ if (this.hideUntilInit) {
+ this.nav.top.style.visibility = 'hidden';
+ }
+
+ if (!this.visible) {
+ this.nav.top.style.display = "none";
+ }
+
+ this.nav.scroll = document.createElement("div");
+ this.nav.scroll.style.height = this._getScrollableHeight() + "px";
+
+ if (this.heightSpec === 'BusinessHours') {
+ this.nav.scroll.style.overflow = "auto";
+ }
+ else
+ {
+ this.nav.scroll.style.overflow = "hidden";
+ }
+
+ this.nav.scroll.style.position = "relative";
+
+ var header = this._drawTopHeaderDiv();
+ this.nav.top.appendChild(header);
+
+ // fixing the column alignment bug
+ // solved thanks to http://stackoverflow.com/questions/139000/div-with-overflowauto-and-a-100-wide-table-problem
+ this.nav.scroll.style.zoom = 1;
+
+ var wrap = this._drawScrollable();
+ this.nav.scrollable = wrap.firstChild;
+ this.nav.scroll.appendChild(wrap);
+ this.nav.top.appendChild(this.nav.scroll);
+
+ this.nav.scrollLayer = document.createElement("div");
+ this.nav.scrollLayer.style.position = 'absolute';
+ this.nav.scrollLayer.style.top = '0px';
+ this.nav.scrollLayer.style.left = '0px';
+ this.nav.top.appendChild(this.nav.scrollLayer);
+
+ this.nav.loading = document.createElement("div");
+ this.nav.loading.style.position = 'absolute';
+ this.nav.loading.style.top = '0px';
+ this.nav.loading.style.left = (this.hourWidth + 5) + "px";
+ // this.nav.loading.style.backgroundColor = this.loadingLabelBackColor;
+ // this.nav.loading.style.fontSize = this.loadingLabelFontSize;
+ // this.nav.loading.style.fontFamily = this.loadingLabelFontFamily;
+ // this.nav.loading.style.color = this.loadingLabelFontColor;
+ // this.nav.loading.style.padding = '2px';
+ this.nav.loading.innerHTML = calendar._xssTextHtml(calendar.loadingLabelText, calendar.loadingLabelHtml);
+ this.nav.loading.style.display = 'none';
+
+ this.nav.top.appendChild(this.nav.loading);
+
+ };
+
+ // used during full update
+ this._drawHourTable = function() {
+ // clear old hour table
+ if (!this.fasterDispose) DayPilot.pu(this.nav.hourTable);
+ this.nav.scrollable.rows[0].cells[0].innerHTML = '';
+ this.nav.hourTable = this._createHourTable();
+ this.nav.scrollable.rows[0].cells[0].appendChild(this.nav.hourTable);
+ };
+
+ // used during initial load only
+ this._drawScrollable = function() {
+ var zoom = document.createElement("div");
+ zoom.style.zoom = 1;
+ zoom.style.position = 'relative';
+ // zoom.onmousemove = calendar._mousemove;
+
+ var table = document.createElement("table");
+
+ table.cellSpacing = "0";
+ table.cellPadding = "0";
+ table.border = "0";
+ table.style.border = "0px none";
+ table.style.width = "100%";
+ table.style.position = 'absolute';
+
+ var r = table.insertRow(-1);
+
+ var c;
+ c = r.insertCell(-1);
+ c.valign = "top";
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ this.nav.hourTable = this._createHourTable();
+ c.appendChild(this.nav.hourTable);
+
+ c = r.insertCell(-1);
+ c.valign = "top";
+ c.width = "100%";
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ var wrap = document.createElement("div");
+ wrap.style.position = "relative";
+ c.appendChild(wrap);
+
+ wrap.appendChild(this._createEventsAndCells());
+ wrap.appendChild(this._createEventsTable());
+
+ zoom.appendChild(table);
+
+ this.nav.zoom = zoom;
+
+ return zoom;
+ };
+
+ this._createEventsAndCells = function() {
+ var table = document.createElement("table");
+
+ table.cellPadding = "0";
+ table.cellSpacing = "0";
+ table.border = "0";
+ table.style.width = "100%";
+ table.style.border = "0px none";
+ table.style.tableLayout = 'fixed';
+
+ this.nav.main = table;
+ this.nav.events = table;
+
+ return table;
+ };
+
+ this._createEventsTable = function() {
+ var table = document.createElement("table");
+
+ //table.style.position = "absolute";
+ table.style.top = "0px";
+ table.cellPadding = "0";
+ table.cellSpacing = "0";
+ table.border = "0";
+ table.style.position = "absolute";
+ table.style.width = "100%";
+ table.style.border = "0px none";
+ table.style.tableLayout = 'fixed';
+ //table.setAttribute("events", "true");
+
+ this.nav.events = table;
+ var create = true;
+ var columns = this._columns;
+ var cl = columns.length;
+
+ var r = (create) ? table.insertRow(-1) : table.rows[0];
+
+ for (var j = 0; j < cl; j++) {
+ var c = (create) ? r.insertCell(-1) : r.cells[j];
+
+ if (create) {
+
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+ c.style.height = '0px';
+ c.style.overflow = 'visible';
+ if (!calendar.rtl) {
+ c.style.textAlign = 'left';
+ }
+
+ // withpct
+ //c.style.width = (100.0 / columns.length) + "%";
+
+ var div = document.createElement("div");
+ div.style.marginRight = calendar.columnMarginRight + "px";
+ div.style.position = 'relative';
+ div.style.height = '1px';
+ div.style.marginTop = '-1px';
+
+ var selection = document.createElement("div");
+ c.selection = selection;
+
+ c.appendChild(div);
+ c.appendChild(selection);
+
+ }
+ }
+
+ return table;
+ };
+
+ this._createHourTable = function() {
+ var table = document.createElement("table");
+ table.cellSpacing = "0";
+ table.cellPadding = "0";
+ table.border = "0";
+ table.style.border = '0px none';
+ table.style.width = this.hourWidth + "px";
+ table.oncontextmenu = function() { return false; };
+
+ var hours = calendar._durationHours();
+ for (var i = 0; i < hours; i++) {
+ this._createHourRow(table, i);
+ }
+
+ return table;
+
+ };
+
+ this._createHourRow = function(table, i) {
+ var height = (this.cellHeight * 2);
+
+ var r = table.insertRow(-1);
+ r.style.height = height + "px";
+
+ var c = r.insertCell(-1);
+ c.valign = "bottom";
+ c.unselectable = "on";
+ c.style.cursor = "default";
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ var frame = document.createElement("div");
+ frame.style.position = "relative";
+ frame.className = this._prefixCssClass("_rowheader");
+ frame.style.width = this.hourWidth + "px";
+ frame.style.height = (height) + "px";
+ frame.style.overflow = 'hidden';
+ frame.unselectable = 'on';
+
+ var block = document.createElement("div");
+ block.className = this._prefixCssClass("_rowheader_inner");
+ block.unselectable = "on";
+
+ var text = document.createElement("div");
+ text.unselectable = "on";
+
+ var start = this.startDate.addHours(i).addHours(calendar._visibleStart());
+ var hour = start.getHours();
+
+ var am = hour < 12;
+ var timeFormat = resolved.timeFormat();
+ if (timeFormat === "Clock12Hours") {
+ hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ }
+
+ text.innerHTML = hour;
+
+ var span = document.createElement("span");
+ span.unselectable = "on";
+ span.className = this._prefixCssClass("_rowheader_minutes");
+
+ var sup;
+ if (timeFormat === "Clock12Hours") {
+ if (am) {
+ sup = "AM";
+ }
+ else {
+ sup = "PM";
+ }
+ }
+ else {
+ sup = "00";
+ }
+
+ span.innerHTML = sup;
+
+ text.appendChild(span);
+
+ block.appendChild(text);
+
+ frame.appendChild(block);
+
+ c.appendChild(frame);
+ };
+
+ this._getScrollableHeight = function() {
+ switch (this.heightSpec) {
+ case "Full":
+ return (24 * 2 * this.cellHeight);
+ case "BusinessHours":
+ var dHours = this._businessHoursSpan();
+ return dHours * this.cellHeight * 2;
+ case "BusinessHoursNoScroll":
+ var dHours = this._businessHoursSpan();
+ return dHours * this.cellHeight * 2;
+ default:
+ throw "DayPilot.Calendar: Unexpected 'heightSpec' value.";
+
+ }
+ };
+
+ this._updateCorner = function() {
+ var parent = calendar.nav.corner ? calendar.nav.corner.parentNode : null;
+ if (!parent) {
+ return;
+ }
+
+ parent.innerHTML = '';
+
+ var corner = this._drawCorner();
+ parent.appendChild(corner);
+ calendar.nav.corner = corner;
+
+ };
+
+ this._drawTopHeaderDiv = function() {
+ var header = document.createElement("div");
+ header.style.overflow = "auto";
+
+ var table = document.createElement("table");
+ table.cellPadding = "0";
+ table.cellSpacing = "0";
+ table.border = "0";
+ table.style.width = "100%";
+ table.style.borderCollapse = 'separate';
+ table.style.border = "0px none";
+
+ var r = table.insertRow(-1);
+
+ // corner
+ var c = r.insertCell(-1);
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ var corner = this._drawCorner();
+ c.appendChild(corner);
+ this.nav.corner = corner;
+
+ // top header
+ c = r.insertCell(-1);
+
+ c.style.width = "100%";
+ c.valign = "top";
+ c.style.position = 'relative'; // ref point
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ this.nav.header = document.createElement("table");
+ this.nav.header.cellPadding = "0";
+ this.nav.header.cellSpacing = "0";
+ this.nav.header.border = "0";
+ this.nav.header.width = "100%";
+ this.nav.header.style.tableLayout = "fixed";
+ this.nav.header.oncontextmenu = function() { return false; };
+
+ var scrollbar = this.nav.scroll.style.overflow !== 'hidden';
+ c.appendChild(this.nav.header);
+
+ if (scrollbar) {
+ c = r.insertCell(-1);
+ c.unselectable = "on";
+
+ var inside = document.createElement("div");
+ inside.unselectable = "on";
+ inside.style.position = "relative";
+ inside.style.width = "16px";
+ inside.style.height = this.headerHeight + "px";
+ inside.className = this._prefixCssClass("_cornerright");
+
+ var inner = document.createElement("div");
+ inner.className = this._prefixCssClass('_cornerright_inner');
+ inside.appendChild(inner);
+
+ c.appendChild(inside);
+
+ this.nav.cornerRight = inside;
+ }
+
+ header.appendChild(table);
+
+ return header;
+
+ };
+
+ this._drawCorner = function() {
+ var wrap = document.createElement("div");
+ wrap.style.position = 'relative';
+ wrap.className = this._prefixCssClass("_corner");
+ wrap.style.width = this.hourWidth + "px";
+ wrap.style.height = this.headerHeight + "px";
+ wrap.oncontextmenu = function() { return false; };
+
+ var corner = document.createElement("div");
+ corner.unselectable = "on";
+ corner.className = this._prefixCssClass("_corner_inner");
+
+ wrap.appendChild(corner);
+
+ return wrap;
+ };
+
+ this._disposeMain = function() {
+ var table = this.nav.main;
+ table.root = null;
+ table.onmouseup = null;
+
+ for (var y = 0; y < table.rows.length; y++) {
+ var r = table.rows[y];
+ for (var x = 0; x < r.cells.length; x++) {
+ var c = r.cells[x];
+ c.root = null;
+
+ c.onmousedown = null;
+ c.onmousemove = null;
+ c.onmouseout = null;
+ c.onmouseup = null;
+ }
+ }
+
+ if (!this.fasterDispose) DayPilot.pu(table);
+ };
+
+ // draw time cells
+ this._drawMain = function() {
+
+ //DayPilotCalendar.selectedCells = [];
+ var cols = [];
+ var dates = [];
+
+ var table = this.nav.main;
+ var step = 30 * 60 * 1000;
+ var rowCount = this._rowCount();
+
+ var columns = calendar._columns;
+ //var create = !this.tableCreated || columns.length !== table.rows[0].cells.length || rowCount !== table.rows.length; // redraw only if number of columns changes
+ var create = true;
+
+ if (table) {
+ this._disposeMain();
+ }
+
+ while (table && table.rows && table.rows.length > 0 && create) {
+ if (!this.fasterDispose) {
+ DayPilot.pu(table.rows[0]);
+ }
+ table.deleteRow(0);
+ }
+
+ this.tableCreated = true;
+
+ var cl = columns.length;
+
+ var events = this.nav.events;
+ while (events && events.rows && events.rows.length > 0 && create) {
+ if (!this.fasterDispose) {
+ DayPilot.pu(events.rows[0]);
+ }
+ events.deleteRow(0);
+ }
+
+ var cl = columns.length;
+
+ var r = (create) ? events.insertRow(-1) : events.rows[0];
+
+ for (var j = 0; j < cl; j++) {
+ var c = (create) ? r.insertCell(-1) : r.cells[j];
+
+ if (create) {
+
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+ c.style.height = '0px';
+ c.style.overflow = 'visible';
+ if (!calendar.rtl) {
+ c.style.textAlign = 'left';
+ }
+
+ // withpct
+ //c.style.width = (100.0 / columns.length) + "%";
+
+ var div = document.createElement("div");
+ div.style.marginRight = calendar.columnMarginRight + "px";
+ div.style.position = 'relative';
+ div.style.height = '1px';
+ div.style.marginTop = '-1px';
+
+ var selection = document.createElement("div");
+ selection.style.position = "relative";
+ c.selection = selection;
+
+ c.appendChild(div);
+ c.appendChild(selection);
+
+ }
+ }
+
+ for (var i = 0; i < rowCount; i++) {
+ var r = (create) ? table.insertRow(-1) : table.rows[i];
+
+ if (create) {
+ r.style.MozUserSelect = 'none';
+ r.style.KhtmlUserSelect = 'none';
+ }
+
+ for (var j = 0; j < cl; j++) {
+ var col = this._columns[j];
+
+ var c = (create) ? r.insertCell(-1) : r.cells[j];
+
+ // always update
+ c.start = col.start.addTime(i * step);
+ c.end = c.start.addTime(step);
+ c.resource = col.id;
+
+ c.onmousedown = this._onCellMousedown;
+ // c.onmousemove = this._mousemove;
+ c.onmouseup = function() {
+ return false;
+ };
+
+ c.onclick = function() {
+ return false;
+ };
+
+ if (create) {
+ c.root = this;
+
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+ c.style.verticalAlign = 'top';
+ c.style.height = calendar.cellHeight + 'px';
+ c.style.overflow = 'hidden';
+ c.unselectable = 'on';
+
+ var div = document.createElement("div");
+ div.unselectable = 'on';
+ div.style.height = calendar.cellHeight + "px";
+ div.style.position = "relative";
+ div.className = this._prefixCssClass("_cell");
+
+ var business = this._isBusinessCell(c.start, c.end);
+
+ var properties = {
+ "business": business,
+ "text": null,
+ "html": null,
+ "cssClass": null,
+ "backColor": null,
+ "backImage": null,
+ "backRepeat": null,
+ };
+
+ (function() {
+ if (typeof calendar.onBeforeCellRender === 'function') {
+ var args = {};
+ args.cell = {
+ "start": c.start,
+ "end": c.end,
+ "resource": c.resource,
+ "properties": properties
+ };
+ calendar.onBeforeCellRender(args);
+ }
+ })();
+
+
+ if (properties.business) {
+ DayPilot.Util.addClass(div, calendar._prefixCssClass("_cell_business"));
+ }
+
+ if (properties.cssClass) {
+ DayPilot.Util.addClass(div, properties.cssClass);
+ }
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = this._prefixCssClass("_cell_inner");
+
+ var html = DayPilot.Util.escapeTextHtml(properties.text, properties.html);
+ if (html) {
+ inner.innerHTML = html;
+ }
+
+ if (properties.backColor) {
+ inner.style.backgroundColor = properties.backColor;
+ }
+
+ if (properties.backImage) {
+ inner.style.backgroundImage = "url(" + properties.backImage + ")";
+ }
+
+ if (properties.backRepeat) {
+ inner.style.backgroundRepeat = properties.backRepeat;
+ }
+
+ div.appendChild(inner);
+ c.appendChild(div);
+
+ c.appendChild(div);
+
+ }
+ }
+ }
+
+ table.root = this;
+
+ calendar.nav.scrollable.onmousemove = function(ev) {
+ var ref = calendar.nav.scrollable;
+ calendar.coords = DayPilot.mo3(ref, ev);
+
+ var mousePos = DayPilot.mc(ev);
+
+ if (DayPilotCalendar.resizing) {
+ if (!DayPilotCalendar.resizingShadow) {
+ DayPilotCalendar.resizingShadow = calendar._createShadow(DayPilotCalendar.resizing, false, calendar.shadow);
+ }
+
+ var _step = calendar.cellHeight;
+ var _startOffset = 1;
+ var delta = (mousePos.y - DayPilotCalendar.originalMouse.y);
+
+ // TODO: clear
+ if (DayPilotCalendar.resizing.dpBorder === 'bottom') {
+ var newHeight = Math.floor(((DayPilotCalendar.originalHeight + DayPilotCalendar.originalTop + delta) + _step / 2) / _step) * _step - DayPilotCalendar.originalTop + _startOffset;
+
+ if (newHeight < _step)
+ newHeight = _step;
+
+ var max = calendar.nav.main.clientHeight;
+ if (DayPilotCalendar.originalTop + newHeight > max)
+ newHeight = max - DayPilotCalendar.originalTop;
+
+ // DayPilotCalendar.resizingShadow.style.height = (newHeight - 4) + 'px';
+ DayPilotCalendar.resizingShadow.style.height = (newHeight) + 'px';
+ }
+ else if (DayPilotCalendar.resizing.dpBorder === 'top') {
+ var newTop = Math.floor(((DayPilotCalendar.originalTop + delta - _startOffset) + _step / 2) / _step) * _step + _startOffset;
+
+ if (newTop < _startOffset) {
+ newTop = _startOffset;
+ }
+
+ if (newTop > DayPilotCalendar.originalTop + DayPilotCalendar.originalHeight - _step) {
+ newTop = DayPilotCalendar.originalTop + DayPilotCalendar.originalHeight - _step;
+ }
+
+ // var newHeight = DayPilotCalendar.originalHeight - (newTop - DayPilotCalendar.originalTop) - 4;
+ var newHeight = DayPilotCalendar.originalHeight - (newTop - DayPilotCalendar.originalTop);
+
+ if (newHeight < _step) {
+ newHeight = _step;
+ }
+ else {
+ DayPilotCalendar.resizingShadow.style.top = newTop + 'px';
+ }
+
+ DayPilotCalendar.resizingShadow.style.height = (newHeight) + 'px';
+ }
+ }
+ else if (DayPilotCalendar.moving) {
+
+ if (!calendar.coords) {
+ return;
+ }
+
+ if (!DayPilotCalendar.movingShadow) {
+
+ var minDistance = 3;
+ var mousePos = DayPilot.mc(ev);
+ var distance = Math.abs(mousePos.x - DayPilotCalendar.originalMouse.x) + Math.abs(mousePos.y - DayPilotCalendar.originalMouse.y);
+ if (distance <= minDistance) {
+ return;
+ }
+ // fixes the ie8 bug (incorrect offsetX and offsetY causes flickering during move if there are inline elements in the event
+ DayPilotCalendar.movingShadow = calendar._createShadow(DayPilotCalendar.moving, true, calendar.shadow);
+ DayPilotCalendar.movingShadow.style.width = (DayPilotCalendar.movingShadow.parentNode.offsetWidth + 1) + 'px';
+ }
+
+ var _step = calendar.cellHeight;
+ var _startOffset = 1;
+
+ var offset = DayPilotCalendar.moveOffsetY;
+ if (!offset) {
+ offset = _step / 2; // for external drag
+ }
+
+ var newTop = Math.floor(((calendar.coords.y - offset - _startOffset) + _step / 2) / _step) * _step + _startOffset;
+
+ if (newTop < _startOffset) {
+ newTop = _startOffset;
+ }
+
+ var main = calendar.nav.events;
+ var max = calendar.nav.main.clientHeight + _startOffset;
+
+ var height = parseInt(DayPilotCalendar.movingShadow.style.height); // DayPilotCalendar.moving.data.height
+ if (newTop + height > max) {
+ newTop = max - height;
+ }
+
+ DayPilot.Util.addClass(DayPilotCalendar.moving, calendar._prefixCssClass("_event_moving_source"));
+
+ DayPilotCalendar.movingShadow.parentNode.style.display = 'none';
+ DayPilotCalendar.movingShadow.style.top = newTop + 'px';
+ DayPilotCalendar.movingShadow.parentNode.style.display = '';
+
+ var colWidth = main.clientWidth / main.rows[0].cells.length;
+ var column = Math.floor((calendar.coords.x - 45) / colWidth);
+
+ if (column < 0) {
+ column = 0;
+ }
+
+ if (column < main.rows[0].cells.length && column >= 0 && DayPilotCalendar.movingShadow.column !== column) {
+ DayPilotCalendar.movingShadow.column = column;
+ DayPilotCalendar.moveShadow(main.rows[0].cells[column]);
+ }
+ }
+ else if (DayPilotCalendar.selecting) {
+
+ var mousePos = DayPilot.mc(ev);
+
+ var cellcoords = calendar._table.getCellCoords();
+ var x = DayPilotCalendar.column;
+ var cell = calendar.nav.main.rows[cellcoords.y].cells[x];
+
+
+ // clean
+ // calendar.clearSelection();
+
+ // new selected cells
+ if (mousePos.y < DayPilotCalendar.firstMousePos.y) {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsBelow(cell);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.selectedCells[0];
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.firstSelected;
+ }
+ else {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsAbove(cell);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.firstSelected;
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.selectedCells[0];
+ }
+
+ calendar._activateSelection();
+
+ }
+ };
+ calendar.nav.scrollable.style.display = '';
+ };
+
+ this._isBusinessCell = function(start, end) {
+ if (this.businessBeginsHour < this.businessEndsHour)
+ {
+ return !(start.getHours() < this.businessBeginsHour || start.getHours() >= this.businessEndsHour || start.getDayOfWeek() === 6 || start.getDayOfWeek() === 0);
+ }
+
+ if (start.getHours() >= this.businessBeginsHour)
+ {
+ return true;
+ }
+
+ if (start.getHours() < this.businessEndsHour)
+ {
+ return true;
+ }
+
+ return false;
+ };
+
+ this._disposeHeader = function() {
+ var table = this.nav.header;
+ if (table && table.rows) {
+ for(var y = 0; y < table.rows.length; y++) {
+ var r = table.rows[y];
+ for (var x = 0; x < r.cells.length; x++) {
+ var c = r.cells[x];
+ c.onclick = null;
+ c.onmousemove = null;
+ c.onmouseout = null;
+ }
+ }
+ }
+ if (!this.fasterDispose) DayPilot.pu(table);
+ };
+
+ this._drawHeaderRow = function(create) {
+
+ // column headers
+ var r = (create) ? this.nav.header.insertRow(-1) : this.nav.header.rows[0];
+
+ var columns = this._columns;
+
+ var len = columns.length;
+
+ function drawHeaderCell(i) {
+ var data = columns[i];
+
+ var cell = (create) ? r.insertCell(-1) : r.cells[i];
+ cell.data = data;
+
+ cell.style.overflow = 'hidden';
+ cell.style.padding = '0px';
+ cell.style.border = '0px none';
+ cell.style.height = (calendar.headerHeight) + "px";
+
+ cell.onclick = calendar._headerClickDispatch;
+
+ var div = (create) ? document.createElement("div") : cell.firstChild;
+ var inner;
+
+ if (create) {
+ div.unselectable = 'on';
+ div.style.MozUserSelect = 'none';
+ div.style.cursor = 'default';
+ div.style.position = 'relative';
+ div.className = calendar._prefixCssClass('_colheader');
+ div.style.height = calendar.headerHeight + "px";
+
+ if (!calendar.headerTextWrappingEnabled) {
+ div.style.whiteSpace = 'nowrap';
+ }
+
+ inner = document.createElement("div");
+ inner.className = calendar._prefixCssClass('_colheader_inner');
+ inner.unselectable = 'on';
+
+ div.appendChild(inner);
+ cell.appendChild(div);
+ }
+ else {
+ inner = div.firstChild;
+ }
+
+ var args = {};
+ args.header = {};
+ args.header.cssClass = null;
+ args.header.verticalAlignment = "center";
+ args.column = calendar._createColumn(data, calendar);
+
+ if (typeof calendar.onBeforeHeaderRender === 'function') {
+ DayPilot.Util.copyProps(data, args.header, ['id', 'start', 'name', 'html', 'backColor', 'toolTip', 'areas']);
+ calendar.onBeforeHeaderRender(args);
+ DayPilot.Util.copyProps(args.header, data, ['html', 'backColor', 'toolTip', 'areas', 'cssClass', 'verticalAlignment']);
+ }
+
+ if (data.toolTip) {
+ inner.title = data.toolTip;
+ }
+
+ if (data.cssClass) {
+ DayPilot.Util.addClass(div, data.cssClass);
+ }
+
+ if (data.backColor) {
+ inner.style.background = data.backColor;
+ }
+
+ if (data.areas) {
+ DayPilot.Areas.attach(div, data);
+ }
+
+ var va = data.verticalAlignment;
+ if (va) {
+ inner.style.display = "flex";
+ switch (va) {
+ case "center":
+ inner.style.alignItems = "center";
+ break;
+ case "top":
+ inner.style.alignItems = "flex-start";
+ break;
+ case "bottom":
+ inner.style.alignItems = "flex-end";
+ break;
+ }
+ }
+
+ var text = div.firstChild;
+ text.innerHTML = calendar._xssTextHtml(data.name, data.html);
+ }
+
+ for (var i = 0; i < len; i++) {
+ drawHeaderCell(i);
+ }
+ };
+
+ this._headerClickDispatch = function(ev) {
+
+ var handling = calendar.headerClickHandling;
+
+ if (handling === "Disabled") {
+ return;
+ }
+
+ var data = this.data;
+ var c = calendar._createColumn(data);
+
+ var args = {};
+ args.header = {};
+ args.header.id = data.id;
+ args.header.name = data.name;
+ args.header.start = data.start;
+ args.column = c;
+ args.originalEvent = ev;
+ args.shift = ev.shiftKey;
+ args.ctrl = ev.ctrlKey;
+ args.meta = ev.metaKey;
+
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onHeaderClick === 'function') {
+ calendar.onHeaderClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ if (typeof calendar.onHeaderClicked === 'function') {
+ calendar.onHeaderClicked(args);
+ }
+
+ };
+
+ this._createColumn = function(data) {
+ return new DayPilot.CalendarColumn(data, calendar);
+ };
+
+ this._widthUnit = function() {
+ if (this.width && this.width.indexOf("px") !== -1) {
+ return "Pixel";
+ }
+ return "Percentage";
+ };
+
+ this._drawHeader = function() {
+
+ var header = this.nav.header;
+ var create = true;
+
+ var columns = this._columns;
+ var len = columns.length;
+
+ while (this.headerCreated && header && header.rows && header.rows.length > 0 && create) {
+ if (!this.fasterDispose) DayPilot.pu(header.rows[0]);
+ header.deleteRow(0);
+ }
+
+ this.headerCreated = true;
+
+ if (!create) {
+ // corner
+ var corner = calendar.nav.corner;
+ if (!this.fasterDispose) DayPilot.pu(corner.firstChild);
+ }
+
+ this._drawHeaderRow(create);
+ };
+
+ this.loadingStart = function() {
+ if (this.loadingLabelVisible) {
+ this.nav.loading.innerHTML = this.loadingLabelText;
+ this.nav.loading.style.top = (this.headerHeight + 5) + "px";
+ this.nav.loading.style.display = '';
+ }
+ };
+
+ this.commandCallBack = function(command, data) {
+ var params = {};
+ params.command = command;
+ this._callBack2('Command', data, params);
+ };
+
+ this.loadingStop = function(msg) {
+ if (this.callbackTimeout) {
+ window.clearTimeout(this.callbackTimeout);
+ }
+
+ this.nav.loading.style.display = 'none';
+ };
+
+ this._enableScrolling = function() {
+
+ var scrollDiv = this.nav.scroll;
+
+ if (!scrollDiv.onscroll) {
+ scrollDiv.onscroll = function() {
+ calendar._saveScrollHour();
+ };
+ }
+
+ var scrollpos = (typeof this._config.scrollpos !== 'undefined') ? this._config.scrollpos : this.initScrollPos;
+
+ if (!scrollpos) {
+ return;
+ }
+
+
+ if (scrollpos === 'Auto') {
+ if (this.heightSpec === "BusinessHours") {
+ scrollpos = 2 * this.cellHeight * this.businessBeginsHour;
+ }
+ else {
+ scrollpos = 0;
+ }
+ }
+
+ scrollDiv.root = this;
+
+ // initial position
+ if (scrollDiv.scrollTop === 0) {
+ scrollDiv.scrollTop = scrollpos;
+ }
+ };
+
+ this.callbackError = function (result, context) {
+ alert("Error!\r\nResult: " + result + "\r\nContext:" + context);
+ };
+
+ this._fixScrollHeader = function() {
+ var w = DayPilot.sw(this.nav.scroll);
+ var d = this.nav.cornerRight;
+ if (d) {
+ d.style.width = w + 'px';
+ }
+ };
+
+ this._registerGlobalHandlers = function() {
+ if (!DayPilotCalendar.globalHandlers) {
+ DayPilotCalendar.globalHandlers = true;
+ DayPilot.re(document, 'mouseup', DayPilotCalendar.gMouseUp);
+ DayPilot.re(window, 'unload', DayPilotCalendar.gUnload);
+ }
+ };
+
+ this.events = {};
+ //this.events.list = [];
+
+ this.events.add = function(e) {
+
+ var data = null;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ data = e;
+ }
+ else {
+ throw "DayPilot.Calendar.events.add() expects an object or DayPilot.Event instance.";
+ }
+
+ if (!calendar.events.list) {
+ calendar.events.list = [];
+ }
+
+ calendar.events.list.push(data);
+ calendar._update({"eventsOnly": true});
+ calendar._angular.notify();
+ };
+
+
+ this.events.find = function(id) {
+ if (!calendar.events.list) {
+ return null;
+ }
+
+ if (typeof id === "function") {
+ var fn = id;
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var e = new DayPilot.Event(calendar.events.list[i], calendar);
+ if (fn(e)) {
+ return e;
+ }
+ }
+ return null;
+ }
+
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var data = calendar.events.list[i];
+ if (data.id === id) {
+ return new DayPilot.Event(data, calendar);
+ }
+ }
+ return null;
+ };
+
+ this.events.update = function(e) {
+ if (e instanceof DayPilot.Event) {
+ e.commit();
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ var index = DayPilot.indexOf(calendar.events.list, target.data);
+ calendar.events.list.splice(index, 1, e);
+ }
+ }
+
+ calendar._update({"eventsOnly": true});
+ calendar._angular.notify();
+ };
+
+ this.events.remove = function(e) {
+ var data;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ data = target.data;
+ }
+ }
+ else if (typeof e === "string" || typeof e === "number") {
+ var target = calendar.events.find(e);
+ if (target) {
+ data = target.data;
+ }
+ }
+
+ var index = DayPilot.indexOf(calendar.events.list, data);
+ calendar.events.list.splice(index, 1);
+ calendar._update({"eventsOnly": true});
+ calendar._angular.notify();
+ };
+
+ this.events.load = function(url, success, error) {
+ var onError = function(args) {
+ var largs = {};
+ largs.exception = args.exception;
+ largs.request = args.request;
+
+ if (typeof error === 'function') {
+ error(largs);
+ }
+ };
+
+ var onSuccess = function(args) {
+ var r = args.request;
+ var data;
+
+ // it's supposed to be JSON
+ try {
+ data = JSON.parse(r.responseText);
+ }
+ catch (e) {
+ var fargs = {};
+ fargs.exception = e;
+ onError(fargs);
+ return;
+ }
+
+ if (DayPilot.isArray(data)) {
+ var sargs = {};
+ sargs.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+ sargs.data = data;
+ if (typeof success === "function") {
+ success(sargs);
+ }
+
+ if (sargs.preventDefault.value) {
+ return;
+ }
+
+ calendar.events.list = data;
+ if (calendar._initialized) {
+ calendar._update({"eventsOnly": true});
+ }
+ }
+ };
+
+ var usePost = calendar.eventsLoadMethod && calendar.eventsLoadMethod.toUpperCase() === "POST";
+
+ if (usePost) {
+ DayPilot.Http.ajax({
+ "method": "POST",
+ "data": { "start": calendar.visibleStart().toString(), "end": calendar.visibleEnd().toString()},
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ else {
+ var fullUrl = url;
+ var queryString = "start=" + calendar.visibleStart().toString() + "&end=" + calendar.visibleEnd().toString();
+ if (fullUrl.indexOf("?") > -1) {
+ fullUrl += "&" + queryString;
+ }
+ else {
+ fullUrl += "?" + queryString;
+ }
+
+ DayPilot.Http.ajax({
+ "method": "GET",
+ "url": fullUrl,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+
+ };
+
+ this._updateTheme = function() {
+ // manually update theme for elements that are not redrawn during update
+ //return;
+
+ var needsUpdate = calendar.nav.top.className !== calendar._prefixCssClass("_main");
+
+ if (!needsUpdate) {
+ return;
+ }
+
+ calendar.nav.top.className = calendar._prefixCssClass("_main");
+ var corner = calendar.nav.corner;
+ corner.className = calendar._prefixCssClass("_corner");
+ corner.firstChild.className = calendar._prefixCssClass("_corner_inner");
+
+ var cr = calendar.nav.cornerRight;
+ if (cr) {
+ cr.className = calendar._prefixCssClass("_cornerright");
+ cr.firstChild.className = calendar._prefixCssClass("_cornerright_inner");
+ }
+ };
+
+ this.update = function(options) {
+
+ if (calendar._disposed) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Calendar instance that has been disposed.");
+ }
+
+ calendar._loadOptions(options);
+ calendar._update();
+ };
+
+ this._update = function(args) {
+
+ if (!this._initialized) {
+ return;
+ }
+
+ var args = args || {};
+ var full = !args.eventsOnly;
+
+ calendar._prepareVariables();
+ calendar._deleteEvents();
+
+ // reset after drag and drop/deleting
+ calendar.nav.top.style.cursor = "auto";
+
+ // calendar._show();
+
+ if (full) {
+ calendar._prepareColumns();
+ calendar._drawHeader();
+ calendar._drawMain();
+ calendar._drawHourTable();
+ calendar._updateHeight();
+ calendar._updateCorner();
+ calendar._fixScrollHeader();
+ calendar._updateTheme();
+
+ calendar._restoreScrollHour();
+ }
+
+ calendar._loadEvents();
+ calendar._updateHeaderHeight();
+
+ calendar._drawEvents();
+ calendar.clearSelection();
+
+ if (this.visible) {
+ this.show();
+ }
+ else {
+ this.hide();
+ }
+ };
+
+
+ this._specialHandling = null;
+ this._loadOptions = function(options) {
+ if (!options) {
+ return;
+ }
+ var specialHandling = {
+ "events": {
+ "preInit": function() {
+ var events = this.data || [];
+ if (DayPilot.isArray(events.list)) {
+ calendar.events.list = events.list;
+ }
+ else {
+ calendar.events.list = events;
+ }
+ }
+ },
+ "columns": {
+ "preInit": function() {
+ calendar.columns.list = this.data;
+ }
+ }
+ };
+ this._specialHandling = specialHandling;
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ var item = specialHandling[name];
+ item.data = options[name];
+ if (item.preInit) {
+ item.preInit();
+ }
+ }
+ else {
+ calendar[name] = options[name];
+ }
+ }
+
+ };
+
+ this._postInit = function() {
+ var specialHandling = this._specialHandling;
+ for (var name in specialHandling) {
+ var item = specialHandling[name];
+ if (item.postInit) {
+ item.postInit();
+ }
+ }
+ };
+
+ this._loadTop = function() {
+ if (this.id && this.id.tagName) {
+ this.nav.top = this.id;
+ }
+ else if (typeof this.id === "string") {
+ this.nav.top = document.getElementById(this.id);
+ if (!this.nav.top) {
+ throw "DayPilot.Calendar: The placeholder element not found: '" + id + "'.";
+ }
+ }
+ else {
+ throw "DayPilot.Calendar() constructor requires the target element or its ID as a parameter";
+ }
+ };
+
+ this._cache = {};
+ this._cache.events = [];
+
+ this._doBeforeEventRender = function(i) {
+ var cache = this._cache.events;
+ var data = this.events.list[i];
+ var evc = {};
+
+ // make a copy
+ for (var name in data) {
+ evc[name] = data[name];
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ var args = {};
+ args.control = calendar;
+ args.data = evc;
+ this.onBeforeEventRender(args);
+ }
+
+ cache[i] = evc;
+
+ };
+
+ this._loadEvents = function() {
+
+ var events = this.events.list;
+
+ calendar._cache.events = [];
+
+ if (!events) {
+ return;
+ }
+
+ if (!DayPilot.isArray(events)) {
+ throw new DayPilot.Exception("DayPilot.Calendar.events.list expects an array object. You supplied: " + (typeof events));
+ }
+
+ var length = events.length;
+ var duration = 24 * 60 * 60 * 1000;
+
+ this.cache.pixels = {};
+
+ var loadCache = [];
+
+ this.scrollLabels = [];
+
+ this.minStart = 10000;
+ this.maxEnd = 0;
+
+ for (var i = 0; i < length; i++) {
+ var e = events[i];
+ var edata = e;
+
+ if (typeof edata !== "object") {
+ throw new DayPilot.Exception("Event data item must be an object");
+ }
+ if (!edata.start) {
+ throw new DayPilot.Exception("Event data item must specify 'start' property");
+ }
+ if (!edata.end) {
+ throw new DayPilot.Exception("Event data item must specify 'end' property");
+ }
+
+ if (edata instanceof DayPilot.Event) {
+ throw new DayPilot.Exception("DayPilot.Calendar: DayPilot.Event object detected in events.list array. Use raw event data instead.");
+ }
+
+ e.start = new DayPilot.Date(e.start);
+ e.end = new DayPilot.Date(e.end);
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ for (var i = 0; i < length; i++) {
+ this._doBeforeEventRender(i);
+ }
+ }
+
+ for(var i = 0; i < this._columns.length; i++) {
+
+ var scroll = {};
+ scroll.minEnd = 1000000;
+ scroll.maxStart = -1;
+ this.scrollLabels.push(scroll);
+
+ var col = this._columns[i];
+ col.events = [];
+ col.lines = [];
+ col.blocks = [];
+
+ var colStart = new DayPilot.Date(col.start);
+ var colStartTicks = colStart.getTime();
+ var colEnd = colStart.addTime(duration);
+ var colEndTicks = colEnd.getTime();
+
+ for (var j = 0; j < length; j++) {
+ if (loadCache[j]) {
+ continue;
+ }
+
+ var e = events[j];
+
+ var start = e.start;
+ var end = e.end;
+
+ var startTicks = start.getTime();
+ var endTicks = end.getTime();
+
+ if (endTicks < startTicks) { // skip invalid events
+ continue;
+ }
+
+ // belongs here
+ var belongsHere = !(endTicks <= colStartTicks || startTicks >= colEndTicks);
+ if (calendar.viewType === "Resources") {
+ belongsHere = belongsHere && col.id === e.resource;
+ }
+
+ if (belongsHere) {
+ var ep = new DayPilot.Event(e, calendar); // event part
+ ep.part.dayIndex = i;
+ ep.part.start = colStartTicks < startTicks ? e.start : colStart;
+ ep.part.end = colEndTicks > endTicks ? e.end : colEnd;
+
+ var partStartPixels = this.getPixels(ep.part.start, col.start);
+ var partEndPixels = this.getPixels(ep.part.end, col.start);
+
+ var top = partStartPixels.top;
+ var bottom = partEndPixels.top;
+
+ // events in the hidden areas
+ if (top === bottom && (partStartPixels.cut || partEndPixels.cut)) {
+ continue;
+ }
+
+ var boxBottom = partEndPixels.boxBottom;
+
+ ep.part.top = Math.floor(top / this.cellHeight) * this.cellHeight + 1;
+ ep.part.height = Math.max(Math.ceil(boxBottom / this.cellHeight) * this.cellHeight - ep.part.top, this.cellHeight - 1) + 1;
+ ep.part.barTop = Math.max(top - ep.part.top - 1, 0); // minimum 0
+ ep.part.barHeight = Math.max(bottom - top - 2, 1); // minimum 1
+
+ var start = ep.part.top;
+ var end = ep.part.top + ep.part.height;
+
+ if (start > scroll.maxStart) {
+ scroll.maxStart = start;
+ }
+ if (end < scroll.minEnd) {
+ scroll.minEnd = end;
+ }
+
+ if (start < this.minStart) {
+ this.minStart = start;
+ }
+ if (end > this.maxEnd) {
+ this.maxEnd = end;
+ }
+ col.events.push(ep);
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ ep.cache = this._cache.events[j];
+ }
+
+ if (ep.part.start.getTime() === startTicks && ep.part.end.getTime() === endTicks) {
+ loadCache[j] = true;
+ }
+ }
+ }
+
+ }
+
+
+ // sort events inside rows
+ for (var i = 0; i < this._columns.length; i++) {
+ var col = this._columns[i];
+ col.events.sort(this._eventComparer);
+
+ // put into lines
+ for (var j = 0; j < col.events.length; j++) {
+ var e = col.events[j];
+ col.putIntoBlock(e);
+ }
+
+ for (var j = 0; j < col.blocks.length; j++) {
+ var block = col.blocks[j];
+ block.events.sort(this._eventComparer);
+ for (var k = 0; k < block.events.length; k++ ) {
+ var e = block.events[k];
+ block.putIntoLine(e);
+ }
+ }
+ }
+ };
+
+ this._eventComparer = function(a, b) {
+ if (!a || !b || !a.start || !b.start) {
+ return 0; // no sorting, invalid arguments
+ }
+
+ var byStart = a.start().getTime() - b.start().getTime();
+ if (byStart !== 0) {
+ return byStart;
+ }
+
+ var byEnd = b.end().getTime() - a.end().getTime(); // desc
+ return byEnd;
+ };
+
+ this.debug = function(msg, append) {
+ if (!this.debuggingEnabled) {
+ return;
+ }
+
+ if (!calendar.debugMessages) {
+ calendar.debugMessages = [];
+ }
+ calendar.debugMessages.push(msg);
+
+ if (typeof console !== 'undefined') {
+ console.log(msg);
+ }
+ };
+
+ this.getPixels = function(date, start) {
+ if (!start) start = this.startDate;
+
+ var startTicks = start.getTime();
+ var ticks = date.getTime();
+
+ var cache = this.cache.pixels[ticks + "_" + startTicks];
+ if (cache) {
+ return cache;
+ }
+
+ startTicks = start.getTime();
+
+ var boxTicks = 30 * 60 * 1000;
+ var topTicks = ticks - startTicks;
+ var boxOffsetTicks = topTicks % boxTicks;
+
+ var boxStartTicks = topTicks - boxOffsetTicks;
+ var boxEndTicks = boxStartTicks + boxTicks;
+ if (boxOffsetTicks === 0) {
+ boxEndTicks = boxStartTicks;
+ }
+
+ // it's linear scale so far
+ var result = {};
+ result.cut = false;
+ result.top = this._ticksToPixels(topTicks);
+ result.boxTop = this._ticksToPixels(boxStartTicks);
+ result.boxBottom = this._ticksToPixels(boxEndTicks);
+
+ this.cache.pixels[ticks + "_" + startTicks] = result;
+
+ return result;
+ };
+
+ this._ticksToPixels = function(ticks) {
+ return Math.floor( (this.cellHeight * ticks) / (1000 * 60 * 30) );
+ };
+
+ this._prepareVariables = function() {
+ this.startDate = new DayPilot.Date(this.startDate).getDatePart();
+ };
+
+ this._updateHeaderHeight = function() {
+ if (this.nav.corner) {
+ this.nav.corner.style.height = this.headerHeight + "px";
+ }
+ };
+
+ this._updateHeight = function() {
+ var sh = this._getScrollableHeight();
+ if (this.nav.scroll && sh > 0) {
+ this.nav.scroll.style.height = sh + "px";
+ }
+ };
+
+ this._angular = {};
+ this._angular.scope = null;
+ this._angular.notify = function() {
+ if (calendar._angular.scope) {
+ calendar._angular.scope["$apply"]();
+ }
+ };
+ this._angular.apply = function(f) {
+ // autoapply has been disabled
+ f();
+
+ /*
+ if (calendar.angularAutoApply && calendar._angular.scope) {
+ calendar._angular.scope["$apply"](f);
+ }
+ else {
+ f();
+ }
+ */
+ };
+
+ this._saveScrollHour = function() {
+ if (!calendar.nav.scroll) {
+ return;
+ }
+ var top = calendar.nav.scroll.scrollTop;
+ var pos = top / (2*calendar.cellHeight);
+ calendar._config.scrollHour = pos;
+ };
+
+ this._restoreScrollHour = function() {
+ var scrollpos = 0;
+ if (typeof calendar._config.scrollHour === "number") {
+ scrollpos = 2 * calendar.cellHeight * calendar._config.scrollHour;
+ //calendar._config.scrollHour = null;
+ }
+ else {
+ if (calendar.initScrollPos === 'Auto') {
+ if (this.heightSpec === "BusinessHours") {
+ scrollpos = 2 * this.cellHeight * this.businessBeginsHour;
+ }
+ else {
+ scrollpos = 0;
+ }
+ }
+ }
+
+ calendar.nav.scroll.scrollTop = scrollpos;
+ };
+
+ this._loadFromServer = function() {
+ // make sure it has a place to ask
+ if (this.backendUrl || typeof WebForm_DoCallback === 'function') {
+ return (typeof calendar.events.list === 'undefined') || (!calendar.events.list);
+ }
+ else {
+ return false;
+ }
+ };
+
+ this._show = function() {
+ if (this.nav.top.style.visibility === 'hidden') {
+ this.nav.top.style.visibility = 'visible';
+ }
+ };
+
+ this.show = function() {
+ calendar.visible = true;
+ calendar.nav.top.style.display = '';
+ this._fixScrollHeader();
+ };
+
+ this.hide = function() {
+ calendar.visible = false;
+ calendar.nav.top.style.display = 'none';
+ };
+ this._initShort = function() {
+ this._prepareVariables();
+ this._prepareColumns();
+ this._drawTop();
+ this._drawHeader();
+ this._drawMain();
+ this._fixScrollHeader();
+ this._enableScrolling();
+ this._registerGlobalHandlers();
+ DayPilotCalendar.register(this);
+
+ this._waitForVisibility();
+ this._callBack2('Init');
+ };
+
+ this._config = {};
+
+ this._saveConfig = function() {
+ this._config.themes = [];
+ this._config.themes.push(this.theme || this.cssClassPrefix);
+ };
+
+ this._clearThemes = function() {
+ var themes = this._config.themes;
+ for (var i = 0; i < themes.length; i++) {
+ var theme = themes[i];
+ DayPilot.Util.removeClass(this.nav.top, theme + "_main");
+ }
+ this._config.themes = [];
+ };
+
+ this._doAfterRender = function() {
+ this.afterRender(null, false);
+ if (typeof this.onAfterRender === "function") {
+ var args = {};
+ args.isCallBack = false;
+ this.onAfterRender(args);
+ }
+ };
+
+ this._doInit = function() {
+ if (typeof this.onInit === "function" && !this._onInitCalled) {
+ this._onInitCalled = true;
+ var args = {};
+ this.onInit(args);
+ }
+ };
+
+ this._visible = function() {
+ var el = calendar.nav.top;
+ if (!el) {
+ return false;
+ }
+ return el.offsetWidth > 0 && el.offsetHeight > 0;
+ };
+
+ this._waitForVisibility = function() {
+ var visible = calendar._visible;
+
+ if (!visible()) {
+ calendar._visibilityInterval = setInterval(function() {
+ if (visible()) {
+ calendar._enableScrolling();
+ calendar._fixScrollHeader();
+ clearInterval(calendar._visibilityInterval);
+ }
+ }, 100);
+ }
+ };
+
+ this._xssTextHtml = function(text, html) {
+
+ if (calendar._resolved._xssProtectionEnabled()) {
+ return DayPilot.Util.escapeTextHtml(text, html);
+ }
+
+ if (!DayPilot.Util.isNullOrUndefined(html)) {
+ return html;
+ }
+ if (DayPilot.Util.isNullOrUndefined(text)) {
+ return "";
+ }
+ return text;
+ };
+
+
+ this.internal = {};
+ this.internal.loadOptions = calendar._loadOptions;
+ this.internal.xssTextHtml = calendar._xssTextHtml;
+
+ this.init = function() {
+ this._loadTop();
+
+ var loadFromServer = this._loadFromServer();
+
+ this._saveConfig();
+
+ if (loadFromServer) {
+ this._initShort();
+ return;
+ }
+
+ this._prepareVariables();
+ this._prepareColumns();
+
+ this._loadEvents();
+
+ this._drawTop();
+ this._drawHeader();
+ this._drawMain();
+
+ this._show();
+
+ this._fixScrollHeader();
+ this._enableScrolling();
+ this._registerGlobalHandlers();
+ DayPilotCalendar.register(this);
+
+ if (this.events) { // are events available?
+ this._updateHeaderHeight();
+ this._drawEvents();
+ }
+
+ this._doAfterRender();
+ this._doInit();
+ this._waitForVisibility();
+ this._initialized = true;
+
+ return this;
+ };
+
+ this.Init = this.init;
+
+ this._loadOptions(options);
+ };
+
+ DayPilot.CalendarColumn = function(col, calendar) {
+ var column = this;
+
+ column.id = col.id;
+ column.name = col.name;
+ column.data = col.data;
+ column.start = new DayPilot.Date(col.start);
+ column.calendar = calendar;
+
+ column.toJSON = function() {
+ var json = {};
+ json.id = this.id;
+ if (this.start) {
+ json.start = this.start.toString();
+ }
+ json.name = this.name;
+ return json;
+ };
+ };
+
+
+ // publish the API
+ DayPilot.Calendar = DayPilotCalendar.Calendar;
+
+ // jQuery plugin
+ if (typeof jQuery !== 'undefined') {
+ (function( $ ){
+ $.fn.daypilotCalendar = function(options) {
+ var first = null;
+ var j = this.each(function() {
+ if (this.daypilot) { // already initialized
+ return;
+ };
+
+ var daypilot = new DayPilot.Calendar(this.id);
+ this.daypilot = daypilot;
+ for (name in options) {
+ daypilot[name] = options[name];
+ }
+ daypilot.init();
+ if (!first) {
+ first = daypilot;
+ }
+ });
+ if (this.length === 1) {
+ return first;
+ }
+ else {
+ return j;
+ }
+ };
+ })( jQuery );
+ }
+
+ (function registerAngularModule() {
+
+ var app = DayPilot.am();
+
+ if (!app) {
+ return;
+ }
+
+ app.directive("daypilotCalendar", ['$parse', function($parse) {
+// app.directive("daypilotCalendar", function() {
+ return {
+ "restrict": "E",
+ "template": "",
+ "replace": true,
+ "link": function (scope, element, attrs) {
+
+ var calendar = new DayPilot.Calendar(element[0]);
+ calendar._angular.scope = scope;
+ calendar.init();
+
+ var oattr = attrs["id"];
+ if (oattr) {
+ scope[oattr] = calendar;
+ }
+
+ // save DayPilot.Calendar object in the specified variable
+ var pas = attrs["publishAs"];
+ if (pas) {
+ var getter = $parse(pas);
+ var setter = getter.assign;
+ setter(scope, calendar);
+ }
+
+ // bind event handlers from attributes starting with "on"
+ for (var name in attrs) {
+ if (name.indexOf("on") === 0) { // event handler
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ scope["$apply"](function() {
+ f(scope, {"args": args});
+ });
+ };
+ })(name);
+ }
+ }
+
+ var watch = scope["$watch"];
+ var config = attrs["config"] || attrs["daypilotConfig"];
+ var events = attrs["events"] || attrs["daypilotEvents"];
+
+ watch.call(scope, config, function (value) {
+ //var bbOld = calendar.businessBeginsHour;
+
+ for (var name in value) {
+ calendar[name] = value[name];
+ }
+ calendar.update();
+ calendar._doInit();
+ }, true);
+
+ watch.call(scope, events, function(value) {
+ //var calendar = element.data("calendar");
+ calendar.events.list = value;
+ calendar.update();
+ }, true);
+
+ }
+ };
+ }]);
+ })();
+
+})();
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+(function() {
+
+ if (typeof DayPilot.DatePicker !== 'undefined' && DayPilot.DatePicker.close) {
+ return;
+ }
+
+ DayPilot.DatePicker = function(properties) {
+ this.v = '2024.3.539-lite';
+ var navigatorId = "navigator_" + new Date().getTime();
+ var This = this;
+
+ this.onShow = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+ // this.dateFormat = null;
+
+ this.prepare = function() {
+ this.locale = "en-us";
+ this.target = null;
+ this.targetAlignment = "left";
+ this.resetTarget = true;
+ this.pattern = this._resolved.locale().datePattern; // "M/d/yyyy"
+ this.theme = "navigator_default";
+ this.patterns = [];
+ this.zIndex = null;
+
+ // load settings
+ if (properties) {
+ for (var name in properties) {
+ this[name] = properties[name];
+ }
+ }
+
+ };
+
+ this.init = function() {
+
+ this.date = new DayPilot.Date(this.date);
+
+ var value = this._readFromTarget();
+
+ if (this.resetTarget && !value) {
+ this._writeToTarget(this.date);
+ }
+ else if (!this.resetTarget) {
+ This.date = value;
+ }
+
+ var target = this._element();
+ if (target) {
+ target.addEventListener("input", function() {
+ This.date = This._readFromTarget();
+ if (This.date) {
+ This.navigator.select(This.date, {dontNotify: true});
+ }
+ });
+ }
+
+ document.addEventListener("mousedown", function() {
+ This.close();
+ });
+
+ return this;
+
+ };
+
+ this.close = function() {
+ if (!this._visible) {
+ return;
+ }
+
+ this._visible = false;
+
+ if (this.navigator) {
+ this.navigator.dispose();
+ }
+ this.div.innerHTML = '';
+ if (this.div && this.div.parentNode === document.body) {
+ document.body.removeChild(this.div);
+ }
+ };
+
+ this.setDate = function(date) {
+ this.date = new DayPilot.Date(date);
+ this._writeToTarget(this.date);
+ };
+
+ this._readFromTarget = function() {
+ // recognized targets: input (value), other DOM elements (innerHTML)
+ var element = this._element();
+
+ if (!element) {
+ return this.date;
+ }
+
+ var value = null;
+ if (element.tagName === "INPUT") {
+ value = element.value;
+ }
+ else {
+ value = element.innerText;
+ }
+
+ if (!value) {
+ return null;
+ }
+
+ var date = DayPilot.Date.parse(value, This.pattern);
+ for (var i = 0; i < This.patterns.length; i++) {
+ if (date) {
+ return date;
+ }
+ date = DayPilot.Date.parse(value, This.patterns[i]);
+ }
+
+ return date;
+ };
+
+ this._writeToTarget = function(date) {
+ var element = this._element();
+
+ if (!element) {
+ return;
+ }
+
+ var value = date.toString(This.pattern, This.locale);
+ if (element.tagName === "INPUT") {
+ element.value = value;
+ }
+ else {
+ element.innerHTML = value;
+ }
+
+ };
+
+ this._resolved = {};
+ this._resolved.locale = function() {
+ return DayPilot.Locale.find(This.locale);
+ };
+
+ /* this._resolved.dateFormat = function() {
+ if (typeof This.dateFormat === "string") {
+ return This.dateFormat;
+ }
+ return This._resolved.locale().datePattern;
+ }*/
+
+ this._element = function() {
+ var id = this.target;
+ // accept DOM element or id (string)
+ var element = (id && id.nodeType && id.nodeType === 1 ) ? id : document.getElementById(id);
+ return element;
+ };
+
+ Object.defineProperty(this, "visible", {
+ get: function() { return This._visible; }
+ });
+
+ this.show = function() {
+
+ if (this._visible) {
+ return;
+ }
+
+ var element = this._element();
+ var navigator = this.navigator;
+
+ var navigator = new DayPilot.Navigator(navigatorId);
+ navigator.api = 2;
+ navigator.cssOnly = true;
+ navigator.theme = This.theme;
+ navigator.weekStarts = "Auto";
+ navigator.locale = This.locale;
+ navigator.onTimeRangeSelected = function(args) {
+ This.date = args.start;
+
+ var start = args.start.addTime(navigator._pickerTimePart);
+ var value = start.toString(This.pattern, This.locale);
+
+ var args = {};
+ args.start = start;
+ args.date = start;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof This.onTimeRangeSelect === 'function') {
+ This.onTimeRangeSelect(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ This._writeToTarget(value);
+ This.close();
+
+ if (typeof This.onTimeRangeSelected === 'function') {
+ This.onTimeRangeSelected(args);
+ }
+ };
+
+ this.navigator = navigator;
+
+ var position = DayPilot.abs(element);
+ var height = element.offsetHeight;
+
+ var align = This.targetAlignment;
+
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+
+ if (align === "left") {
+ div.style.left = position.x + "px";
+ }
+
+
+ div.style.top = (position.y + height) + "px";
+ if (This.zIndex) {
+ div.style.zIndex = This.zIndex;
+ }
+
+ var nav = document.createElement("div");
+ nav.id = navigatorId;
+ div.appendChild(nav);
+
+ div.addEventListener("mousedown", function(ev) {
+ var ev = ev || window.event;
+ ev.cancelBubble = true;
+ ev.stopPropagation && ev.stopPropagation();
+ });
+
+ document.body.appendChild(div);
+
+ this.div = div;
+
+ var selected = This._readFromTarget() || new DayPilot.Date().getDatePart();
+
+ navigator.startDate = selected;
+ navigator._pickerTimePart = selected.getTimePart();
+ // navigator.selectionStart = selected.getDatePart();
+ navigator.selectionDay = selected.getDatePart();
+ navigator.init();
+
+ if (align === "right") {
+ var left = (position.x + element.offsetWidth - navigator.nav.top.offsetWidth);
+ div.style.left = left + "px";
+ }
+
+ this._visible = true;
+ if (this.onShow) {
+ this.onShow();
+ }
+ };
+
+ this.prepare(); // prepare only called once, in the constructor
+ this.init();
+ };
+
+})();
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function(DayPilot) {
+ 'use strict';
+
+ if (typeof DayPilot.Menu !== 'undefined' && DayPilot.Menu.def) {
+ return;
+ }
+
+ var doNothing = function() {};
+
+ var DayPilotMenu = {};
+
+ DayPilotMenu.mouse = null;
+ DayPilotMenu.menu = null;
+ DayPilotMenu.handlersRegistered = false;
+ DayPilotMenu.hideTimeout = null;
+ DayPilotMenu.waitingSubmenu = null;
+
+ DayPilot.Menu = function(items) {
+ var menu = this;
+ var initiatorAreaDiv = null;
+
+ this.v = '2024.3.539-lite';
+ this.zIndex = 120; // more than 10,001 used by ModalPopupExtender
+ //this.useShadow = true;
+ this.cssClassPrefix = "menu_default";
+ this.cssOnly = true;
+ this.menuTitle = null;
+ this.showMenuTitle = false;
+ this.hideOnMouseOut = false;
+ this.hideAfter = 200;
+ this.theme = null;
+
+ this.onShow = null;
+
+ // hiding internal properties for angular
+ this._state = function() {};
+ //this._state.ref = null; // ref object, used for position
+
+ if (items && DayPilot.isArray(items)) {
+ this.items = items;
+ }
+
+ // angular change detection
+ this.toJSON = function() {
+ return null;
+ };
+
+ this.show = function(e, options) {
+ options = options || {};
+
+ var value = null;
+ if (!e) {
+ value = null;
+ }
+ else if (typeof e.id === 'string' || typeof e.id === 'number') {
+ value = e.id;
+ }
+ else if (typeof e.id === 'function') {
+ value = e.id();
+ }
+ else if (typeof e.value === 'function') {
+ value = e.value();
+ }
+
+ if (typeof(DayPilot.Bubble) !== 'undefined') { // hide any bubble if active
+ DayPilot.Bubble.hideActive();
+ }
+
+ if (!options.submenu) {
+ DayPilotMenu.menuClean();
+ }
+
+ // clear old data
+ this._state.submenu = null;
+
+ if (DayPilotMenu.mouse === null) { // not possible to execute before mouse move (TODO)
+ return;
+ }
+
+ if (!menu.cssOnly) {
+ menu.cssOnly = true;
+ }
+
+ var source = null;
+ if (e && e.isRow && e.$.row.task) {
+ source = new DayPilot.Task(e.$.row.task, e.calendar);
+ source.menuType = "resource";
+ }
+ else if (e && e.isEvent && e.data.task) {
+ source = new DayPilot.Task(e, e.calendar);
+ }
+ else {
+ source = e;
+ }
+
+ if (typeof menu.onShow === "function") {
+ var args = {};
+ args.source = source;
+ args.menu = menu;
+ args.preventDefault = function () {
+ args.preventDefault.value = true;
+ };
+ menu.onShow(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+ div.style.top = "0px";
+ div.style.left = "0px";
+ div.style.display = 'none';
+ div.style.overflow = 'hidden';
+ div.style.zIndex = this.zIndex + 1;
+ div.className = this._applyCssClass('main');
+ div.onclick = function(ev) {
+ ev.cancelBubble = true;
+ this.parentNode.removeChild(this);
+ };
+
+ if (this.hideOnMouseOut) {
+ div.onmousemove = function(ev) {
+ clearTimeout(DayPilotMenu.hideTimeout);
+ };
+ div.onmouseleave = function(ev) {
+ menu.delayedHide({"hideParent": true});
+ };
+ }
+
+ if (!this.items || this.items.length === 0) {
+ throw "No menu items defined.";
+ }
+
+ if (this.showMenuTitle) {
+ var title = document.createElement("div");
+ title.innerHTML = this.menuTitle;
+ title.className = this._applyCssClass("title");
+ div.appendChild(title);
+ }
+
+ for (var i = 0; i < this.items.length; i++) {
+ var mi = this.items[i];
+ var item = document.createElement("div");
+ DayPilot.Util.addClass(item, this._applyCssClass("item"));
+ if (mi.items) {
+ DayPilot.Util.addClass(item, this._applyCssClass("item_haschildren"));
+ DayPilot.Util.addClass(div, this._applyCssClass(("withchildren")));
+ }
+
+ if (typeof mi === 'undefined') {
+ continue;
+ }
+
+ if (mi.hidden) {
+ continue;
+ }
+
+
+ if (mi.text === '-') {
+ var separator = document.createElement("div");
+ separator.addEventListener("click", function(ev) {
+ ev.stopPropagation();
+ });
+
+ item.appendChild(separator);
+ }
+ else {
+ var link = document.createElement("a");
+ link.style.position = 'relative';
+ link.style.display = "block";
+
+ if (mi.cssClass) {
+ DayPilot.Util.addClass(link, mi.cssClass);
+ }
+
+ if (mi.disabled) {
+ DayPilot.Util.addClass(link, menu._applyCssClass("item_disabled"));
+ }
+ else {
+ if (mi.onclick || mi.onClick) {
+ link.item = mi;
+ link.onclick = (function(mi, link) {
+ return function (e) {
+ if (typeof mi.onClick === "function") {
+ var args = {};
+ args.item = mi;
+ args.source = link.source;
+ args.originalEvent = e;
+ args.preventDefault = function () {
+ args.preventDefault.value = true;
+ };
+ mi.onClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+ if (mi.onclick) {
+ mi.onclick.call(link, e);
+ }
+ };
+ })(mi, link);
+
+ var assignTouchEnd = function(mi, link) {
+ return function(e) {
+ e.stopPropagation();
+ e.preventDefault();
+
+ var cleanup = function() {
+ window.setTimeout(function() {
+ link.source.calendar.internal.touch.active = false;
+ }, 500);
+ };
+
+ if (typeof mi.onClick === "function") {
+ var args = {};
+ args.item = mi;
+ args.source = link.source;
+ args.originalEvent = e;
+ args.preventDefault = function() {
+ args.preventDefault.value = true;
+ };
+ mi.onClick(args);
+ if (args.preventDefault.value) {
+ cleanup();
+ return;
+ }
+ }
+
+ if (mi.onclick) {
+ mi.onclick.call(link, e);
+ }
+
+ DayPilotMenu.menuClean();
+ cleanup();
+ };
+ };
+
+ DayPilot.reNonPassive(link, "touchstart", function(ev) {
+ ev.stopPropagation();
+ ev.preventDefault();
+
+ link.source.calendar.internal.touch.active = true;
+ });
+ DayPilot.reNonPassive(link, "touchend", assignTouchEnd(mi, link));
+
+ // link.ontouchend = assignTouchEnd(mi, link);
+ }
+
+
+ if (mi.items && !mi.disabled) {
+ var assign = function(mi, link) {
+ return function(ev) {
+ ev.preventDefault();
+ ev.stopPropagation();
+ menu._showSubmenu(mi, link);
+ };
+ };
+ link.ontouchend = assign(mi, link);
+ }
+
+ if (mi.onclick) {
+ doNothing();
+ }
+ else if (mi.href) {
+ link.href = mi.href.replace(/\x7B0\x7D/gim, value); // for NavigateUrl actions, only for backwards compatibility
+ if (mi.target) {
+ link.setAttribute("target", mi.target);
+ }
+ }
+ else if (mi.command) {
+ var assign = function(mi, link) {
+ return function(e) {
+ var source = link.source;
+ var item = mi;
+ item.action = item.action ? item.action : 'CallBack';
+ var cal = source.calendar || source.root;
+
+ if (source instanceof DayPilot.Link) {
+ cal.internal.linkMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Selection) {
+ cal.internal.timeRangeMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Event) {
+ cal.internal.eventMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Selection) {
+ cal.internal.timeRangeMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Task) {
+ if (source.menuType === "resource") {
+ cal.internal.resourceHeaderMenuClick(item.command, link.menuSource, item.action);
+ }
+ else {
+ cal.internal.eventMenuClick(item.command, link.menuSource, item.action);
+ }
+ return;
+ }
+ else {
+ switch (source.menuType) { // TODO legacy, remove
+ case 'resource':
+ cal.internal.resourceHeaderMenuClick(item.command, source, item.action);
+ return;
+ case 'selection': // fully replaced
+ cal.internal.timeRangeMenuClick(item.command, source, item.action);
+ return;
+ default: // fully replaced
+ cal.internal.eventMenuClick(item.command, source, item.action);
+ return;
+ }
+ }
+
+ e.preventDefault();
+ };
+ };
+ link.onclick = assign(mi, link);
+ link.ontouchend = assign(mi, link);
+ }
+
+ }
+
+
+ if (mi.items) {
+ link.addEventListener("click", function(ev) {
+ ev.stopPropagation();
+ });
+ }
+
+ link.source = source;
+ link.menuSource = e;
+
+ var span = document.createElement("span");
+ span.className = menu._applyCssClass("item_text");
+ span.innerHTML = DayPilot.Util.escapeTextHtml(mi.text, mi.html);
+ link.appendChild(span);
+
+ if (mi.image) {
+ var image = document.createElement("img");
+ image.src = mi.image;
+ image.style.position = 'absolute';
+ image.style.top = '0px';
+ image.style.left = '0px';
+
+ link.appendChild(image);
+ }
+
+ if (mi.icon) {
+ var icon = document.createElement("span");
+ icon.className = menu._applyCssClass("item_icon");
+
+ var iel = document.createElement("i");
+ iel.className = mi.icon;
+ icon.appendChild(iel);
+
+ link.appendChild(icon);
+ }
+
+ if (mi.symbol) {
+ var ns = "http://www.w3.org/2000/svg";
+ var svg = document.createElementNS(ns,"svg");
+ svg.setAttribute("width", "100%");
+ svg.setAttribute("height", "100%");
+ var use = document.createElementNS(ns,"use");
+ use.setAttribute("href", mi.symbol);
+ svg.appendChild(use);
+
+ var svgWrap = document.createElement("span");
+ svgWrap.className = menu._applyCssClass("item_symbol");
+ svgWrap.style.position = "absolute";
+ svgWrap.style.top = "0px";
+ svgWrap.style.left = "0px";
+ svgWrap.appendChild(svg);
+
+ link.appendChild(svgWrap);
+ }
+
+ var assignOnMouseOver = function(mi, link) {
+ return function() {
+ var source = link.source;
+ var item = mi;
+
+ var ws = DayPilotMenu.waitingSubmenu;
+ if (ws) {
+ if (ws.parent === item) {
+ return;
+ }
+ else {
+ clearTimeout(ws.timeout);
+ DayPilotMenu.waitingSubmenu = null;
+ }
+ }
+
+ if (mi.disabled) {
+ return;
+ }
+
+ DayPilotMenu.waitingSubmenu = {};
+ DayPilotMenu.waitingSubmenu.parent = item;
+ DayPilotMenu.waitingSubmenu.timeout = setTimeout(function() {
+
+ DayPilotMenu.waitingSubmenu = null;
+
+ menu._showSubmenu(item, link);
+
+ }, 300);
+ };
+ };
+
+ link.onmouseover = assignOnMouseOver(mi, link);
+
+ item.appendChild(link);
+ }
+
+ div.appendChild(item);
+
+ }
+
+ var delayedDismiss = function(e) {
+ window.setTimeout(function() {
+ DayPilotMenu.menuClean();
+ DayPilot.MenuBar.deactivate();
+ }, 100);
+ };
+
+ div.onclick = delayedDismiss;
+ div.ontouchend = delayedDismiss;
+
+ div.onmousedown = function(e) {
+ e = e || window.event;
+ e.cancelBubble = true;
+ if (e.stopPropagation)
+ e.stopPropagation();
+ };
+ div.oncontextmenu = function() {
+ return false;
+ };
+
+ document.body.appendChild(div);
+ menu._state.visible = true;
+ menu._state.source = e;
+
+ div.style.display = '';
+ var height = div.offsetHeight;
+ var width = div.offsetWidth;
+ div.style.display = 'none';
+
+ // don't show the menu outside of the visible window
+ var windowHeight = document.documentElement.clientHeight;
+ // required for ipad with zoom instead of document.documentElement.clientWidth
+ var windowWidth = window.innerWidth;
+
+ var windowMargin = (typeof options.windowMargin == "number") ? options.windowMargin : 5;
+
+ (function showInitiator() {
+ var initiator = options.initiator;
+ // initiator = options.initiator;
+ if (!initiator) {
+ return;
+ }
+ var div = initiator.div;
+ var e = initiator.e;
+ var area = initiator.area;
+
+ var v = area.visibility || area.v || "Visible";
+ var a = initiator.a;
+ if (v !== "Visible") {
+ // make sure the source area is visible
+ a = DayPilot.Areas.createArea(div, e, area);
+ div.appendChild(a);
+ // will be used to remove it on hide
+ initiatorAreaDiv = a;
+ }
+
+ if (a) {
+ var abs = DayPilot.abs(a);
+ options.x = abs.x;
+ options.y = abs.y + abs.h + 2;
+ }
+
+ })();
+
+
+ (function adjustPosition() {
+
+ // don't show it exactly under the cursor
+ var x = (typeof options.x === "number") ? options.x : DayPilotMenu.mouse.x + 1;
+ var y = (typeof options.y === "number") ? options.y : DayPilotMenu.mouse.y + 1;
+
+ var topOffset = document.body.scrollTop || document.documentElement.scrollTop;
+ var leftOffset = document.body.scrollLeft || document.documentElement.scrollLeft;
+
+ if (y - topOffset > windowHeight - height && windowHeight !== 0) {
+ var offsetY = y - topOffset - (windowHeight - height) + windowMargin;
+ div.style.top = (y - offsetY) + 'px';
+ }
+ else {
+ div.style.top = y + 'px';
+ }
+
+ if (options.align === "right") {
+ x -= width;
+ }
+
+ if (x - leftOffset > windowWidth - width && windowWidth !== 0) {
+ var offsetX = x - leftOffset - (windowWidth - width) + windowMargin;
+ div.style.left = (x - offsetX) + 'px';
+ }
+ else {
+ div.style.left = x + 'px';
+ }
+/*
+ if (DayPilotMenu.mouse.clientY > windowHeight - height && windowHeight !== 0) {
+ var offsetY = DayPilotMenu.mouse.clientY - (windowHeight - height) + 5;
+ div.style.top = (y - offsetY) + 'px';
+ }
+ else {
+ div.style.top = y + 'px';
+ }
+
+ if (DayPilotMenu.mouse.clientX > windowWidth - width && windowWidth !== 0) {
+ var offsetX = DayPilotMenu.mouse.clientX - (windowWidth - width) + 5;
+ div.style.left = (x - offsetX) + 'px';
+ }
+ else {
+ div.style.left = x + 'px';
+ }
+*/
+ })();
+
+ if (options.parentLink) {
+
+ var parent = options.parentLink;
+
+ var verticalOffset = parseInt(new DayPilot.StyleReader(div).get("border-top-width"));
+
+ var pos = DayPilot.abs(options.parentLink.parentNode);
+ var x = pos.x + parent.offsetWidth;
+ var y = pos.y - verticalOffset;
+
+ if (x + width > windowWidth) {
+ x = Math.max(0, pos.x - width);
+ }
+
+ var docScrollTop = document.body.scrollTop + document.documentElement.scrollTop;
+ if (y + height - docScrollTop > windowHeight) {
+ y = Math.max(0, windowHeight - height + docScrollTop);
+ }
+
+ div.style.left = x + "px";
+ div.style.top = y + "px";
+
+ }
+ div.style.display = '';
+
+ this.addShadow(div);
+ this._state.div = div;
+
+ if (!options.submenu) {
+ DayPilot.Menu.active = this;
+ }
+
+ //this._initiator = null;
+
+ };
+
+ this._showSubmenu = function(item, link) {
+ var mi = item;
+ var source = link.source;
+
+ if (menu._state.submenu && menu._state.submenu.item === item) { // already visible
+ return;
+ }
+
+ if (menu._state.submenu && menu._state.submenu.item !== item) { // hide submenus of other items
+ DayPilot.Util.removeClass(menu._state.submenu.link.parentNode, menu._applyCssClass("item_haschildren_active"));
+ menu._state.submenu.menu.hide();
+ menu._state.submenu = null;
+ }
+
+ if (!item.items) { // no submenu for this item
+ return;
+ }
+
+ var options = menu.cloneOptions();
+ options.items = item.items;
+
+ menu._state.submenu = {};
+ menu._state.submenu.menu = new DayPilot.Menu(options);
+ menu._state.submenu.menu._parentMenu = menu;
+ menu._state.submenu.menu.show(source, {"submenu": true, "parentLink": link, "parentItem": mi});
+ menu._state.submenu.item = item;
+ menu._state.submenu.link = link;
+ DayPilot.Util.addClass(link.parentNode, menu._applyCssClass("item_haschildren_active"));
+
+ };
+
+ this._applyCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ var sep = (this.cssOnly ? "_" : "");
+ if (prefix) {
+ return prefix + sep + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this.cloneOptions = function() {
+ return DayPilot.Util.copyProps(options, {}, ["cssClassPrefix", "theme", "hideAfter", "hideOnMouseOut", "zIndex"]);
+ };
+
+ this.hide = function(props) {
+ props = props || {};
+
+ if (this._state.submenu) {
+ this._state.submenu.menu.hide();
+ }
+
+ var ws = DayPilotMenu.waitingSubmenu;
+ if (ws) {
+ DayPilotMenu.waitingSubmenu = null;
+ clearTimeout(ws.timeout);
+ }
+
+ this.removeShadow();
+ if (this._state.div && this._state.div.parentNode === document.body) {
+ document.body.removeChild(this._state.div);
+ }
+
+/*
+ if (this._initiator) {
+ DayPilot.de(this._initiator);
+ this._initiator = null;
+ }
+*/
+
+ if (initiatorAreaDiv) {
+ DayPilot.de(initiatorAreaDiv);
+ initiatorAreaDiv = null;
+ }
+
+ menu._state.visible = false;
+ menu._state.source = null;
+
+ if (menu._parentMenu && props.hideParent) {
+ menu._parentMenu.hide(props);
+ }
+
+ if (DayPilot.Menu.active === menu) {
+ DayPilot.Menu.active = null;
+ }
+
+ if (typeof this.onHide === "function") {
+ var args = {};
+ this.onHide(args);
+ }
+
+ };
+
+ this.delayedHide = function(props) {
+ DayPilotMenu.hideTimeout = setTimeout(function() {
+ menu.hide(props);
+ }, menu.hideAfter);
+ };
+
+ this.cancelHideTimeout = function() {
+ clearTimeout(DayPilotMenu.hideTimeout);
+ };
+
+ // detects the mouse position, use when creating menu right before opening (.show)
+ this.init = function(ev) {
+ DayPilotMenu.mouseMove(ev);
+ return this;
+ };
+
+ // disabled
+ this.addShadow = function(object) {};
+
+ // disabled
+ this.removeShadow = function() {
+ /* if (!this._state.shadows) {
+ return;
+ }
+
+ for (var i = 0; i < this._state.shadows.length; i++) {
+ document.body.removeChild(this._state.shadows[i]);
+ }
+ this._state.shadows = [];*/
+ };
+
+
+ var options = DayPilot.isArray(items) ? null : items;
+ if (options) {
+ for (var name in options) {
+ this[name] = options[name];
+ }
+ }
+
+ };
+
+ DayPilot.MenuBar = function(id, options) {
+ var menubar = this;
+
+ options = options || {};
+
+ this.items = [];
+ this.theme = "menubar_default";
+ this.windowMargin = 0;
+
+ this.nav = {};
+ this.elements = {};
+ this.elements.items = DayPilot.list();
+
+ this._active = null;
+ this._initialized = false;
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+
+ this._cssClass = function(cl) {
+ return this.theme + "_" + cl;
+ };
+
+ this._show = function() {
+ this.nav.top = document.getElementById(id);
+
+ var top = this.nav.top;
+ top.className = this._cssClass("main");
+
+ DayPilot.list(menubar.items).forEach(function(item) {
+ var div = document.createElement("span");
+ div.innerHTML = DayPilot.Util.escapeTextHtml(item.text, item.html);
+ div.className = menubar._cssClass("item");
+ if (item.cssClass) {
+ div.classList.add(item.cssClass);
+ }
+ div.data = item;
+ div.onclick = function(e) {
+ if (menubar.active && menubar.active.item === item) {
+ menubar._hideActive();
+ }
+ else if (item.children) {
+ menubar._activate(div);
+ return;
+ }
+
+ if (typeof item.onClick === "function") {
+ var args = {};
+ args.item = item;
+ args.originalEvent = e;
+ item.onClick(args);
+ }
+ };
+ div.onmousedown = function(ev) {
+ ev.stopPropagation();
+ };
+ div.onmouseover = function() {
+ if (menubar.active && menubar.active.item !== item) {
+ menubar._activate(div);
+ }
+ };
+
+ top.appendChild(div);
+ menubar.elements.items.push(div);
+ });
+ };
+
+ this._hideActive = function() {
+ var activeCss = menubar._cssClass("item_active");
+ menubar.elements.items.forEach(function(div) {
+ DayPilot.Util.removeClass(div, activeCss);
+ });
+
+ if (menubar.active && menubar.active.menu) {
+ menubar.active.menu.hide();
+ }
+ menubar.active = null;
+ };
+
+ this._isActive = function(div) {
+ if (!menubar.active) {
+ return false;
+ }
+ return menubar.active.item === div.data;
+ };
+
+ this._activate = function(div) {
+ if (menubar._isActive(div)) {
+ return;
+ }
+
+ menubar._hideActive();
+
+ var item = div.data;
+ var a = menubar.active = {};
+ a.item = item;
+ a.div = div;
+
+ var activeCss = menubar._cssClass("item_active");
+ DayPilot.Util.addClass(div, activeCss);
+
+ var abs = DayPilot.abs(div);
+
+ if (item.children) {
+ a.menu = new DayPilot.Menu({"items": item.children});
+ // a.menu.show(null, { "x": abs.x + abs.w, "y": abs.y + abs.h, "align": item.align, "windowMargin": menubar.windowMargin});
+ var x = abs.x;
+ if (item.align === "right") {
+ x += abs.w;
+ }
+ a.menu.show(null, { "x": x, "y": abs.y + abs.h, "align": item.align, "windowMargin": menubar.windowMargin});
+ }
+
+ DayPilot.MenuBar.active = menubar;
+ };
+
+ this.init = function() {
+ this._show();
+ this._initialized = true;
+ return this;
+ };
+
+ this.dispose = function() {
+ if (!this._initialized) {
+ return;
+ }
+ this.nav.top.innerHTML = "";
+ this.elements.items = [];
+ };
+
+ };
+
+ DayPilot.MenuBar.deactivate = function() {
+ if (DayPilot.MenuBar.active) {
+ DayPilot.MenuBar.active._hideActive();
+ DayPilot.MenuBar.active = null;
+ }
+ };
+
+ DayPilotMenu.menuClean = function() {
+ if (typeof(DayPilot.Menu.active) === 'undefined')
+ return;
+
+ if (DayPilot.Menu.active) {
+ DayPilot.Menu.active.hide();
+ DayPilot.Menu.active = null;
+ }
+
+ };
+
+ DayPilotMenu.mouseDown = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ DayPilotMenu.menuClean();
+
+ DayPilot.MenuBar.deactivate();
+ };
+
+ DayPilotMenu.mouseMove = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ DayPilotMenu.mouse = DayPilotMenu.mousePosition(ev);
+ };
+
+ DayPilotMenu.touchMove = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ DayPilotMenu.mouse = DayPilotMenu.touchPosition(ev);
+ };
+
+ DayPilotMenu.touchStart = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ //DayPilotMenu.menuClean();
+ DayPilotMenu.mouse = DayPilotMenu.touchPosition(ev);
+ };
+
+ DayPilotMenu.touchEnd = function(ev) {
+ // do not call menuClean() here, it doesn't work with eventTapAndHoldHandling="ContextMenu"
+ // DayPilotMenu.menuClean();
+ };
+
+ DayPilotMenu.touchPosition = function(ev) {
+ if (!ev || !ev.touches) {
+ return null;
+ }
+ var touch = ev.touches[0];
+ var mouse = {};
+ mouse.x = touch.pageX;
+ mouse.y = touch.pageY;
+ return mouse;
+ };
+
+ DayPilotMenu.mousePosition = function(e) {
+ return DayPilot.mo3(null, e);
+ };
+
+ DayPilot.Menu.touchPosition = function(ev) {
+ if (ev.touches) {
+ DayPilotMenu.mouse = DayPilotMenu.touchPosition(ev);
+ }
+ };
+ // publish the API
+
+ DayPilot.Menu.hide = function(options) {
+ options = options || {};
+
+ if (options.calendar) {
+ var active = DayPilot.Menu.active;
+ if (active) {
+ var source = active._state.source;
+ if (source && source.calendar === options.calendar) {
+ DayPilotMenu.menuClean();
+ }
+ }
+ }
+ else {
+ DayPilotMenu.menuClean();
+ }
+ };
+
+ // current
+ //DayPilot.Menu = DayPilotMenu.Menu;
+ if (!DayPilotMenu.handlersRegistered && typeof document !== 'undefined') {
+ DayPilot.re(document, 'mousemove', DayPilotMenu.mouseMove);
+ DayPilot.re(document, 'mousedown', DayPilotMenu.mouseDown);
+ DayPilot.re(document, 'touchmove', DayPilotMenu.touchMove);
+ DayPilot.re(document, 'touchstart', DayPilotMenu.touchStart);
+ DayPilot.re(document, 'touchend', DayPilotMenu.touchEnd);
+ DayPilotMenu.handlersRegistered = true;
+ }
+
+ DayPilot.Menu.def = {};
+
+})(DayPilot);
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+
+v3.15.0
+https://modal.daypilot.org/
+
+*/
+
+if (typeof(DayPilot) === 'undefined') {
+ DayPilot = {};
+}
+
+(function(DayPilot) {
+ 'use strict';
+
+ // modal.js -> 1482
+
+ if (DayPilot.ModalStatic) {
+ return;
+ }
+
+ DayPilot.ModalStatic = {};
+
+ DayPilot.ModalStatic.list = [];
+
+ // hide the last one
+ DayPilot.ModalStatic.hide = function() {
+ if (this.list.length > 0) {
+ var last = this.list.pop();
+ if (last) {
+ last.hide();
+ }
+ }
+ };
+
+ DayPilot.ModalStatic.remove = function(modal) {
+ var list = DayPilot.ModalStatic.list;
+ for (var i = 0; i < list.length; i++) {
+ if (list[i] === modal) {
+ list.splice(i, 1);
+ return;
+ }
+ }
+ };
+
+ DayPilot.ModalStatic.close = function(result) {
+ DayPilot.ModalStatic.result(result);
+ DayPilot.ModalStatic.hide();
+ };
+
+ DayPilot.ModalStatic.result = function(r) {
+ var list = DayPilot.ModalStatic.list;
+ if (list.length > 0) {
+ list[list.length - 1].result = r;
+ }
+ };
+
+ DayPilot.ModalStatic.displayed = function(modal) {
+ var list = DayPilot.ModalStatic.list;
+ for (var i = 0; i < list.length; i++) {
+ if (list[i] === modal) {
+ return true;
+ }
+ }
+ return false;
+ };
+
+ DayPilot.ModalStatic.stretch = function() {
+ if (this.list.length > 0) {
+ var last = this.list[this.list.length - 1];
+ if (last) {
+ last.stretch();
+ }
+ }
+ };
+
+ DayPilot.ModalStatic.last = function() {
+ var list = DayPilot.ModalStatic.list;
+ if (list.length > 0) {
+ return list[list.length - 1];
+ }
+ return null;
+ };
+
+ var Sheet = function() {
+
+ if (typeof window === "undefined") {
+ // next.js server-side rendering
+ var sheet = {};
+ sheet.add = function() {};
+ sheet.commit = function() {};
+ return sheet;
+ }
+
+ var style = document.createElement("style");
+ style.setAttribute("type", "text/css");
+ if (!style.styleSheet) { // ie
+ style.appendChild(document.createTextNode(""));
+ }
+
+ var h = document.head || document.getElementsByTagName('head')[0];
+ h.appendChild(style);
+
+ var oldStyle = !! style.styleSheet; // old ie
+
+ var sheet = {};
+ sheet.rules = [];
+ sheet.commit = function() {
+ try {
+ if (oldStyle) {
+ style.styleSheet.cssText = this.rules.join("\n");
+ }
+ }
+ catch (e) {
+ //alert("Error registering the built-in stylesheet (IE stylesheet limit reached). Stylesheet count: " + document.styleSheets.length);
+ }
+ };
+
+ sheet.add = function(selector, rules, index) {
+ if (oldStyle) {
+ this.rules.push(selector + "{" + rules + "\u007d");
+ return;
+ }
+ if(style.sheet.insertRule) { // normal browsers, ie9+
+ if (typeof index === "undefined") {
+ index = style.sheet.cssRules.length;
+ }
+ style.sheet.insertRule(selector + "{" + rules + "\u007d", index);
+ }
+ else if (style.sheet.addRule) {
+ style.sheet.addRule(selector, rules, index);
+ }
+ else {
+ throw "No CSS registration method found";
+ }
+ };
+ return sheet;
+ };
+
+ var iconCalendar = "";
+ var iconExpand = "";
+
+ var sheet = new Sheet();
+ sheet.add(".modal_default_main", "border: 10px solid #ccc; max-width: 90%;");
+ sheet.add(".modal_default_main:focus", "outline: none;");
+ sheet.add(".modal_default_content", "padding: 10px 0px;");
+ sheet.add(".modal_default_inner", "padding: 20px;");
+ sheet.add(".modal_default_input", "padding: 10px 0px;");
+ sheet.add(".modal_default_buttons", "margin-top: 10px;");
+ sheet.add(".modal_default_buttons", "padding: 10px 0px;");
+ sheet.add(".modal_default_form_item", "padding: 10px 0px; position: relative;");
+ sheet.add(".modal_default_form_item_level1", "border-left: 2px solid #ccc; margin-left: 10px; padding-left: 20px;");
+ sheet.add(".modal_default_form_item.modal_default_form_title", "font-size: 1.5rem; font-weight: bold;");
+ sheet.add(".modal_default_form_item input[type=text]", "width: 100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item textarea", "width: 100%; height: 200px; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item input[type=select]", "width: 100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item label", "display: block;");
+ sheet.add(".modal_default_form_item select", "width: 100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item_label", "margin: 2px 0px;");
+ sheet.add(".modal_default_form_item_image img", "max-width: 100%; height: auto;");
+
+ sheet.add(".modal_default_form_item_invalid", "");
+ sheet.add(".modal_default_form_item_invalid_message", "position: absolute; right: 0px; top: 9px; background-color: red; color: #ffffff; padding: 2px; border-radius: 2px;");
+
+ sheet.add(".modal_default_background", "opacity: 0.5; background-color: #000;");
+ sheet.add(".modal_default_ok", "padding: 3px; width: 80px;");
+ sheet.add(".modal_default_cancel", "padding: 3px; width: 80px;");
+
+ sheet.add(".modal_default_form_item_date", "position: relative;");
+ sheet.add(".modal_default_form_item_date:after", "content: ''; position: absolute; right: 7px; top: 50%; margin-top: 3px; width: 10px; height: 15px; background-image:url(" + iconCalendar + ")");
+
+ if (typeof navigator !== "undefined" && navigator.userAgent.indexOf("Edge") !== -1) {
+ sheet.add(".modal_default_form_item_date input::-ms-clear", "display: none;");
+ }
+
+ sheet.add(".modal_default_form_item_scrollable_scroll", "width: 100%; height: 200px; box-sizing: border-box; border: 1px solid #ccc; overflow-y: auto;");
+ sheet.add(".modal_default_form_item_scrollable_scroll_content", "padding: 5px;");
+
+ sheet.add(".modal_default_form_item_searchable", "position: relative;");
+ sheet.add(".modal_default_form_item_searchable_icon", "");
+ sheet.add(".modal_default_form_item_searchable_icon:after", "content:''; position: absolute; right: 5px; top: 50%; margin-top: -8px; width: 10px; height: 15px; background-image:url(" + iconExpand + ");");
+ sheet.add(".modal_default_form_item_searchable_list", "box-sizing: border-box; border: 1px solid #999; max-height: 150px; overflow-y: auto;");
+ sheet.add(".modal_default_form_item_searchable_list_item", "background: white; padding: 2px; cursor: default;");
+ sheet.add(".modal_default_form_item_searchable_list_item_highlight", "background: #ccc;");
+
+ sheet.add(".modal_default_form_item_time", "position: relative;");
+ sheet.add(".modal_default_form_item_time_icon", "");
+ sheet.add(".modal_default_form_item_time_icon:after", "content:''; position: absolute; right: 5px; top: 50%; margin-top: -8px; width: 10px; height: 15px; background-image:url(" + iconExpand + ");");
+ sheet.add(".modal_default_form_item_time_list", "box-sizing: border-box; border: 1px solid #999; max-height: 150px; overflow-y: auto;");
+ sheet.add(".modal_default_form_item_time_list_item", "background: white; padding: 2px; cursor: default;");
+ sheet.add(".modal_default_form_item_time_list_item_highlight", "background: #ccc;");
+
+ sheet.add(".modal_default_form_item_datetime_parent", "display: flex;");
+ sheet.add(".modal_default_form_item_datetime .modal_default_form_item_time_main", "margin-left: 5px;");
+ sheet.add(".modal_default_form_item_datetime input[type='text'].modal_default_input_date ", "");
+
+ sheet.add(".modal_default_form_item_tabular_main", "margin-top: 10px;");
+ sheet.add(".modal_default_form_item_tabular_table", "display: table; width: 100%; xbackground-color: #fff; border-collapse: collapse;");
+ sheet.add(".modal_default_form_item_tabular_tbody", "display: table-row-group;");
+ sheet.add(".modal_default_form_item_tabular_row", "display: table-row;");
+ sheet.add(".modal_default_form_item_tabular_row.modal_default_form_item_tabular_header", ""); // used to be bold
+ sheet.add(".modal_default_form_item_tabular_cell.modal_default_form_item_tabular_rowaction", "padding: 0px; width: 23px;"); // _delete: width + marginLeft
+ sheet.add(".modal_default_form_item_tabular_cell", "display: table-cell; border: 0px; padding: 2px 2px 2px 0px; cursor: default; vertical-align: bottom;");
+ sheet.add(".modal_default_form_item_tabular_header .modal_default_form_item_tabular_cell", "padding-left: 0px; padding-bottom: 0px;");
+ sheet.add(".modal_default_form_item_tabular_table input[type=text], .modal_default_form_item_tabular_table input[type=number]", "width:100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item_tabular_table select", "width:100%; height:100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item_tabular_plus", "display: inline-block; background-color: #ccc; color: white; width: 20px; height: 20px; border-radius: 10px; box-sizing: border-box; position: relative; margin-left: 3px; margin-top: 3px; cursor: pointer;");
+ sheet.add(".modal_default_form_item_tabular_plus:after", "content: ''; position: absolute; left: 5px; top: 5px; width: 10px; height: 10px; background-image: url(\"\")");
+ sheet.add(".modal_default_form_item_tabular_delete", "display: inline-block; background-color: #ccc; color: white; width: 20px; height: 20px; border-radius: 10px; box-sizing: border-box; position: relative; margin-left: 3px; margin-top: 3px; cursor: pointer;");
+ sheet.add(".modal_default_form_item_tabular_delete:after", "content: ''; position: absolute; left: 5px; top: 5px; width: 10px; height: 10px; background-image: url(\"\")");
+ sheet.add(".modal_default_form_item_tabular_disabled .modal_default_form_item_tabular_plus", "display: none;");
+ sheet.add(".modal_default_form_item_tabular_plus_max.modal_default_form_item_tabular_plus", "display: none;");
+ sheet.add(".modal_default_form_item_tabular_disabled .modal_default_form_item_tabular_delete", "visibility: hidden;");
+ sheet.add(".modal_default_form_item_tabular_empty", "height: 1px; margin: 5px 23px 5px 0px; background-color: #ccc;");
+ sheet.add(".modal_default_form_item_tabular_spacer .modal_default_form_item_tabular_cell", "padding: 0px;");
+
+ sheet.add(".modal_min_main", "border: 1px solid #ccc; max-width: 90%;");
+ sheet.add(".modal_min_background", "opacity: 0.5; background-color: #000;");
+ sheet.add(".modal_min_ok", "padding: 3px 10px;");
+ sheet.add(".modal_min_cancel", "padding: 3px 10px;");
+
+
+ sheet.add(".navigator_modal_main", "border-left: 1px solid #c0c0c0;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;background-color: white;color: #000000; box-sizing: content-box;");
+ sheet.add(".navigator_modal_main *, .navigator_modal_main *:before, .navigator_modal_main *:after", "box-sizing: content-box;");
+ sheet.add(".navigator_modal_month", "font-size: 11px;");
+ sheet.add(".navigator_modal_day", "color: black;");
+ sheet.add(".navigator_modal_weekend", "background-color: #f0f0f0;");
+ sheet.add(".navigator_modal_dayheader", "color: black;");
+ sheet.add(".navigator_modal_line", "border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".navigator_modal_dayother", "color: gray;");
+ sheet.add(".navigator_modal_todaybox", "border: 1px solid red;");
+ sheet.add(".navigator_modal_title, .navigator_modal_titleleft, .navigator_modal_titleright", 'border-top: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;color: #333;background: #f3f3f3;');
+ sheet.add(".navigator_modal_busy", "font-weight: bold;");
+ sheet.add(".navigator_modal_cell", "text-align: center;");
+ sheet.add(".navigator_modal_select .navigator_modal_cell_box", "background-color: #FFE794; opacity: 0.5;");
+ sheet.add(".navigator_modal_title", "text-align: center;");
+ sheet.add(".navigator_modal_titleleft, .navigator_modal_titleright", "text-align: center;");
+ sheet.add(".navigator_modal_dayheader", "text-align: center;");
+ sheet.add(".navigator_modal_weeknumber", "text-align: center;");
+ sheet.add(".navigator_modal_cell_text", "cursor: pointer;");
+
+ sheet.commit();
+
+ DayPilot.Modal = function(options) {
+
+ // default values
+ this.autoFocus = true;
+ this.focus = null; // field to be focused - field id as string or {id, value}
+ this.autoStretch = true; // height will be increased automatically to avoid scrollbar, until this.maxHeight is reached
+ this.autoStretchFirstLoadOnly = false;
+ this.className = null;
+ this.theme = "modal_default";
+ this.disposeOnClose = true;
+ this.dragDrop = true;
+ this.loadingHtml = null;
+ this.maxHeight = null; // if not set, it will stretch until the bottom space is equal to this.top
+ this.scrollWithPage = true; // modal window will scroll with the page
+ this.useIframe = true; // only for showHtml()
+ this.zIndex = 99999;
+
+ this.left = null; // will be centered if null
+ this.width = 600;
+ this.top = 20;
+ this.height = 200; // see also autoStretch
+ this.locale = null; // used for DayPilot.Modal.form() items with "date" type
+
+ // event handler
+ this.closed = null;
+ this.onClose = null;
+ this.onClosed = null;
+ this.onShow = null;
+
+ // internal
+ var This = this;
+ this.id = '_' + new Date().getTime() + 'n' + (Math.random() * 10);
+ this._registered = false;
+
+ // drag&drop
+ this._start = null;
+ this._coords = null;
+
+ this.showHtml = function(html) {
+
+ if (DayPilot.ModalStatic.displayed(this)) {
+ throw "This modal dialog is already displayed.";
+ }
+
+ if (!this.div) {
+ this._create();
+ }
+ this._update();
+
+ if (this.useIframe) {
+ var delayed = function(p, innerHTML) {
+ return function() {
+ p.setInnerHTML(p.id + "iframe", innerHTML);
+ };
+ };
+
+ window.setTimeout(delayed(this, html), 0);
+ }
+ else {
+ if (html.nodeType) {
+ this.div.appendChild(html);
+ }
+ else {
+ this.div.innerHTML = html;
+ }
+ }
+
+ this._update();
+ this._register();
+ this._doShow();
+
+ };
+
+ this.showUrl = function(url) {
+
+ if (DayPilot.ModalStatic.displayed(this)) {
+ throw "This modal dialog is already displayed.";
+ }
+
+ //this.useIframe = true; // forced
+
+ if (this.useIframe) {
+ if (!this.div) {
+ this._create();
+ }
+
+ var loadingHtml = this.loadingHtml;
+ if (loadingHtml) {
+ this.iframe.src = "about:blank";
+ this.setInnerHTML(this.id + "iframe", loadingHtml);
+ }
+
+ this.re(this.iframe, "load", this._onIframeLoad);
+
+ this.iframe.src = url;
+ //this.iframe.contentWindow.modal = This;
+
+ this._update();
+ this._register();
+ this._doShow();
+ }
+ else {
+ This._ajax({
+ "url": url,
+ "success": function(args) {
+ var html = args.request.responseText;
+ This.showHtml(html);
+ },
+ "error": function(args) {
+ This.showHtml("Error loading the modal dialog");
+ }
+ });
+ }
+
+ };
+
+ this._doShow = function() {
+ if (typeof This.onShow === "function") {
+ var args = {};
+ args.root = This._body();
+ args.modal = This;
+ This.onShow(args);
+ }
+ };
+
+ this._body = function() {
+ return This.iframe ? This.iframe.contentWindow.document : This.div;
+ };
+
+ this._ajax = function(object) {
+ var req = new XMLHttpRequest();
+ if (!req) {
+ return;
+ }
+
+ var method = object.method || "GET";
+ var success = object.success || function() {};
+ var error = object.error || function() {};
+ var data = object.data;
+ var url = object.url;
+
+ req.open(method, url, true);
+ req.setRequestHeader('Content-type', 'text/plain');
+ req.onreadystatechange = function() {
+ if (req.readyState !== 4)
+ return;
+ if (req.status !== 200 && req.status !== 304) {
+ if (error) {
+ var args = {};
+ args.request = req;
+ error(args);
+ }
+ else {
+ if (window.console) { console.log('HTTP error ' + req.status); }
+ }
+ return;
+ }
+ var args = {};
+ args.request = req;
+ success(args);
+ };
+ if (req.readyState === 4) {
+ return;
+ }
+ if (typeof data === 'object') {
+ data = JSON.stringify(data);
+ }
+ req.send(data);
+ };
+
+ this._update = function() {
+
+ delete this.result;
+
+ var win = window;
+ var doc = document;
+
+ var scrollY = win.pageYOffset ? win.pageYOffset : ((doc.documentElement && doc.documentElement.scrollTop) ? doc.documentElement.scrollTop : doc.body.scrollTop);
+
+ var height = function() {
+ return This._windowRect().y;
+ };
+
+ //this.hideDiv.style.height = height() + "px";
+ if (this.theme) {
+ this.hideDiv.className = this.theme + "_background";
+ }
+ if (this.zIndex) {
+ this.hideDiv.style.zIndex = this.zIndex;
+ }
+ this.hideDiv.style.display = '';
+
+ window.setTimeout(function() {
+ if (This.hideDiv) {
+ This.hideDiv.onclick = function() {
+ This.hide({"backgroundClick": true});
+ };
+ }
+ }, 500);
+
+ if (this.theme) {
+ this.div.className = this.theme + "_main";
+ }
+ else {
+ this.div.className = "";
+ }
+
+ if (this.className) {
+ this.div.className += " " + this.className;
+ }
+ if (this.left) {
+ this.div.style.left = this.left + "px";
+ }
+ else {
+ this.div.style.marginLeft = '-' + Math.floor(this.width / 2) + "px"; // '-45%'
+ }
+ this.div.style.position = 'absolute';
+ this.div.style.boxSizing = "content-box";
+ this.div.style.top = (scrollY + this.top) + 'px';
+ this.div.style.width = this.width + 'px'; // '90%'
+ if (this.zIndex) {
+ this.div.style.zIndex = this.zIndex;
+ }
+
+ if (this.height) {
+ if (this.useIframe || !this.autoStretch) {
+ this.div.style.height = this.height + 'px';
+ }
+ else {
+ this.div.style.height = '';
+ }
+ }
+ if (this.useIframe && this.height) {
+ this.iframe.style.height = (this.height) + 'px';
+ }
+
+ this.div.style.display = '';
+
+ this._updateHorizontal();
+
+ // make sure it's there just once
+ DayPilot.ModalStatic.remove(this);
+ DayPilot.ModalStatic.list.push(this);
+
+ /*
+ if (this.iframe) {
+ this.iframe.onload = null;
+ }
+ */
+ };
+
+ this._onIframeLoad = function() {
+ This.iframe.contentWindow.modal = This;
+ if (This.autoStretch) {
+ This.stretch();
+ }
+ };
+
+ this.stretch = function() {
+ var height = function() {
+ return This._windowRect().y;
+ };
+
+ var width = function() {
+ return This._windowRect().x;
+ };
+
+
+ if (this.useIframe) {
+ // width first
+ var maxWidth = width() - 40; // fixed 20px margin
+ for (var w = this.width; w < maxWidth && this._hasHorizontalScrollbar(); w += 10) {
+ //this.iframe.style.width = (w) + 'px';
+ this.div.style.width = w + 'px';
+ this.div.style.marginLeft = '-' + Math.floor(w / 2) + "px"; //
+ }
+
+ // height
+ var maxHeight = this.maxHeight || height() - 2 * this.top;
+ for (var h = this.height; h < maxHeight && this._hasVerticalScrollbar(); h += 10) {
+ this.iframe.style.height = (h) + 'px';
+ this.div.style.height = h + 'px';
+ }
+
+ if (this.autoStretchFirstLoadOnly) {
+ this.ue(this.iframe, "load", this._onIframeLoad);
+ }
+ }
+ else {
+ this.div.style.height = '';
+ }
+
+
+ };
+
+ this._hasHorizontalScrollbar = function() {
+ var document = this.iframe.contentWindow.document;
+ var root = document.compatMode === 'BackCompat' ? document.body : document.documentElement;
+
+ var scrollWidth = root.scrollWidth;
+ var children = document.body.children;
+ for (var i = 0; i < children.length; i++) {
+ var bottom = children[i].offsetLeft + children[i].offsetWidth;
+ scrollWidth = Math.max(scrollWidth, bottom);
+ }
+
+ var isHorizontalScrollbar = scrollWidth > root.clientWidth;
+ return isHorizontalScrollbar;
+
+ };
+
+ this._hasVerticalScrollbar = function() {
+ var document = this.iframe.contentWindow.document;
+ var root = document.compatMode === 'BackCompat' ? document.body : document.documentElement;
+
+ var scrollHeight = root.scrollHeight;
+ var children = document.body.children;
+ for (var i = 0; i < children.length; i++) {
+ var bottom = children[i].offsetTop + children[i].offsetHeight;
+ scrollHeight = Math.max(scrollHeight, bottom);
+ }
+
+ var isVerticalScrollbar = scrollHeight > root.clientHeight;
+ //var isHorizontalScrollbar = root.scrollWidth > root.clientWidth;
+ return isVerticalScrollbar;
+ };
+
+ this._windowRect = function() {
+ var doc = document;
+
+ if (doc.compatMode === "CSS1Compat" && doc.documentElement && doc.documentElement.clientWidth) {
+ var x = doc.documentElement.clientWidth;
+ var y = doc.documentElement.clientHeight;
+ return { x: x, y: y };
+ }
+ else {
+ var x = doc.body.clientWidth;
+ var y = doc.body.clientHeight;
+ return { x: x, y: y };
+ }
+ };
+
+ this._register = function() {
+ if (this._registered) {
+ return;
+ }
+ this.re(window, 'resize', this._onWindowResize);
+ this.re(window, 'scroll', this._onWindowScroll);
+
+ if (this.dragDrop) {
+ this.re(document, 'mousemove', this._onMouseMove);
+ this.re(document, 'mouseup', this._onMouseUp);
+ }
+ this._registered = true;
+ };
+
+ this._unregister = function() {
+ this.ue(window, 'resize', this._onWindowResize);
+ this.ue(window, 'scroll', this._onWindowScroll);
+ if (this.dragDrop) {
+ this.ue(document, 'mousemove', this._onMouseMove);
+ this.ue(document, 'mouseup', this._onMouseUp);
+ }
+ this._registered = false;
+ };
+
+ this._onDragStart = function(e) {
+ if (e.target !== This.div) {
+ return;
+ }
+ e.preventDefault();
+ This.div.style.cursor = "move";
+ This._maskIframe();
+ This._coords = This.mc(e || window.event);
+ This._start = { x: This.div.offsetLeft, y: This.div.offsetTop };
+
+ };
+
+ this._onMouseMove = function(e) {
+ if (!This._coords) {
+ return;
+ }
+
+ var e = e || window.event;
+ var now = This.mc(e);
+
+ var x = now.x - This._coords.x;
+ var y = now.y - This._coords.y;
+
+ //This.iframe.style.display = 'none';
+ This.div.style.marginLeft = '0px';
+ This.div.style.top = (This._start.y + y) + "px";
+ This.div.style.left = (This._start.x + x) + "px";
+
+ };
+
+ this._onMouseUp = function(e) {
+ // no drag&drop
+ if (!This._coords) {
+ return;
+ }
+ //This.iframe.style.display = '';
+ This._unmaskIframe();
+ This.div.style.cursor = null;
+
+ This._coords = null;
+ };
+
+ this._maskIframe = function() {
+ if (!this.useIframe) {
+ return;
+ }
+
+ var opacity = 80;
+
+ var mask = document.createElement("div");
+ mask.style.backgroundColor = "#ffffff";
+ mask.style.filter = "alpha(opacity=" + opacity + ")";
+ mask.style.opacity = "0." + opacity;
+ mask.style.width = "100%";
+ mask.style.height = this.height + "px";
+ mask.style.position = "absolute";
+ mask.style.left = '0px';
+ mask.style.top = '0px';
+
+ this.div.appendChild(mask);
+ this.mask = mask;
+ };
+
+ this._unmaskIframe = function() {
+ if (!this.useIframe) {
+ return;
+ }
+
+ this.div.removeChild(this.mask);
+ this.mask = null;
+ };
+
+ this._onWindowResize = function() {
+ This._updateTop();
+ This._updateHorizontal();
+ };
+
+ this._onWindowScroll = function() {
+ This._updateTop();
+ };
+
+ this._updateHorizontal = function() {
+ if (This.left) {
+ return;
+ }
+
+ if (!This.div) {
+ return;
+ }
+
+ var width = This.div.offsetWidth;
+ This.div.style.marginLeft = '-' + Math.floor(width / 2) + "px"; // '-45%'
+ };
+
+ this._updateTop = function() {
+ if (!This.hideDiv) {
+ return;
+ }
+ if (!This.div) {
+ return;
+ }
+ if (This.hideDiv.style.display === 'none') {
+ return;
+ }
+ if (This.div.style.display === 'none') {
+ return;
+ }
+
+ var scrollY = This._parent.scrollY();
+
+
+ //This.hideDiv.style.height = height() + "px";
+ if (!This.scrollWithPage) {
+ This.div.style.top = (scrollY + This.top) + 'px';
+ }
+ };
+
+ this._parent = {};
+ this._parent.container = function() {
+ return This.container || document.body;
+ };
+ this._parent.scrollY = function() {
+ var c = This._parent.container();
+ if (c === document.body) {
+ return window.pageYOffset ? window.pageYOffset : ((document.documentElement && document.documentElement.scrollTop) ? document.documentElement.scrollTop : document.body.scrollTop);
+ }
+ else {
+ return c.scrollTop;
+ }
+ };
+
+ // already available in common.js but this file should be standalone
+ this.re = function(el, ev, func) {
+ if (el.addEventListener) {
+ el.addEventListener(ev, func, false);
+ } else if (el.attachEvent) {
+ el.attachEvent("on" + ev, func);
+ }
+ };
+
+ // unregister event
+ this.ue = function(el, ev, func) {
+ if (el.removeEventListener) {
+ el.removeEventListener(ev, func, false);
+ } else if (el.detachEvent) {
+ el.detachEvent("on" + ev, func);
+ }
+ };
+
+ // mouse coords
+ this.mc = function(ev) {
+ if (ev.pageX || ev.pageY) {
+ return { x: ev.pageX, y: ev.pageY };
+ }
+ return {
+ x: ev.clientX + document.documentElement.scrollLeft,
+ y: ev.clientY + document.documentElement.scrollTop
+ };
+ };
+
+ // absolute element position on page
+ this.abs = function(element) {
+ var r = {
+ x: element.offsetLeft,
+ y: element.offsetTop
+ };
+
+ while (element.offsetParent) {
+ element = element.offsetParent;
+ r.x += element.offsetLeft;
+ r.y += element.offsetTop;
+ }
+
+ return r;
+ };
+
+ this._create = function() {
+
+ var container = This._parent.container();
+ var isRoot = container === document.body;
+ var position = isRoot ? "fixed" : "absolute";
+
+ var hide = document.createElement("div");
+ hide.id = this.id + "hide";
+ hide.style.position = position;
+ hide.style.left = "0px";
+ hide.style.top = "0px";
+ hide.style.right = "0px";
+ hide.style.bottom = "0px";
+ hide.oncontextmenu = function() { return false; };
+ hide.onmousedown = function() { return false; }; // prevent selecting
+
+ container.appendChild(hide);
+
+ var div = document.createElement("div");
+ div.id = this.id + 'popup';
+ div.style.position = position;
+ div.style.left = '50%';
+ div.style.top = '0px';
+ div.style.backgroundColor = 'white';
+ div.style.width = "50px";
+ div.style.height = "50px";
+ if (this.dragDrop) {
+ div.onmousedown = this._onDragStart;
+ }
+ div.addEventListener("keydown", function(e) {
+ // prevent interaaction with the document using keyboard
+ e.stopPropagation();
+ });
+
+ var defaultHeight = 50;
+
+ var iframe = null;
+ if (this.useIframe) {
+ iframe = document.createElement("iframe");
+ iframe.id = this.id + "iframe";
+ iframe.name = this.id + "iframe";
+ iframe.frameBorder = '0';
+ iframe.style.width = '100%';
+ iframe.style.height = defaultHeight + 'px';
+ div.appendChild(iframe);
+ }
+
+ container.appendChild(div);
+
+ this.div = div;
+ this.iframe = iframe;
+ this.hideDiv = hide;
+ };
+
+ this.setInnerHTML = function(id, innerHTML) {
+ var frame = window.frames[id];
+
+ var doc = frame.contentWindow || frame.document || frame.contentDocument;
+ if (doc.document) {
+ doc = doc.document;
+ }
+
+ if (doc.body == null) { // null in IE
+ doc.write("");
+ }
+
+ if (innerHTML.nodeType) {
+ doc.body.appendChild(innerHTML);
+ }
+ else {
+ doc.body.innerHTML = innerHTML;
+ }
+
+ if (This.autoStretch) {
+ if (!This.autoStretchFirstLoadOnly || !This._stretched) {
+ This.stretch();
+ This._stretched = true;
+ }
+ }
+ };
+
+ this.close = function(result) {
+ this.result = result;
+ this.hide();
+ };
+
+ this.closeSerialized = function() {
+ var ref = This._body();
+ var fields = ref.querySelectorAll("input, textarea, select");
+ var result = {};
+ for (var i = 0; i < fields.length; i++) {
+ var field = fields[i];
+ var name = field.name;
+ if (!name) {
+ continue;
+ }
+ var value = field.value;
+ /*if (field.picker) {
+ value = field.picker.date ? field.picker.date.toString() : null;
+ }
+ else*/ /*if (field.table) {
+ value = field.table.save();
+ }
+ else*/ /*if (field.searchable) {
+ value = field.searchable.selected && field.searchable.selected.id;
+ }
+ else *//*if (field.tagName === "SELECT") {
+ var option = field.options[field.selectedIndex];
+ if (option && typeof option._originalValue !== "undefined") {
+ value = option._originalValue;
+ }
+ }
+ else*/ /*if (field.type === "radio") {
+ if (!field.checked) {
+ continue;
+ }
+ value = field._originalValue;
+ }
+ else*/ /*if (field.type === "checkbox") {
+ value = field.checked;
+ }*/
+ result[name] = value;
+ }
+ This.close(result);
+ };
+
+ this.hide = function(options) {
+
+ options = options || {};
+
+ var args = {};
+ args.backgroundClick = !!options.backgroundClick;
+ args.result = this.result;
+ args.canceled = typeof this.result === "undefined";
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof this.onClose === "function") {
+ this.onClose(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ if (this.div) {
+ this.div.style.display = 'none';
+ this.hideDiv.style.display = 'none';
+ if (!this.useIframe) {
+ this.div.innerHTML = null;
+ }
+ }
+
+ // return focus to the main window (Firefox)
+ window.focus();
+
+ //DayPilot.ModalStatic = null;
+ DayPilot.ModalStatic.remove(this);
+
+ if (typeof this.onClosed === "function") {
+ this.onClosed(args);
+ }
+ else if (this.closed) {
+ this.closed();
+ }
+
+ delete this.result;
+
+ if (this.disposeOnClose) {
+
+ This._unregister();
+
+ This._de(This.div);
+ This._de(This.hideDiv);
+
+ This.div = null;
+ This.hideDiv = null;
+ This.iframe = null;
+ }
+ };
+
+ this._de = function(e) {
+ if (!e) {
+ return;
+ }
+ e.parentNode && e.parentNode.removeChild(e);
+ };
+
+ this._applyOptions = function() {
+ if (!options) {
+ return;
+ }
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+ };
+
+ this._applyOptions();
+
+ };
+
+ DayPilot.Modal.alert = function(message, options) {
+ options = options || {};
+ options.height = options.height || 40;
+ options.useIframe = false;
+
+ var okText = options.okText || "OK";
+ var cancelText = options.cancelText || "Cancel";
+
+ return DayPilot.getPromise(function(success, failure) {
+ options.onClosed = function(args) {
+ success(args);
+ /*
+ if (typeof args.result === "undefined") {
+ failure(args);
+ }
+ else {
+
+ }*/
+ };
+
+ var modal = new DayPilot.Modal(options);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+ // div.style.padding = "10px";
+
+ var text = document.createElement("div");
+ text.className = modal.theme + "_content";
+ text.innerHTML = message;
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+ // buttons.style.margin = "10px 0px";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+ buttonOK.onclick = function(e) {
+ DayPilot.ModalStatic.close("OK");
+ };
+
+ buttons.appendChild(buttonOK);
+
+ div.appendChild(text);
+ div.appendChild(buttons);
+
+ //var buttons = "";
+
+ modal.showHtml(div);
+
+ if (modal.autoFocus) {
+ buttonOK.focus();
+ }
+ });
+
+ };
+
+ DayPilot.Modal.confirm = function(message, options) {
+ options = options || {};
+ options.height = options.height || 40;
+ options.useIframe = false;
+
+ var okText = options.okText || "OK";
+ var cancelText = options.cancelText || "Cancel";
+
+ return DayPilot.getPromise(function(success, failure) {
+ options.onClosed = function(args) {
+ success(args);
+ };
+
+ var modal = new DayPilot.Modal(options);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+ // div.style.padding = "10px";
+
+ var text = document.createElement("div");
+ text.className = modal.theme + "_content";
+ text.innerHTML = message;
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+ // buttons.style.margin = "10px 0px";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+ buttonOK.onclick = function(e) {
+ DayPilot.ModalStatic.close("OK");
+ };
+
+ var space = document.createTextNode(" ");
+
+ var buttonCancel = document.createElement("button");
+ buttonCancel.innerText = cancelText;
+ buttonCancel.className = modal.theme + "_cancel";
+ buttonCancel.onclick = function(e) {
+ DayPilot.ModalStatic.close();
+ };
+
+ buttons.appendChild(buttonOK);
+ buttons.appendChild(space);
+ buttons.appendChild(buttonCancel);
+
+ div.appendChild(text);
+ div.appendChild(buttons);
+
+ modal.showHtml(div);
+
+ if (modal.autoFocus) {
+ buttonOK.focus();
+ }
+
+ });
+ };
+
+
+ DayPilot.Modal.prompt = function(message, defaultValue, options) {
+ if (typeof defaultValue === "object") {
+ options = defaultValue;
+ defaultValue = "";
+ }
+
+ options = options || {};
+ options.height = options.height || 40;
+ options.useIframe = false;
+
+ var okText = options.okText || "OK";
+ var cancelText = options.cancelText || "Cancel";
+
+ var inputText = defaultValue || "";
+
+ return DayPilot.getPromise(function(success, failure) {
+ options.onClosed = function(args) {
+ success(args);
+ };
+
+ var modal = new DayPilot.Modal(options);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+
+ var text = document.createElement("div");
+ text.className = modal.theme + "_content";
+ text.innerHTML = message;
+
+ var inputs = document.createElement("div");
+ inputs.className = modal.theme + "_input";
+ //inputs.style.margin = "10px 0px";
+
+ var input = document.createElement("input");
+ input.value = inputText;
+ input.style.width = "100%";
+ input.onkeydown = function(e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13:
+ modal.close(this.value);
+ break;
+ case 27:
+ modal.close();
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ inputs.appendChild(input);
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+ //buttons.style.margin = "10px 0px";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+ buttonOK.onclick = function(e) {
+ modal.close(input.value);
+ };
+
+ var space = document.createTextNode(" ");
+
+ var buttonCancel = document.createElement("button");
+ buttonCancel.innerText = cancelText;
+ buttonCancel.className = modal.theme + "_cancel";
+ buttonCancel.onclick = function(e) {
+ modal.close();
+ };
+
+ buttons.appendChild(buttonOK);
+ buttons.appendChild(space);
+ buttons.appendChild(buttonCancel);
+
+
+ div.appendChild(text);
+ div.appendChild(inputs);
+ div.appendChild(buttons);
+
+ modal.showHtml(div);
+
+ if (modal.autoFocus) {
+ input.focus();
+ }
+
+ });
+ };
+
+ var isArray = function(arg) {
+ return Object.prototype.toString.call(arg) === '[object Array]';
+ };
+
+ function setPathValue(target, path, value) {
+ var iodot = path.indexOf(".");
+ if (iodot === -1) {
+ if (path !== "__proto__" && path !== "constructor") {
+ target[path] = value;
+ }
+ return;
+ }
+ var segment = path.substring(0, iodot);
+ if (segment === "__proto__" || segment === "constructor") {
+ return;
+ }
+ var remainder = path.substring(iodot + 1);
+ var child = target[segment];
+ if (typeof child !== "object" || child === null) {
+ target[segment] = {};
+ child = target[segment];
+ }
+ setPathValue(child, remainder, value);
+ }
+
+ // form = array of items
+ // data = object with properties
+ // options = standard modal properties
+ // DayPilot.Modal.form([]); // form only - new data
+ // DayPilot.Modal.form({}); // data only - automatic form
+ //
+ DayPilot.Modal.form = function(form, data, options) {
+ if (arguments.length === 1) {
+ var arg = form;
+ var isa = isArray(arg);
+
+ if (isa) {
+ data = {};
+ }
+ else if (typeof arg === "object") {
+ data = form;
+ form = [];
+ for (var name in data) {
+ var item = {};
+ item.name = name;
+ item.id = name;
+ form.push(item);
+ }
+ }
+ else {
+ throw "Invalid DayPilot.Modal.form() parameter";
+ }
+
+ }
+
+ // make a copy
+ var opts = {};
+ for (var name in options) {
+ opts[name] = options[name];
+ }
+
+ // options = options || {};
+ opts.height = opts.height || 40;
+ opts.useIframe = false;
+
+ /* if (typeof opts.autoFocus === "undefined") {
+ opts.autoFocus = false;
+ }*/
+
+ var okText = opts.okText || "OK";
+ var cancelText = opts.cancelText || "Cancel";
+
+ // var message = opts.message || "";
+
+ return DayPilot.getPromise(function(success, failure) {
+ opts.onClosed = function(args) {
+
+ if (args.result) {
+ // deep copy
+ var mergedResult = JSON.parse(JSON.stringify(data));
+
+ // unflatten
+ for (var name in args.result) {
+ setPathValue(mergedResult, name, args.result[name]);
+ }
+ args.result = mergedResult;
+ }
+
+ success(args);
+ };
+
+ var modal = new DayPilot.Modal(opts);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+
+ var inputs = document.createElement("div");
+ inputs.className = modal.theme + "_input";
+
+ var f = new Form({
+ theme: modal.theme,
+ form: form,
+ data: data,
+ zIndex: modal.zIndex,
+ locale: modal.locale,
+ plugins: modal.plugins,
+ onKey: function(args) {
+ switch (args.key) {
+ case "Enter":
+ // modal.closeSerialized();
+ // modal.close(f.serialize());
+ if (f.validate()) {
+ modal.close(f.serialize());
+ }
+ break;
+ case "Escape":
+ modal.close();
+ break;
+ }
+ },
+ onChange: function(args) {
+ if (typeof modal.onChange === "function") {
+ modal.onChange(args);
+ }
+ }
+ });
+ var el = f.create();
+
+ inputs.append(el);
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+
+ if (opts.okDisabled) {
+ buttonOK.disabled = true;
+ }
+ buttonOK.onclick = function(e) {
+ if (f.validate()) {
+ modal.close(f.serialize());
+ }
+ // modal.closeSerialized();
+ };
+
+ var space = document.createTextNode(" ");
+
+ var buttonCancel = document.createElement("button");
+ buttonCancel.innerText = cancelText;
+ buttonCancel.className = modal.theme + "_cancel";
+ buttonCancel.onclick = function(e) {
+ modal.close();
+ };
+ buttonCancel.onmousedown = function(e) {
+ f.canceling = true;
+ };
+
+ buttons.appendChild(buttonOK);
+ buttons.appendChild(space);
+ buttons.appendChild(buttonCancel);
+
+
+ // div.appendChild(text);
+ div.appendChild(inputs);
+ div.appendChild(buttons);
+
+ modal.showHtml(div);
+
+ modal.div.setAttribute("tabindex", "-1");
+ modal.div.addEventListener("keydown", function(e) {
+ switch (e.keyCode) {
+ case 27:
+ modal.close();
+ break;
+ case 13:
+ if (f.validate()) {
+ modal.close(f.serialize());
+ }
+ // modal.closeSerialized();
+ break;
+ }
+ });
+
+ if (modal.focus) {
+ var toBeFocused = null;
+ if (typeof modal.focus === "object") {
+ var id = modal.focus.id;
+ var value = modal.focus.value;
+ toBeFocused = f.findViewById(id, value);
+ }
+ else if (typeof modal.focus === "string") {
+ toBeFocused = f.findViewById(modal.focus)
+ }
+ if (toBeFocused) {
+ toBeFocused.focus();
+ }
+ }
+ else {
+ var first = f.firstFocusable();
+ if (modal.autoFocus && first) {
+ first.focus();
+ }
+ else {
+ modal.div.focus();
+ }
+ }
+
+ });
+ };
+
+ DayPilot.Modal.close = function(result) {
+ var opener = DayPilot.Modal.opener();
+ if (!opener) {
+ return;
+ }
+ opener.close(result);
+ };
+
+ DayPilot.Modal.stretch = function(result) {
+ var opener = DayPilot.Modal.opener();
+ if (!opener) {
+ throw "Unable to find the opener DayPilot.Modal instance.";
+ }
+ opener.stretch();
+ };
+
+ DayPilot.Modal.closeSerialized = function() {
+ var last = DayPilot.Modal.opener() || DayPilot.ModalStatic.last();
+ if (last) {
+ last.closeSerialized();
+ }
+ };
+
+ DayPilot.Modal.opener = function() {
+ if (typeof DayPilot !== "undefined" && typeof DayPilot.ModalStatic !== "undefined" && DayPilot.ModalStatic.list.length > 0) {
+ return DayPilot.ModalStatic.list[DayPilot.ModalStatic.list.length - 1];
+ }
+ return parent && parent.DayPilot && parent.DayPilot.ModalStatic && parent.DayPilot.ModalStatic.list[parent.DayPilot.ModalStatic.list.length - 1];
+ };
+
+ DayPilot.Modal.Experimental = {};
+ DayPilot.Modal.Experimental.Form = Form;
+
+ if (typeof DayPilot.getPromise === "undefined") {
+ DayPilot.getPromise = function(f) {
+ if (typeof Promise !== 'undefined') {
+ return new Promise(f);
+ }
+
+ DayPilot.Promise = function(f) {
+ var p = this;
+
+ this.then = function(onFulfilled, onRejected) {
+ onFulfilled = onFulfilled || function() {};
+ onRejected = onRejected || function() {};
+ f(onFulfilled, onRejected);
+ return DayPilot.getPromise(f);
+ };
+
+ this['catch'] = function(onRejected) {
+ p.then(null, onRejected);
+ return DayPilot.getPromise(f);
+ };
+ };
+
+ return new DayPilot.Promise(f);
+
+ };
+ }
+
+ // end of modal.js
+
+ // form.js -> 2877
+
+ var Form = function (options) {
+
+ // properties
+ this.form = [];
+ this.data = {};
+ this.theme = "form_default";
+ this.zIndex = 99999;
+ this.locale = "en-us";
+ this.plugins = {};
+
+ // events
+ this.onKey = null;
+
+ // state
+ this._rows = [];
+ this._newRows = null;
+ this.canceling = false;
+
+ this._validationTimeouts = [];
+
+ // view
+ this._views = [];
+ this._div = null;
+
+ options = options || {};
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+ };
+
+ Form.prototype.create = function () {
+ this.load();
+ this.render();
+
+ return this._div;
+ };
+
+ Form.prototype.render = function () {
+ var form = this;
+ this._div = document.createElement("div");
+ this._rows.forEach(function (row) {
+ form.createView(row);
+ });
+ this.applyState();
+ };
+
+ Form.prototype.createView = function (row) {
+ var theme = this.theme;
+ var form = this;
+
+ var div = document.createElement("div");
+ div.className = theme + "_form_item " + theme + "_form_item_level" + row.level;
+ if (!row.interactive && row.type === "title") {
+ /*
+ if (row.type === "title") {
+ div.className += " " + theme + "_form_title";
+ }
+ */
+ div.className += " " + theme + "_form_title";
+ }
+ else {
+ div.className += " " + theme + "_form_item_" + row.type;
+ }
+ if (row.data.cssClass) {
+ div.className += " " + row.data.cssClass;
+ }
+
+ if (!row.isValue) {
+ var label = document.createElement("div");
+ label.className = theme + "_form_item_label";
+ label.innerText = row.text;
+ div.appendChild(label);
+ }
+
+ var interactive = this.createInteractive(row);
+ interactive.onInput = function(options) {
+ options = options || {};
+ form._validateInteractive(interactive, {
+ "debounce": !options.immediate
+ });
+ if (typeof form.onChange === "function") {
+ var args = {};
+ args.result = form.serialize();
+ form.onChange(args);
+ }
+ };
+ interactive.onBlur = function() {
+ if (!form.canceling) {
+ form._validateInteractive(interactive);
+ }
+ };
+ interactive.apply(row);
+ interactive._div = div;
+ interactive.row = row;
+
+ if (interactive.element) {
+ div.appendChild(interactive.element);
+ }
+
+ this._views.push(interactive);
+
+ this._div.appendChild(div);
+
+ };
+
+ Form.prototype.validate = function() {
+ var form = this;
+ var valid = true;
+ this._views.forEach(function(interactive) {
+ var iv = form._validateInteractive(interactive);
+ valid = valid && iv;
+ });
+ return valid;
+ };
+
+ Form.prototype._validateInteractive = function(interactive, options) {
+ options = options || {};
+ var debounce = options.debounce;
+ var silent = options.silent;
+
+
+ var row = interactive.row;
+ var valid = true;
+
+ var onValidate = typeof row.data.onValidate === "function" ? row.data.onValidate : null;
+ var validate = typeof row.data.validate === "function" ? row.data.validate : null; // legacy
+
+ var validateHandler = onValidate || validate;
+
+ if (validateHandler) {
+
+ var args = {};
+ args.valid = true;
+ args.value = interactive.save()[row.field];
+ args.message = "Error";
+ args.values = this.serialize(); // legacy
+ args.result = this.serialize();
+
+ validateHandler(args);
+
+ var cssClassInvalid = this.theme + "_form_item_invalid";
+ var cssClassMessage = this.theme + "_form_item_invalid_message";
+ if (args.valid) {
+ clearTimeout(this._validationTimeouts[row.field]);
+
+ if (interactive._errorMsg) {
+ interactive._errorMsg.remove();
+ interactive._errorMsg = null;
+ }
+ interactive._div.classList.remove(cssClassInvalid);
+ }
+ else {
+
+ function showInvalid() {
+ if (interactive._errorMsg) {
+ interactive._errorMsg.remove();
+ interactive._errorMsg = null;
+ }
+
+ interactive._div.classList.add(cssClassInvalid);
+ var msg = document.createElement("div");
+ msg.classList.add(cssClassMessage);
+ msg.innerText = args.message;
+
+ interactive._errorMsg = msg;
+
+ interactive._div.appendChild(msg);
+ }
+
+ if (!silent) {
+ if (debounce) {
+
+ var debounceDelay = 1000;
+
+ clearTimeout(this._validationTimeouts[row.field]);
+
+ this._validationTimeouts[row.field] = setTimeout(function() {
+ showInvalid();
+ }, debounceDelay);
+ }
+ else {
+ showInvalid();
+ }
+
+ }
+
+ }
+ valid = args.valid;
+ }
+ return valid;
+ };
+
+ Form.prototype.load = function () {
+ // transform this.form + this.data into state (_rows)
+ var t = this;
+ this.form.forEach(function (item) {
+ t.processFormItem(item, 0);
+ });
+
+ var flat;
+ // sanity check (especially for circular references)
+ try {
+ var stringified = JSON.stringify(this.data);
+ var rebuilt = JSON.parse(stringified);
+ flat = flatten(rebuilt);
+ }
+ catch (e) {
+ throw new Error("The 'data' object is not serializable (it may contain circular dependencies): " + e);
+ }
+
+ // set values
+ for (var name in flat) {
+ this.setValue(name, flat[name]);
+ }
+
+ // set state depending on values
+ // this.updateDependentState();
+ };
+
+ Form.prototype.setValue = function (name, value) {
+ this._rows.forEach(function (row) {
+ row.applyValue(name, value);
+ });
+ };
+
+ Form.prototype.updateDependentState = function () {
+ var form = this;
+ var enabled = [true];
+
+ var source = this._newRows ? this._newRows : this._rows;
+
+ source.forEach(function (row) {
+
+ var updatedRow = form.updateState(row, {
+ enabled: enabled[row.level] && !row.data.disabled
+ });
+
+ if (updatedRow.isValue) {
+ enabled[updatedRow.level + 1] = updatedRow.enabled && updatedRow.checked;
+ }
+ });
+ };
+
+ Form.prototype.processFormItem = function (item, level) {
+ var form = this;
+ var type = this.getFieldType(item);
+
+ var rows = [];
+
+ if (type === "radio") {
+
+ if (item.name) {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = "label";
+ row.interactive = false;
+ row.text = item.name;
+ form._rows.push(row);
+ rows.push(row);
+ }
+
+ item.options.forEach(function (option) {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = option;
+ row.level = level;
+ row.type = type;
+ row.isValue = true;
+ row.text = option.name;
+ row.resolved = option.id;
+ form._rows.push(row);
+ rows.push(row);
+
+ if (option.children) {
+ option.children.forEach(function (child) {
+ var childRows = form.processFormItem(child, level + 1);
+ rows = rows.concat(childRows);
+ })
+ }
+ });
+ }
+ else if (type === "title") {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = item.name;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else if (type === "image") {
+ var row = new RowModel();
+ row.isValue = true;
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = null;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else if (type === "html") {
+ var row = new RowModel();
+ row.isValue = true;
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = null;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else if (type === "scrollable") {
+ var row = new RowModel();
+ row.isValue = true;
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = null;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.text = item.name;
+ row.children = [];
+ form._rows.push(row);
+ rows.push(row);
+ }
+
+ if (type === "checkbox") {
+ row.isValue = true;
+ row.resolved = true;
+
+ if (item.children) {
+ item.children.forEach(function (child) {
+ var childRows = form.processFormItem(child, level + 1);
+ rows = rows.concat(childRows);
+ })
+ }
+ }
+
+ return rows;
+
+ };
+
+ Form.prototype.doOnKey = function (key) {
+ if (typeof this.onKey === "function") {
+ var args = {
+ key: key
+ };
+ this.onKey(args);
+ }
+ };
+
+ Form.prototype.createInteractive = function (row) {
+ var form = this;
+ var views = {
+ "label": function () {
+ return new Interactive();
+ },
+ "title": function () {
+ return new Interactive();
+ },
+ "image": function() {
+ var interactive = new Interactive();
+
+ var image = document.createElement("img");
+ image.src = row.data.image;
+
+ interactive.element = image;
+
+ return interactive;
+ },
+ "html": function() {
+ var interactive = new Interactive();
+
+ var div = document.createElement("div");
+ if (typeof row.data.text === "string") {
+ div.innerText = row.data.text;
+ }
+ else if (typeof row.data.html === "string") {
+ div.innerHTML = row.data.html;
+ }
+
+ interactive.element = div;
+
+ return interactive;
+ },
+ "scrollable": function() {
+ var interactive = new Interactive();
+
+ var scroll = document.createElement("div");
+ scroll.className = form.theme + "_form_item_scrollable_scroll";
+
+ if (row.data.height) {
+ scroll.style.height = row.data.height + "px";
+ }
+
+ var div = document.createElement("div");
+ div.className = form.theme + "_form_item_scrollable_scroll_content";
+ if (typeof row.data.text === "string") {
+ div.innerText = row.data.text;
+ }
+ else if (typeof row.data.html === "string") {
+ div.innerHTML = row.data.html;
+ }
+
+ scroll.appendChild(div);
+
+ interactive.element = scroll;
+
+ return interactive;
+ },
+ "text": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+
+ var input = interactive.element;
+ input.value = row.value;
+ input.disabled = !row.enabled;
+ };
+
+ var input = document.createElement("input");
+ input.name = row.field;
+ input.type = "text";
+ input.autocomplete = "off";
+ input.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13:
+ form.doOnKey("Enter");
+ break;
+ case 27:
+ form.doOnKey("Escape");
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ input.oninput = function(e) {
+ interactive.onInput();
+ };
+ input.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.element = input;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ interactive.element.setSelectionRange(0, interactive.element.value.length);
+ };
+ interactive.save = function() {
+ var result = {};
+ result[row.field] = input.value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "textarea": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+
+ var input = interactive.element;
+ input.value = row.value;
+ input.disabled = !row.enabled;
+ };
+
+ var textarea = document.createElement("textarea");
+ textarea.name = row.field;
+ if (row.data.height) {
+ textarea.style.height = row.data.height + "px";
+ }
+ // input.type = "text";
+ textarea.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13:
+ // form.doOnKey("Enter");
+ if (e.ctrlKey || e.metaKey) {
+ form.doOnKey("Enter");
+ }
+ letcontinue = false;
+ break;
+ case 27:
+ form.doOnKey("Escape");
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ // e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ textarea.oninput = function(e) {
+ interactive.onInput();
+ };
+ textarea.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.element = textarea;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ interactive.element.setSelectionRange(0, 0);
+ };
+ interactive.save = function() {
+ var result = {};
+ result[row.field] = textarea.value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "date": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var input = interactive.element;
+ var picker = interactive.picker;
+
+ if (row.data.dateFormat) {
+ picker.pattern = row.data.dateFormat;
+ }
+ var locale = row.data.locale || form.locale;
+ if (locale) {
+ picker.locale = locale;
+ }
+
+ input.disabled = !row.enabled;
+
+ picker.date = new DayPilot.Date(row.value);
+ var formatted = new DayPilot.Date(row.value).toString(row.data.dateFormat || picker.pattern, picker.locale);
+ input.value = formatted;
+ };
+
+ var input = document.createElement("input");
+ input.name = row.field;
+
+ var picker = new DayPilot.DatePicker({
+ target: input,
+ theme: "navigator_modal",
+ zIndex: form.zIndex + 1,
+ resetTarget: false,
+ targetAlignment: "left",
+ onTimeRangeSelect: function(args) {
+ interactive.onInput({"immediate": true});
+ }
+ });
+
+ // required for serialization - to get the normalized value
+ input.picker = picker;
+ input.className = form.theme + "_input_date";
+ input.type = "text";
+ input.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13: // enter
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Enter");
+ }
+ break;
+ case 27: // escape
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Escape");
+ }
+ break;
+ case 9: // tab
+ picker.close();
+ letcontinue = true;
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ input.onfocus = function () {
+ picker.show();
+ };
+
+ input.onclick = function () {
+ picker.show();
+ };
+
+ /*
+ input.onblur = function () {
+ // input.picker.close();
+ };
+ */
+
+ input.oninput = function(e) {
+ interactive.onInput();
+ };
+ input.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.element = input;
+ interactive.picker = picker;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ };
+ interactive.save = function() {
+ var value = picker.date ? picker.date.toString() : null;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+ return interactive;
+ },
+ "time": function () {
+ return form._createInteractiveTime(row);
+ },
+ "datetime": function() {
+ return form._createInteractiveDateTime(row);
+ },
+ "select": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var select = interactive.element;
+
+ select.value = row.value;
+ select.disabled = !row.enabled;
+ };
+
+ var select = document.createElement("select");
+ select.name = row.field;
+
+ if (row.data.options && row.data.options.forEach) {
+ row.data.options.forEach(function (i) {
+ var option = document.createElement("option");
+ option.innerText = i.name || i.id;
+ option.value = i.id;
+ option._originalValue = i.id;
+ select.appendChild(option);
+ });
+ }
+
+ select.onchange = function(e) {
+ interactive.onInput({"immediate": true});
+ };
+ select.onblur = function(e) {
+ interactive.onBlur();
+ };
+
+ interactive.element = select;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ };
+ interactive.save = function() {
+ var value = null;
+ var option = select.options[select.selectedIndex];
+ if (option && typeof option._originalValue !== "undefined") {
+ value = option._originalValue;
+ }
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+
+ },
+ "searchable": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var searchable = interactive.searchable;
+
+ searchable.disabled = !row.enabled;
+ searchable.select(row.value);
+ };
+
+ var searchable = new Searchable({
+ data: row.data.options || [],
+ name: row.field,
+ theme: form.theme + "_form_item_searchable",
+ listZIndex: form.zIndex + 1,
+ // disabled: !row.enabled
+ onSelect: function(args) {
+ if (args.ui) {
+ interactive.onInput({"immediate": true});
+ }
+ }
+ });
+
+ var element = searchable.create();
+
+ interactive.element = element;
+ interactive.searchable = searchable;
+ interactive.canFocus = function() {
+ return !interactive.searchable.disabled;
+ };
+ interactive.focus = function () {
+ interactive.searchable.focus();
+ };
+ interactive.save = function() {
+ var value = searchable.selected && searchable.selected.id;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "radio": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var radio = interactive.radio;
+
+ radio.checked = row.checked;
+ radio.disabled = !row.enabled;
+ };
+
+ var label = document.createElement("label");
+
+ var radio = document.createElement("input");
+ radio.type = "radio";
+ radio.name = row.field;
+ radio._originalValue = row.resolved;
+
+ radio.onchange = function (ev) {
+ // activation only
+
+ var row = interactive.row;
+ form.findRowsByField(row.field).forEach(function (row) {
+ form.updateState(row, {
+ checked: false
+ });
+ });
+ form.updateState(row, {
+ checked: true
+ });
+
+ form.applyState();
+
+ interactive.onInput({"immediate": true});
+ };
+
+ radio.onblur = function(e) {
+ interactive.onBlur();
+ };
+
+ label.appendChild(radio);
+
+ var text = document.createTextNode(row.text);
+ label.append(text);
+
+ interactive.element = label;
+ interactive.radio = radio;
+ interactive.canFocus = function() {
+ return false;
+ };
+ interactive.focus = function () {
+ interactive.radio.focus();
+ }
+ interactive.save = function() {
+ if (!radio.checked) {
+ return {};
+ }
+ var value = radio._originalValue;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "checkbox": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var checkbox = interactive.checkbox;
+
+ checkbox.checked = row.checked;
+ checkbox.disabled = !row.enabled;
+ };
+
+ var label = document.createElement("label");
+
+ var checkbox = document.createElement("input");
+ checkbox.type = "checkbox";
+ checkbox.name = row.field;
+ checkbox._originalValue = row.resolved;
+
+ checkbox.onchange = function (ev) {
+ var row = interactive.row;
+ form.updateState(row, {
+ checked: this.checked
+ });
+ form.applyState();
+
+ interactive.onInput({"immediate": true});
+ };
+
+ checkbox.onblur = function(e) {
+ interactive.onBlur();
+ };
+
+ label.appendChild(checkbox);
+
+ var text = document.createTextNode(row.text);
+ label.append(text);
+
+ interactive.element = label;
+ interactive.checkbox = checkbox;
+ interactive.canFocus = function() {
+ return false;
+ };
+ interactive.focus = function () {
+ interactive.checkbox.focus();
+ }
+ interactive.save = function() {
+ var value = checkbox.checked;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "table": function() {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var table = interactive.table;
+
+ table.disabled = !row.enabled;
+ table.load(row.value || []);
+
+ /*
+ if (row.value) {
+ table.load(row.value);
+ }
+ */
+ };
+
+ var table = new Table({
+ name: row.field,
+ // form: row.data.columns,
+ form: form,
+ theme: form.theme + "_form_item_tabular",
+ item: row.data,
+ onInput: function(args) {
+ interactive.onInput();
+ }
+ });
+
+ var element = table.create();
+
+ interactive.element = element;
+ interactive.table = table;
+ interactive.canFocus = function() {
+ // return !interactive.table.disabled;
+ return false;
+ };
+ interactive.focus = function () {
+ interactive.table.focus();
+ };
+ interactive.save = function() {
+ var value = table.save();
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ }
+ };
+ if (form.plugins && form.plugins[row.type]) {
+ return form.plugins[row.type](row);
+ }
+ return views[row.type]();
+ };
+
+ Form.prototype._createInteractiveTime = function(row) {
+ var form = this;
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var searchable = interactive.searchable;
+
+ searchable.disabled = !row.enabled;
+ searchable.select(row.value);
+ };
+
+ var data = [];
+
+ var interval = row.data.timeInterval || 15; // allowed values: 1, 5, 10, 15, 20, 30, 60
+ var allowedIntervals = [1, 5, 10, 15, 20, 30, 60];
+ if (!allowedIntervals.includes(interval)) {
+ interval = 15;
+ }
+ var perHour = 60 / interval;
+
+ var localeStr = row.data.locale || form.locale;
+ var locale = DayPilot.Locale.find(localeStr) || DayPilot.Locale.US;
+
+ var date = DayPilot.Date.today();
+
+ for (var i = 0; i < 24*perHour; i++) {
+ var time = date.addMinutes(interval*i);
+ var item = {};
+ item.name = time.toString(row.data.timeFormat || locale.timePattern, locale);
+ item.id = time.toString("HH:mm");
+ data.push(item);
+ }
+
+ var searchable = new Searchable({
+ data: data,
+ name: row.field,
+ theme: form.theme + "_form_item_time",
+ listZIndex: form.zIndex + 1,
+ strategy: "startsWith",
+ // disabled: !row.enabled
+ onSelect: function(args) {
+ if (args.ui) {
+ interactive.onInput({"immediate": true});
+ }
+ }
+ });
+
+ var element = searchable.create();
+
+ interactive.element = element;
+ interactive.searchable = searchable;
+ interactive.canFocus = function() {
+ return !interactive.searchable.disabled;
+ };
+ interactive.focus = function () {
+ interactive.searchable.focus();
+ };
+ interactive.save = function() {
+ var value = searchable.selected && searchable.selected.id;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+
+ };
+
+ Form.prototype._createInteractiveDateTime = function(row) {
+ var form = this;
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var searchable = interactive.searchable;
+
+ searchable.disabled = !row.enabled;
+
+ var timePart = new DayPilot.Date(row.value).toString("HH:mm");
+
+ searchable.select(timePart);
+
+ var input = interactive.dateInput;
+ var picker = interactive.picker;
+
+ if (row.data.dateFormat) {
+ picker.pattern = row.data.dateFormat;
+ }
+ var localeStr = row.data.locale || form.locale;
+ if (localeStr) {
+ var locale = DayPilot.Locale.find(localeStr) || DayPilot.Locale.US;
+ picker.locale = localeStr;
+ picker.pattern = locale.datePattern;
+ }
+
+ input.disabled = !row.enabled;
+
+ picker.date = new DayPilot.Date(row.value);
+ var formatted = new DayPilot.Date(row.value).toString(row.data.dateFormat || picker.pattern, picker.locale);
+ input.value = formatted;
+
+ };
+
+ var dateElement = (function createDatePicker() {
+ var input = document.createElement("input");
+ input.name = row.field;
+
+ var picker = new DayPilot.DatePicker({
+ target: input,
+ theme: "navigator_modal",
+ zIndex: form.zIndex + 1,
+ resetTarget: false,
+ targetAlignment: "left",
+ onTimeRangeSelect: function(args) {
+ interactive.onInput({"immediate": true});
+ }
+ });
+
+ // required for serialization - to get the normalized value
+ input.picker = picker;
+ input.className = form.theme + "_input_date";
+ input.type = "text";
+ input.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13: // enter
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Enter");
+ }
+ break;
+ case 27: // escape
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Escape");
+ }
+ break;
+ case 9: // tab
+ picker.close();
+ letcontinue = true;
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ input.onfocus = function () {
+ picker.show();
+ };
+
+ input.onclick = function () {
+ picker.show();
+ };
+
+ input.oninput = function(e) {
+ interactive.onInput();
+ };
+ input.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.dateInput = input;
+ interactive.picker = picker;
+
+ return input;
+ })();
+
+
+ var timeElement = (function createTimePicker() {
+ var data = [];
+
+ var interval = row.data.timeInterval || 15; // allowed values: 1, 5, 10, 15, 20, 30, 60
+ var allowedIntervals = [1, 5, 10, 15, 20, 30, 60];
+ if (!allowedIntervals.includes(interval)) {
+ interval = 15;
+ }
+ var perHour = 60 / interval;
+
+ var localeStr = row.data.locale || form.locale;
+ var locale = DayPilot.Locale.find(localeStr) || DayPilot.Locale.US;
+
+ var date = DayPilot.Date.today();
+
+ for (var i = 0; i < 24*perHour; i++) {
+ var time = date.addMinutes(interval*i);
+ var item = {};
+ item.name = time.toString(row.data.timeFormat || locale.timePattern, locale);
+ item.id = time.toString("HH:mm");
+ data.push(item);
+ }
+
+ var searchable = new Searchable({
+ data: data,
+ name: row.field,
+ theme: form.theme + "_form_item_time",
+ listZIndex: form.zIndex + 1,
+ strategy: "startsWith",
+ // disabled: !row.enabled
+ onSelect: function(args) {
+ if (args.ui) {
+ interactive.onInput({"immediate": true});
+ }
+ }
+ });
+
+ interactive.searchable = searchable;
+ return searchable.create();
+
+ })();
+
+
+ var element = document.createElement("div");
+ element.className = form.theme + "_form_item_datetime_parent";
+ element.appendChild(dateElement);
+ element.appendChild(timeElement);
+
+ interactive.element = element;
+ interactive.canFocus = function() {
+ return !interactive.searchable.disabled;
+ };
+ interactive.focus = function () {
+ interactive.dateInput.focus();
+ };
+ interactive.save = function() {
+ var timeValue = interactive.searchable.selected && interactive.searchable.selected.id;
+ var dateValue = interactive.picker.date ? interactive.picker.date.toString() : null;
+
+ var date = new DayPilot.Date(dateValue).getDatePart();
+ var value = DayPilot.Date.parse(date.toString("yyyy-dd-MM ") + timeValue, "yyyy-dd-MM HH:mm");
+
+ var result = {};
+ result[row.field] = value;
+ return result;
+
+ };
+
+ return interactive;
+
+ };
+
+ Form.prototype.findRowsByField = function (field) {
+ return this._rows.filter(function (row) {
+ return row.field === field;
+ });
+ };
+
+ // ooptional value param applies to radio which can have multiple views, one for each value
+ Form.prototype.findViewById = function (id, value) {
+ return this._views.find(function (v) {
+ if (v.row.field === id) {
+ if (v.row.type === "radio") {
+ return v.row.resolved === value;
+ } else {
+ return true;
+ }
+ }
+ return false;
+ });
+ };
+
+ Form.prototype.firstFocusable = function () {
+ return this._views.find(function (v) {
+ return v.canFocus && v.canFocus();
+ });
+ };
+
+ Form.prototype.updateState = function (row, props) {
+ var source = this._newRows ? this._newRows : this._rows;
+ var index = source.indexOf(row);
+ this._newRows = source.map(function (srow) {
+ if (srow !== row) {
+ return srow;
+ }
+ // our row
+ if (row.propsEqual(props)) {
+ return row;
+ }
+ var cloned = row.clone();
+ for (var name in props) {
+ cloned[name] = props[name];
+ }
+ return cloned;
+ });
+ return this._newRows[index];
+ };
+
+ // dirty row received, a member of _newRows
+ Form.prototype.updateInteractive = function (row) {
+ var index = this._newRows.indexOf(row);
+ this._views[index].apply(row);
+ };
+
+ Form.prototype.applyState = function () {
+ var form = this;
+
+ this.updateDependentState();
+
+ if (!this._newRows) {
+ return;
+ }
+
+ var dirtyRows = this._newRows.filter(function (row, i) {
+ return form._rows[i] !== row;
+ });
+
+ dirtyRows.forEach(function (row) {
+ form.updateInteractive(row);
+ });
+
+ this._rows = this._newRows;
+
+ this._newRows = null;
+ };
+
+ Form.prototype.getFieldType = function (item) {
+
+ var known = ["text", "date", "select", "searchable", "radio", "checkbox", "table", "title", "image", "html", "textarea", "scrollable", "time", "datetime"];
+ if (known.indexOf(item.type) !== -1) {
+ return item.type;
+ }
+
+ if (item.type && this.plugins && this.plugins[item.type]) {
+ return item.type;
+ }
+
+ if (item.image) {
+ return "image";
+ }
+
+ if (item.html || item.text) {
+ return "html";
+ }
+
+ if (!item.id) {
+ return "title";
+ }
+
+ if (item.options) {
+ return "searchable";
+ }
+
+ if (item.dateFormat) {
+ return "date";
+ }
+
+ if (item.columns) {
+ return "table";
+ }
+
+ return "text";
+ };
+
+ Form.prototype.serialize = function() {
+ var result = {};
+
+ this._views.forEach(function(interactive) {
+ var out = interactive.save();
+ for (var name in out) {
+ result[name] = out[name];
+ }
+ });
+
+ return result;
+ };
+
+ var RowModel = function () {
+ this.id = this.guid();
+ this.field = null;
+ this.data = null;
+ this.type = null;
+ this.level = 0;
+ this.enabled = true;
+ this.value = null;
+ this.text = null;
+
+ this.interactive = true;
+
+ this.isValue = false;
+ this.checked = false;
+ this.resolved = null; // value resolves to
+ };
+
+ RowModel.prototype.clone = function () {
+ var rm = new RowModel();
+ for (var name in this) {
+ if (name === "id") {
+ continue;
+ }
+ rm[name] = this[name];
+ }
+ return rm;
+ };
+
+ RowModel.prototype.propsEqual = function (props) {
+ for (var name in props) {
+ if (this[name] !== props[name]) {
+ return false;
+ }
+ }
+ return true;
+ };
+
+ RowModel.prototype.guid = function () {
+ var S4 = function () {
+ return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
+ };
+ return ("" + S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4());
+ };
+
+ RowModel.prototype.applyValue = function (name, value) {
+ if (this.field !== name) {
+ return;
+ }
+ this.value = value;
+ if (this.isValue && value === this.resolved) {
+ this.checked = true;
+ }
+ };
+
+ var Interactive = function () {
+ // this.row = row;
+ this.element = null;
+
+ // only used for autofocus
+ this.canFocus = function() {
+ return false;
+ };
+ this.apply = function (row) {
+ };
+ this.focus = function () {
+ };
+ this.save = function() {
+ return {};
+ }
+ };
+
+ function flatten(object, result, prefix) {
+ result = result || {};
+ prefix = prefix || "";
+ for (var name in object) {
+ var src = object[name];
+ if (typeof src === "object") {
+ if (Object.prototype.toString.call(src) === '[object Array]') {
+ result[prefix + name] = src;
+ }
+ else if (src && src.toJSON) {
+ result[prefix + name] = src.toJSON();
+ }
+ else {
+ flatten(src, result, prefix + name + ".");
+ }
+ }
+ else {
+ result[prefix + name] = src;
+ }
+ }
+ return result;
+ }
+
+ // end of form.js
+
+ // searchable.js -> 3265
+ var Searchable = function(options) {
+
+ // properties
+ this.data = [];
+ this.name = null;
+ this.theme = "searchable_default";
+ this._disabled = false;
+ this.listZIndex = 100000;
+
+
+ // events
+ this.onSelect = null;
+
+ // state
+ this._selected = null;
+ this._highlighted = null;
+ this._collapsed = false;
+ // this._focused = false;
+ // this._filter = null;
+
+ // view
+ this._input = null;
+ this._list = null;
+ this._options = [];
+ this._hidden = null;
+
+ options = options || {};
+
+ var t = this;
+
+ var specialHandling = {
+ "selected": {
+ post: function(val) {
+ if (typeof val === "object" && val.id) {
+ t._selected = val;
+ }
+ else if (typeof val === "string" || typeof val === "number") {
+ t.select(val);
+ }
+ }
+ }
+ };
+
+ Object.defineProperty(this, "selected", {
+ get: function() {
+ return this._selected;
+ },
+ });
+
+ Object.defineProperty(this, "disabled", {
+ get: function() {
+ return this._disabled;
+ },
+ set: function(val) {
+ this._disabled = val;
+ if (this._input) {
+ this._input.disabled = val;
+ if (val) {
+ // verify
+ this._cancel();
+ }
+ }
+ }
+ });
+
+ for (var name in options) {
+ if (!specialHandling[name]) {
+ this[name] = options[name];
+ }
+ }
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ specialHandling[name].post(options[name]);
+ }
+ }
+
+ };
+
+ Searchable.prototype.select = function(id) {
+ this._selected = this.data.find(function(item) { return item.id === id });
+ this._doOnSelect(false);
+ return this;
+ };
+
+
+ Searchable.prototype.create = function() {
+ var component = this;
+ var t = this;
+
+ var div = document.createElement("div");
+ div.className = this.theme + "_main";
+ div.style.position = "relative";
+
+ var icon = document.createElement("div");
+ icon.className = this.theme + "_icon";
+ icon.style.position = "absolute";
+ icon.style.right = "0";
+ icon.style.top = "0";
+ icon.style.bottom = "0";
+ icon.style.width = "20px";
+ icon.addEventListener("mousedown", function(ev) {
+ ev.preventDefault();
+ if (component._collapsed) {
+ component.focus();
+ expand();
+ }
+ else {
+ cancel();
+ collapse();
+ }
+ });
+
+ var list = document.createElement("div");
+ list.className = this.theme + "_list";
+ list.style.display = "none";
+ list.style.position = "absolute";
+ list.style.zIndex = this.listZIndex;
+
+ var hidden = document.createElement("input");
+ hidden.type = "hidden";
+ hidden.name = this.name;
+ hidden.searchable = t;
+ this._hidden = hidden;
+
+ var input = document.createElement("input");
+ input.type = "text";
+ input.className = this.theme + "_input";
+ input.disabled = this._disabled;
+ input.addEventListener("click", function(ev) {
+ // console.log("click");
+ expand();
+ });
+ input.addEventListener("focus", function(ev) {
+ // do not call expand() here, the readonly status has to stay
+ // input.value = "";
+ // console.log("focus");
+ filter("all");
+ });
+ input.addEventListener("input", function(ev) {
+ filter();
+ });
+ input.addEventListener("blur", function(ev) {
+ input.removeAttribute("readonly");
+ cancel();
+ });
+
+ input.addEventListener("keydown", function(ev) {
+ if (component._collapsed) {
+ if (ev.key === "Enter") {
+ return;
+ }
+ if (ev.key === "Esc" || ev.key === "Escape") {
+ return;
+ }
+ expand();
+ }
+
+ if (ev.key === "ArrowDown") {
+ var index = t._options.indexOf(t._highlighted);
+ if (index + 1 < t._options.length) {
+ t._highlighted = t._options[index + 1];
+ }
+ updateHiglight();
+ } else if (ev.key === "ArrowUp") {
+ var index = t._options.indexOf(t._highlighted);
+ if (index - 1 >= 0) {
+ t._highlighted = t._options[index - 1];
+ }
+ updateHiglight();
+ } else if (ev.key === "Enter") {
+ if (component._highlighted) {
+ ev.stopPropagation();
+ selectOption(component._highlighted);
+ }
+ else {
+ ev.stopPropagation();
+ // t._cancel();
+ cancel();
+ collapse();
+ }
+ } else if (ev.key === "Esc" || ev.key === "Escape") {
+ ev.stopPropagation();
+ cancel();
+ collapse();
+ }
+ });
+ this._input = input;
+ this._list = list;
+
+ if (!this._selected) {
+ this._selected = this.data[0];
+ if (this._selected) {
+ input.value = this._selected.name;
+ }
+ }
+
+ function filter(strategy) {
+ // component._collapsed = false;
+
+ var defaultStrategy = component.strategy;
+ if (component.strategy !== "includes" && component.strategy !== "startsWith") {
+ defaultStrategy = "includes";
+ }
+
+ strategy = strategy || defaultStrategy || "includes";
+
+ list.style.display = "";
+ list.style.top = input.offsetHeight + "px";
+ list.style.left = "0px";
+ list.style.width = input.offsetWidth + "px";
+ list.innerHTML = "";
+
+ // allow scrollbar access
+ list.addEventListener("mousedown", function(ev) {
+ ev.preventDefault();
+ });
+
+ component._highlighted = null;
+
+ component._options = [];
+
+ var first = null;
+
+ component.data.forEach(function(item) {
+
+ var name = item.name || item.id;
+
+ if (strategy === "includes") {
+ if (name.toLowerCase().indexOf(input.value.toLowerCase()) === -1) {
+ return;
+ }
+ }
+ else if (strategy === "startsWith") {
+ if (name.toLowerCase().indexOf(input.value.toLowerCase()) !== 0) {
+ return;
+ }
+ }
+ else if (strategy === "all") {
+ // don't skip
+ }
+
+
+ var option = document.createElement("div");
+ option.className = component.theme + "_list_item";
+ option.innerText = name;
+
+ option.item = item;
+
+ if (item === component._selected) {
+ component._highlighted = option;
+ }
+ if (!first) {
+ first = option;
+ }
+
+ // "mousedown" goes before blur, "click" goes after
+ option.addEventListener("mousedown", function(ev) {
+ selectOption(option);
+ ev.preventDefault();
+ });
+
+ option.addEventListener("mousemove", function(ev) {
+ if (component._highlighted === option) {
+ return;
+ }
+ component._highlighted = option;
+ updateHiglight({dontScroll: true});
+ });
+
+ list.appendChild(option);
+
+ component._options.push(option);
+ });
+
+ if (!component._highlighted) {
+ component._highlighted = first;
+ }
+ updateHiglight();
+ }
+
+ function updateHiglight(options) {
+ options = options || {};
+ var scrollIntoView = !options.dontScroll;
+
+ // redo, avoid selectors
+ var previous = document.querySelectorAll("." + component.theme + "_list_item_highlight");
+ previous.forEach(function(p) {
+ p.className = p.className.replace(component.theme + "_list_item_highlight", "");
+ });
+
+ if (component._highlighted) {
+ component._highlighted.className += " " + component.theme + "_list_item_highlight";
+
+ if (scrollIntoView && !isScrolledIntoView(component._highlighted, list)) {
+ component._highlighted.scrollIntoView();
+ }
+ }
+ }
+
+ function isScrolledIntoView(target, viewport) {
+ var tRect = target.getBoundingClientRect();
+ var vRect = viewport.getBoundingClientRect();
+ return tRect.top >= vRect.top && tRect.bottom <= vRect.bottom;
+ }
+
+ function selectOption(option) {
+ var item = option.item;
+
+ // input.value = option.innerText;
+ component._selected = item;
+ component._doOnSelect(true);
+ hide();
+ collapse();
+ }
+
+ function cancel() {
+ component._cancel();
+ }
+
+ function hide() {
+ component._hide();
+ }
+
+ function collapse() {
+ component._collapsed = true;
+ input.setAttribute("readonly", "readonly");
+ input.focus();
+ }
+
+ function expand() {
+ component._collapsed = false;
+ input.removeAttribute("readonly");
+ // input.value = "";
+ input.select();
+ filter("all");
+ }
+
+ div.appendChild(input);
+ div.appendChild(icon);
+ div.appendChild(hidden);
+ div.appendChild(list);
+ return div;
+ };
+
+ Searchable.prototype._cancel = function() {
+ this._hide();
+ if (!this._selected) {
+ this._input.value = "";
+ // not sure about this:
+ this._doOnSelect(true);
+ } else {
+ this._input.value = this._selected.name;
+ }
+ // collapse();
+ }
+
+ Searchable.prototype.focus = function() {
+ // this._input.focus();
+
+ this._collapsed = true;
+ this._input.setAttribute("readonly", "readonly");
+ this._input.focus();
+ this._cancel();
+ };
+
+ Searchable.prototype._hide = function() {
+ this._list.style.display = "none";
+ };
+
+ Searchable.prototype._doOnSelect = function(byUser) {
+ this._hidden.value = this.selected ? this.selected.id : null;
+ if (this._selected) {
+ this._input.value = this._selected.name;
+ }
+ else {
+ this._input.value = "";
+ }
+
+ if (typeof this.onSelect === "function") {
+ var args = {
+ control: this,
+ ui: byUser
+ };
+ this.onSelect(args);
+ }
+ };
+
+ // end of searchable.js
+
+ // table.js -> 3641
+ var Table = function(options) {
+
+ this.form = null; // Form instance
+ this.item = null; // table def from form
+ this.data = null;
+ this.name = null;
+ this.theme = "edit_table_default";
+ this.onInput = null;
+
+ this.nav = {};
+
+ this._activeEdit = null;
+ this._rows = [];
+
+ options = options || {};
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+
+ };
+
+ Table.prototype.create = function() {
+ var table = this;
+
+ var div = document.createElement("div");
+ div.className = this.theme + "_main";
+ div.style.position = "relative";
+
+ var hidden = document.createElement("input");
+ hidden.type = "hidden";
+ hidden.name = table.name;
+ hidden.table = this;
+ div.appendChild(hidden);
+
+ var tableElement = document.createElement("div");
+ tableElement.className = this.theme + "_table";
+ var header = this._createHeader();
+ tableElement.appendChild(header);
+
+ var spacerRow = table._createRowState({});
+ spacerRow.spacer = true;
+ var spacer = this._renderRow(spacerRow);
+ // spacer.style.visibility = "hidden";
+ spacer.classList.add(table.theme + "_spacer");
+ tableElement.appendChild(spacer);
+
+ var body = document.createElement("div");
+ body.className = table.theme + "_tbody";
+ tableElement.appendChild(body);
+
+ div.appendChild(tableElement);
+
+ var after = document.createElement("div");
+ div.appendChild(after);
+
+ this.nav.body = body;
+ this.nav.table = tableElement;
+ this.nav.main = div;
+ this.nav.after = after;
+
+ var add = document.createElement("div");
+ var plus = document.createElement("span");
+ plus.className = this.theme + "_plus";
+
+ plus.addEventListener("click", function(ev) {
+ if (table.disabled) {
+ return;
+ }
+
+ var generate = table.item.onNewRow;
+ var value = {};
+ if (typeof generate === "function") {
+ var args = {};
+ args.result = table.form.serialize();
+ args.value = {};
+ generate(args);
+ value = args.value;
+ }
+
+ var row = table._createRowState(value);
+ table._rows.push(row);
+ table._render();
+ table._doOnInput();
+ });
+
+ this.nav.plus = plus;
+
+ add.appendChild(plus);
+ div.appendChild(add);
+
+ return div;
+ };
+
+ Table.prototype._createHeader = function() {
+ var table = this;
+ var row = document.createElement("div");
+ row.classList.add(this.theme + "_row");
+ row.classList.add(this.theme + "_header");
+ this.item.columns.forEach(function(item) {
+ var cell = document.createElement("div");
+ cell.classList.add(table.theme + "_cell");
+ cell.innerText = item.name;
+ row.appendChild(cell);
+ });
+ return row;
+ };
+
+ Table.prototype._maxRowsReached = function() {
+ var max = this.item.max || 0;
+ if (max && this._rows.length >= max) {
+ return true;
+ }
+ return false;
+ };
+
+ Table.prototype.save = function() {
+ var table = this;
+ var data = [];
+
+ table._rows.forEach(function(row) {
+ var item = {};
+ row.cells.forEach(function(cell) {
+ item[cell.id] = cell.value;
+ });
+ data.push(item);
+ });
+
+ return data;
+ };
+
+ Table.prototype.load = function(data) {
+ var table = this;
+
+ var isArray = Object.prototype.toString.call(data) === '[object Array]';
+ if (!isArray) {
+ throw new Error("Array expected");
+ }
+
+ this.data = data;
+
+ this._createState();
+ this._render();
+ };
+
+ Table.prototype._updateCss = function() {
+ if (this.disabled) {
+ this.nav.main.classList.add(this.theme + "_disabled");
+ }
+ else {
+ this.nav.main.classList.remove(this.theme + "_disabled");
+ }
+
+ var maxReached = this._maxRowsReached();
+ if (maxReached) {
+ this.nav.plus.classList.add(this.theme + "_plus_max");
+ }
+ else {
+ this.nav.plus.classList.remove(this.theme + "_plus_max");
+ }
+ };
+
+ Table.prototype._createState = function() {
+ var table = this;
+ this._rows = [];
+ this.data.forEach(function(dataRow) {
+ var row = table._createRowState(dataRow);
+ table._rows.push(row);
+ });
+ };
+
+ Table.prototype._removeRow = function(row) {
+ var table = this;
+
+ var index = table._rows.indexOf(row);
+ table._rows.splice(index, 1);
+ };
+
+ Table.prototype._createRowState = function(dataRow) {
+ var table = this;
+
+ var row = {};
+ row.data = dataRow;
+ row.cells = [];
+
+ table.item.columns.forEach(function(formItem) {
+ var id = formItem.id;
+ var value = dataRow[id];
+
+ var type = table._formItemType(formItem);
+ if (typeof value === "undefined") {
+ if (type === "text") {
+ value = "";
+ }
+ else if (type === "number") {
+ value = 0;
+ }
+ else if (type === "select") {
+ var options = formItem.options;
+ value = options && options[0].id;
+ }
+ }
+
+ var cell = {};
+ cell.id = id;
+ cell.value = value;
+ cell.type = type;
+ cell.data = formItem;
+ row.cells.push(cell);
+ });
+
+ return row;
+ };
+
+ Table.prototype._formItemType = function(formItem) {
+ var type = formItem.type;
+ if (!type) {
+ if (formItem.options) {
+ type = "select";
+ }
+ else {
+ type = "text";
+ }
+ }
+ return type;
+ };
+
+ Table.prototype._render = function() {
+
+ var table = this;
+ this.nav.body.innerHTML = "";
+ this.nav.after.innerHTML = "";
+
+ this._rows.forEach(function(row) {
+ var el = table._renderRow(row);
+ table.nav.body.appendChild(el);
+ });
+
+ if (this._rows.length === 0) {
+ var el = table._renderEmpty();
+ table.nav.after.appendChild(el);
+ }
+
+ this._updateCss();
+ };
+
+ Table.prototype._renderEmpty = function() {
+ var div = document.createElement("div");
+ div.className = this.theme + "_empty";
+
+ return div;
+ };
+
+ Table.prototype._renderRow = function(row) {
+ var table = this;
+ var el = document.createElement("div");
+ el.className = table.theme + "_row";
+
+ row.cells.forEach(function(cell) {
+
+ var cellEl = document.createElement("div");
+ cellEl.className = table.theme + "_cell";
+ var interactive = table._renderCell(cell);
+
+ if (row.spacer) {
+ var wrap = document.createElement("div");
+ wrap.style.height = "0px";
+ wrap.style.overflow = "hidden";
+ wrap.appendChild(interactive);
+
+ cellEl.appendChild(wrap);
+ }
+ else {
+ cellEl.appendChild(interactive);
+
+ }
+
+ el.appendChild(cellEl);
+ });
+
+ var cell = document.createElement("div");
+ cell.classList.add(table.theme + "_cell");
+ cell.classList.add(table.theme + "_rowaction");
+
+ var span = document.createElement("span");
+ span.className = this.theme + "_delete";
+
+ span.addEventListener("click", function(ev) {
+ if (table.disabled) {
+ return;
+ }
+ table._removeRow(row);
+ table._render();
+ table._doOnInput();
+ });
+
+ if (!row.spacer) {
+ cell.appendChild(span);
+ }
+
+
+ el.appendChild(cell);
+
+ return el;
+ };
+
+ Table.prototype._doOnInput = function() {
+ var table = this;
+ if (typeof table.onInput === "function") {
+ var args = {}
+ table.onInput(args);
+ }
+ };
+
+ Table.prototype._renderCell = function(cell) {
+ var table = this;
+ var type = cell.type;
+ if (type === "text" || type === "number") {
+ var input = document.createElement("input");
+ input.type = type;
+ if (table.disabled) {
+ input.disabled = true;
+ }
+ if (cell.value) {
+ input.value = cell.value;
+ }
+ input.addEventListener("keyup", function(ev) {
+ if (type === "number") {
+ cell.value = Number(this.value);
+ }
+ else {
+ cell.value = this.value;
+ }
+ table._doOnInput();
+ });
+ return input;
+ }
+ else if (type === "select") {
+ var select = document.createElement("select");
+ if (table.disabled) {
+ select.disabled = true;
+ }
+
+ cell.data.options.forEach(function(item) {
+ var option = document.createElement("option");
+ option.innerText = item.name;
+ option.value = item.id;
+ option._originalValue = item.id;
+
+ select.appendChild(option);
+
+ if (cell.value === item.id) {
+ option.setAttribute("selected", true);
+ }
+ });
+
+ select.addEventListener("change", function(ev) {
+ var option = select.options[select.selectedIndex];
+ if (option && typeof option._originalValue !== "undefined") {
+ cell.value = option._originalValue;
+ }
+ table._doOnInput();
+ });
+
+ return select;
+ }
+
+ throw new Error("Unsupported item type: " + type);
+ };
+
+ Table.prototype.focus = function() {
+ };
+
+ // end of table.js
+
+})(DayPilot);
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function() {
+
+ var doNothing = function() { };
+
+ if (typeof DayPilot.Month !== 'undefined' && DayPilot.Month.events) {
+ return;
+ }
+
+ var DayPilotMonth = {};
+
+ DayPilotMonth.Month = function(placeholder, options) {
+ this.v = '2024.3.539-lite';
+ this.nav = {};
+
+ var calendar = this;
+
+ this.id = placeholder;
+ this.isMonth = true;
+
+ // this.angularAutoApply = false;
+ this.api = 2;
+ this.backendUrl = null;
+ this.cellHeaderHeight = 24;
+ this.cellHeight = 100; // default cell height is 100 pixels (it's a minCellHeight, it will be extended if needed)
+ this.cellMarginBottom = 0;
+ this.contextMenu = null;
+ this.cssClassPrefix = "month_default";
+ this.eventBarVisible = true;
+ this.eventHeight = 25;
+ this.eventsLoadMethod = "GET";
+ this.headerHeight = 30;
+ this.hideUntilInit = true;
+ this.lineSpace = 1;
+ this.locale = "en-us";
+ this.showToolTip = true;
+ this.startDate = new DayPilot.Date(); // today
+ this.theme = null;
+ this.visible = true;
+ this.weekStarts = "Auto";
+ this.width = '100%'; // default width is 100%
+ this.xssProtection = "Enabled";
+
+ this.afterRender = function() { };
+
+ this.cellHeaderClickHandling = "Enabled";
+ this.eventClickHandling = "Enabled";
+ this.eventDeleteHandling = "Disabled";
+ this.eventMoveHandling = "Update";
+ this.eventResizeHandling = "Update";
+ this.eventRightClickHandling = "ContextMenu";
+ this.headerClickHandling = "Enabled";
+ this.timeRangeSelectedHandling = "Enabled";
+
+ this.onCellHeaderClick = null;
+ this.onCellHeaderClicked = null;
+ this.onEventClick = null;
+ this.onEventClicked = null;
+ this.onEventDelete = null;
+ this.onEventDeleted = null;
+ this.onEventMove = null;
+ this.onEventMoved = null;
+ this.onEventResize = null;
+ this.onEventResized = null;
+ this.onEventRightClick = null;
+ this.onEventRightClicked = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+
+ this.onBeforeEventRender = null;
+ this.onBeforeCellRender = null;
+
+ this.cellEvents = [];
+
+ this.elements = {};
+ this.elements.events = [];
+
+ this.cache = {};
+
+ this._disposed = false;
+
+ this._updateView = function(result, context) {
+
+ var result = JSON.parse(result);
+
+ if (result.CallBackRedirect) {
+ document.location.href = result.CallBackRedirect;
+ return;
+ }
+
+ if (result.UpdateType === "None") {
+ calendar.fireAfterRenderDetached(result.CallBackData, true);
+ return;
+ }
+
+ calendar.events.list = result.Events;
+
+ if (result.UpdateType === "Full") {
+
+ // properties
+ calendar.startDate = result.StartDate;
+ // calendar.headerBackColor = result.HeaderBackColor ? result.HeaderBackColor : calendar.headerBackColor;
+ // calendar.backColor = result.BackColor ? result.BackColor : calendar.backColor;
+ // calendar.nonBusinessBackColor = result.NonBusinessBackColor ? result.NonBusinessBackColor : calendar.nonBusinessBackColor;
+ calendar.timeFormat = result.TimeFormat ? result.TimeFormat : calendar.timeFormat;
+ if (typeof result.WeekStarts !== 'undefined') { calendar.weekStarts = result.WeekStarts; } // number, can be 0
+
+ calendar.hashes = result.Hashes;
+ }
+
+ calendar._deleteEvents();
+ calendar._prepareRows();
+ calendar._loadEvents();
+
+ if (result.UpdateType === "Full") {
+ calendar._clearTable();
+ calendar._drawTable();
+ }
+ calendar._updateHeight();
+
+ calendar.show();
+
+ calendar._drawEvents();
+
+ calendar.fireAfterRenderDetached(result.CallBackData, true);
+
+ };
+
+ this.fireAfterRenderDetached = function(data, isCallBack) {
+ var afterRenderDelayed = function(data, isc) {
+ return function() {
+ if (calendar.afterRender) {
+ calendar.afterRender(data, isc);
+ }
+ };
+ };
+
+ window.setTimeout(afterRenderDelayed(data, isCallBack), 0);
+ };
+
+ this.lineHeight = function() {
+ return this.eventHeight + this.lineSpace;
+ };
+
+ this.events = {};
+ //this.events.list = [];
+
+ this.events.add = function(e) {
+
+ var data = null;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ data = e;
+ }
+ else {
+ throw "DayPilot.Month.events.add() expects an object or DayPilot.Event instance.";
+ }
+
+ // e.calendar = calendar;
+
+ if (!calendar.events.list) {
+ calendar.events.list = [];
+ }
+
+ calendar.events.list.push(data);
+ calendar.update();
+ calendar._angular.notify();
+ };
+
+ this.events.find = function(id) {
+ if (!calendar.events.list) {
+ return null;
+ }
+
+ if (typeof id === "function") {
+ var fn = id;
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var e = new DayPilot.Event(calendar.events.list[i], calendar);
+ if (fn(e)) {
+ return e;
+ }
+ }
+ return null;
+ }
+
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var data = calendar.events.list[i];
+ if (data.id === id) {
+ return new DayPilot.Event(data, calendar);
+ }
+ }
+ return null;
+ };
+
+ this.events.update = function(e) {
+ if (e instanceof DayPilot.Event) {
+ e.commit();
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ var index = DayPilot.indexOf(calendar.events.list, target.data);
+ calendar.events.list.splice(index, 1, e);
+ }
+ }
+
+ calendar.update();
+ calendar._angular.notify();
+ };
+
+ this.events.remove = function(e) {
+ var data;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ data = target.data;
+ }
+ }
+ else if (typeof e === "string" || typeof e === "number") {
+ var target = calendar.events.find(e);
+ if (target) {
+ data = target.data;
+ }
+ }
+
+ var index = DayPilot.indexOf(calendar.events.list, data);
+ calendar.events.list.splice(index, 1);
+ calendar.update();
+ calendar._angular.notify();
+ };
+
+ this.events.load = function(url, success, error) {
+ var onError = function (args) {
+ var largs = {};
+ largs.exception = args.exception;
+ largs.request = args.request;
+
+ if (typeof error === 'function') {
+ error(largs);
+ }
+ };
+
+ var onSuccess = function (args) {
+ var r = args.request;
+ var data;
+
+ // it's supposed to be JSON
+ try {
+ data = JSON.parse(r.responseText);
+ }
+ catch (e) {
+ var fargs = {};
+ fargs.exception = e;
+ onError(fargs);
+ return;
+ }
+
+ if (DayPilot.isArray(data)) {
+ var sargs = {};
+ sargs.preventDefault = function () {
+ this.preventDefault.value = true;
+ };
+ sargs.data = data;
+ if (typeof success === "function") {
+ success(sargs);
+ }
+
+ if (sargs.preventDefault.value) {
+ return;
+ }
+
+ calendar.events.list = data;
+ if (calendar._initialized) {
+ calendar.update();
+ }
+ }
+ };
+
+ var usePost = calendar.eventsLoadMethod && calendar.eventsLoadMethod.toUpperCase() === "POST";
+
+ if (usePost) {
+ DayPilot.Http.ajax({
+ "method": "POST",
+ "data": {"start": calendar.visibleStart().toString(), "end": calendar.visibleEnd().toString()},
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ else {
+ var fullUrl = url;
+ var queryString = "start=" + calendar.visibleStart().toString() + "&end=" + calendar.visibleEnd().toString();
+ if (fullUrl.indexOf("?") > -1) {
+ fullUrl += "&" + queryString;
+ }
+ else {
+ fullUrl += "?" + queryString;
+ }
+
+ DayPilot.Http.ajax({
+ "method": "GET",
+ "url": fullUrl,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ };
+
+ this.update = function(options) {
+
+ calendar._loadOptions(options);
+
+ if (!this._initialized) {
+ return;
+ }
+
+ if (calendar._disposed) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Month instance that has been disposed.");
+ }
+
+ if (!this.cells) {
+ return;
+ }
+
+ var full = true;
+
+ calendar._deleteEvents();
+ calendar._prepareRows();
+ calendar._loadEvents();
+
+ if (full) {
+ calendar._clearTable();
+ calendar._drawTable();
+ }
+ calendar._updateHeight();
+ calendar._show();
+ calendar._drawEvents();
+
+ if (this.visible) {
+ this.show();
+ }
+ else {
+ this.hide();
+ }
+ };
+
+ this._specialHandling = null;
+ this._loadOptions = function(options) {
+ if (!options) {
+ return;
+ }
+ var specialHandling = {
+ "events": {
+ "preInit": function() {
+ var events = this.data || [];
+ if (DayPilot.isArray(events.list)) {
+ calendar.events.list = events.list;
+ }
+ else {
+ calendar.events.list = events;
+ }
+ }
+ }
+ };
+ this._specialHandling = specialHandling;
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ var item = specialHandling[name];
+ item.data = options[name];
+ if (item.preInit) {
+ item.preInit();
+ }
+ }
+ else {
+ calendar[name] = options[name];
+ }
+ }
+
+ };
+
+ this._postInit = function() {
+ var specialHandling = this._specialHandling;
+ for (var name in specialHandling) {
+ var item = specialHandling[name];
+ if (item.postInit) {
+ item.postInit();
+ }
+ }
+ };
+
+ this._cache = {};
+ this._cache.events = [];
+
+ this._doBeforeEventRender = function(i) {
+ var cache = this._cache.events;
+ var data = this.events.list[i];
+ var evc = {};
+
+ // make a copy
+ for (var name in data) {
+ evc[name] = data[name];
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ var args = {};
+ args.control = calendar;
+ args.data = evc;
+ this.onBeforeEventRender(args);
+ }
+
+ cache[i] = evc;
+
+ };
+
+ this._loadEvents = function() {
+ var events = this.events.list;
+
+ if (!events) {
+ return;
+ }
+
+ if (!DayPilot.isArray(events)) {
+ throw new DayPilot.Exception("DayPilot.Month.events.list expects an array object. You supplied: " + (typeof events));
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ for (var i = 0; i < events.length; i++) {
+ this._doBeforeEventRender(i);
+ }
+ }
+
+ // prepare rows and columns
+ for (var x = 0; x < events.length; x++) {
+ var data = events[x];
+
+ if (typeof data !== "object") {
+ throw new DayPilot.Exception("Event data item must be an object");
+ }
+ if (!data.start) {
+ throw new DayPilot.Exception("Event data item must specify 'start' property");
+ }
+ if (!data.end) {
+ throw new DayPilot.Exception("Event data item must specify 'end' property");
+ }
+
+ var start = new DayPilot.Date(data.start);
+ var end = new DayPilot.Date(data.end);
+ if (start.getTime() > end.getTime()) { // skip invalid events, zero duration allowed
+ continue;
+ }
+ for (var i = 0; i < this.rows.length; i++) {
+ var row = this.rows[i];
+ var ep = new DayPilot.Event(data, this);
+ if (row.belongsHere(ep)) {
+ row.events.push(ep);
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ ep.cache = this._cache.events[x];
+ }
+ }
+ }
+ }
+
+ // arrange events into lines
+ for (var ri = 0; ri < this.rows.length; ri++) {
+ var row = this.rows[ri];
+ row.events.sort(this._eventComparer);
+
+ for (var ei = 0; ei < this.rows[ri].events.length; ei++) {
+ var ev = row.events[ei];
+ var colStart = row.getStartColumn(ev);
+ var colWidth = row.getWidth(ev);
+ var line = row.putIntoLine(ev, colStart, colWidth, ri);
+ }
+ }
+ };
+
+
+ this._deleteEvents = function() {
+ for (var i = 0; i < this.elements.events.length; i++) {
+ var e = this.elements.events[i];
+ e.event = null;
+ e.click = null;
+ e.parentNode.removeChild(e);
+ }
+
+ this.elements.events = [];
+
+ };
+
+ this._drawEvents = function() {
+ //this.cache.events = {}; // reset DayPilotMonth.Event object cache
+
+ this._drawEventsRows();
+ };
+
+ this._drawEventsRows = function() {
+ this.elements.events = [];
+
+ // draw events
+ for (var ri = 0; ri < this.rows.length; ri++) {
+ var row = this.rows[ri];
+
+ for (var li = 0; li < row.lines.length; li++) {
+ var line = row.lines[li];
+
+ for (var pi = 0; pi < line.length; pi++) {
+ this._drawEvent(line[pi]);
+ }
+ }
+ }
+
+ };
+
+ this._eventComparer = function(a, b) {
+ if (!a || !b || !a.start || !b.start) {
+ return 0; // no sorting, invalid arguments
+ }
+
+ var byStart = a.start().getTime() - b.start().getTime();
+ if (byStart !== 0) {
+ return byStart;
+ }
+
+ var byEnd = b.end().getTime() - a.end().getTime(); // desc
+ return byEnd;
+ };
+
+ this.drawShadow = function(x, y, line, width, offset, e) {
+
+ if (!offset) {
+ offset = 0;
+ }
+
+ var remains = width;
+
+ this.shadow = {};
+ this.shadow.list = [];
+ this.shadow.start = { x: x, y: y };
+ this.shadow.width = width;
+
+ // something before the first day
+ var hidden = y * 7 + x - offset;
+ if (hidden < 0) {
+ //document.title = hidden + ' ' + new Date();
+ remains += hidden;
+ x = 0;
+ y = 0;
+ }
+
+ var remainingOffset = offset;
+ while (remainingOffset >= 7) {
+ y--;
+ remainingOffset -= 7;
+ }
+ if (remainingOffset > x) {
+ var plus = 7 - this.getColCount();
+ if (remainingOffset > (x + plus)) {
+ y--;
+ x = x + 7 - remainingOffset;
+ }
+ else {
+ remains = remains - remainingOffset + x;
+ x = 0;
+ }
+ }
+ else {
+ x -= remainingOffset;
+ }
+
+ if (y < 0) {
+ y = 0;
+ x = 0;
+ }
+
+ var cursor = null;
+ if (DayPilotMonth.resizingEvent) {
+ cursor = 'w-resize';
+ }
+ else if (DayPilotMonth.movingEvent) {
+ cursor = "move";
+ }
+
+ this.nav.top.style.cursor = cursor;
+
+ while (remains > 0 && y < this.rows.length) {
+ var drawNow = Math.min(this.getColCount() - x, remains);
+ var row = this.rows[y];
+
+
+ var top = this.getRowTop(y);
+ var height = row.getHeight();
+
+ var shadow = document.createElement("div");
+ shadow.setAttribute("unselectable", "on");
+ shadow.style.position = 'absolute';
+ shadow.style.left = (this.getCellWidth() * x) + '%';
+ shadow.style.width = (this.getCellWidth() * drawNow) + '%';
+ shadow.style.top = (top) + 'px';
+ shadow.style.height = (height) + 'px';
+ shadow.style.cursor = cursor;
+ shadow.classList.add(calendar._prefixCssClass("_shadow"));
+
+ var inside = document.createElement("div");
+ inside.setAttribute("unselectable", "on");
+ shadow.appendChild(inside);
+
+ inside.style.position = "absolute";
+ inside.style.top = "0px";
+ inside.style.right = "0px";
+ inside.style.left = "0px";
+ inside.style.bottom = "0px";
+ inside.classList.add(calendar._prefixCssClass("_shadow_inner"));
+
+ // inside.style.backgroundColor = "#aaaaaa";
+ // inside.style.opacity = 0.5;
+ // inside.style.filter = "alpha(opacity=50)";
+ //inside.style.border = '2px solid #aaaaaa';
+ /*
+ if (e) {
+ inside.style.overflow = 'hidden';
+ inside.style.fontSize = this.eventFontSize;
+ inside.style.fontFamily = this.eventFontFamily;
+ inside.style.color = this.eventFontColor;
+ inside.innerHTML = e.client.html();
+ }
+ */
+
+ this.nav.top.appendChild(shadow);
+ this.shadow.list.push(shadow);
+
+ remains -= (drawNow + 7 - this.getColCount());
+ x = 0;
+ y++;
+ }
+
+ };
+
+ this.clearShadow = function() {
+ if (this.shadow) {
+ for (var i = 0; i < this.shadow.list.length; i++) {
+ this.nav.top.removeChild(this.shadow.list[i]);
+ }
+ this.shadow = null;
+ this.nav.top.style.cursor = '';
+ }
+ };
+
+ this.getEventTop = function(row, line) {
+ var top = this.headerHeight;
+ for (var i = 0; i < row; i++) {
+ top += this.rows[i].getHeight();
+ }
+ top += this.cellHeaderHeight; // space on top
+ top += line * this.lineHeight();
+ return top;
+ };
+
+ this.getDateFromCell = function(x, y) {
+ //return DayPilot.Date.addDays(this.firstDate, y * 7 + x);
+ return this.firstDate.addDays(y * 7 + x);
+ };
+
+ this._drawEvent = function(e) {
+ /*
+
+ supported properties:
+ * cssClass
+ * backColor
+ * borderColor
+ * barColor
+ * barHidden
+ * fontColor
+ * html
+ * toolTip
+ *
+ * clickEnabled
+ */
+
+ var data = e.cache || e.data;
+
+ //var ev = eventPart.event;
+ var row = e.part.row;
+ var line = e.part.line;
+ var colStart = e.part.colStart;
+ var colWidth = e.part.colWidth;
+
+ var left = this.getCellWidth() * (colStart);
+ var width = this.getCellWidth() * (colWidth);
+ var top = this.getEventTop(row, line);
+
+ var div = document.createElement("div");
+ div.setAttribute("unselectable", "on");
+ div.style.height = this.eventHeight + 'px';
+ div.className = this._prefixCssClass("_event");
+
+ if (data.cssClass) {
+ DayPilot.Util.addClass(div, data.cssClass);
+ }
+
+ if (!e.part.startsHere) {
+ DayPilot.Util.addClass(div, this._prefixCssClass("_event_continueleft"));
+ }
+
+ if (!e.part.endsHere) {
+ DayPilot.Util.addClass(div, this._prefixCssClass("_event_continueright"));
+ }
+
+ div.event = e;
+
+ div.style.width = width + '%';
+ div.style.position = 'absolute';
+ div.style.left = left + '%';
+ div.style.top = top + 'px'; // plus space on top
+
+ if (this.showToolTip && e.client.toolTip()) {
+ div.title = e.client.toolTip();
+ }
+
+ div.onclick = calendar._eventClickDispatch;
+ div.oncontextmenu = calendar._onEventContextMenu;
+ div.onmousedown = function(ev) {
+ ev = ev || window.event;
+ var button = ev.which || ev.button;
+
+ ev.cancelBubble = true;
+ if (ev.stopPropagation) {
+ ev.stopPropagation();
+ }
+
+ if (button === 1) {
+
+ DayPilotMonth.movingEvent = null;
+ if (this.style.cursor === 'w-resize' || this.style.cursor === 'e-resize') {
+ var resizing = {};
+ resizing.start = {};
+ resizing.start.x = colStart;
+ resizing.start.y = row;
+ resizing.event = div.event;
+ resizing.width = DayPilot.DateUtil.daysSpan(resizing.event.start(), resizing.event.end()) + 1;
+ resizing.direction = this.style.cursor;
+ DayPilotMonth.resizingEvent = resizing;
+ }
+ else if (this.style.cursor === 'move' || e.client.moveEnabled()) {
+ calendar.clearShadow();
+
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+ if (!coords) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+
+ var hidden = DayPilot.DateUtil.daysDiff(e.start(), calendar.rows[row].start);
+ var offset = (cell.y * 7 + cell.x) - (row * 7 + colStart);
+ if (hidden) {
+ offset += hidden;
+ }
+
+ var moving = {};
+ moving.start = {};
+ moving.start.x = colStart;
+ moving.start.y = row;
+ moving.start.line = line;
+ moving.offset = calendar.eventMoveToPosition ? 0 : offset;
+ moving.colWidth = colWidth;
+ moving.event = div.event;
+ moving.coords = coords;
+ DayPilotMonth.movingEvent = moving;
+
+ }
+
+ }
+ };
+
+ div.onmousemove = function(ev) {
+ if (typeof (DayPilotMonth) === 'undefined') {
+ return;
+ }
+
+ if (DayPilotMonth.movingEvent || DayPilotMonth.resizingEvent) {
+ return;
+ }
+
+ // position
+ var offset = DayPilot.mo3(div, ev);
+ if (!offset) {
+ return;
+ }
+
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "";
+ }
+
+ var resizeMargin = 6;
+
+ if (offset.x <= resizeMargin && e.client.resizeEnabled()) {
+ if (e.part.startsHere) {
+ div.style.cursor = "w-resize";
+ div.dpBorder = 'left';
+ }
+ else {
+ div.style.cursor = 'not-allowed';
+ }
+ }
+ else if (div.clientWidth - offset.x <= resizeMargin && e.client.resizeEnabled()) {
+ if (e.part.endsHere) {
+ div.style.cursor = "e-resize";
+ div.dpBorder = 'right';
+ }
+ else {
+ div.style.cursor = 'not-allowed';
+ }
+ }
+ else if (e.client.clickEnabled()) {
+ div.style.cursor = "pointer";
+ }
+ else {
+ div.style.cursor = 'default';
+ }
+
+ };
+
+ div.onmouseleave = function(ev) {
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "none";
+ }
+ div.style.cursor = '';
+ };
+
+ div.onmouseenter = function(ev) {
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "";
+ }
+ };
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = this._prefixCssClass("_event_inner");
+
+ if (data.borderColor === "darker" && data.backColor) {
+ inner.style.borderColor = DayPilot.ColorUtil.darker(data.backColor, 2);
+ }
+ else {
+ inner.style.borderColor = data.borderColor;
+ }
+
+ if (data.backColor) {
+ inner.style.background = data.backColor;
+ }
+
+ if (data.fontColor) {
+ inner.style.color = data.fontColor;
+ }
+
+
+ inner.innerHTML = e.client.html();
+
+ div.appendChild(inner);
+
+ if (e.client.barVisible()) {
+
+ var bar = document.createElement("div");
+ bar.setAttribute("unselectable", "on");
+ bar.className = this._prefixCssClass("_event_bar");
+ bar.style.position = "absolute";
+
+ var barInner = document.createElement("div");
+ barInner.setAttribute("unselectable", "on");
+ barInner.className = this._prefixCssClass("_event_bar_inner");
+ barInner.style.top = "0%";
+ barInner.style.height = "100%";
+
+ if (data.barColor) {
+ barInner.style.backgroundColor = data.barColor;
+ }
+
+ bar.appendChild(barInner);
+ div.appendChild(bar);
+ }
+
+
+ if (e.client.deleteEnabled()) {
+ var dheight = 18;
+ var dwidth = 18;
+ var dtop = Math.floor(calendar.eventHeight / 2 - dheight / 2);
+
+ var del = document.createElement("div");
+ del.style.position = "absolute";
+ del.style.right = "2px";
+ del.style.top = dtop + "px";
+ del.style.width = dwidth + "px";
+ del.style.height = dheight + "px";
+ del.className = calendar._prefixCssClass("_event_delete");
+ del.onmousedown = function(ev) {
+ ev.stopPropagation();
+ };
+ del.onclick = function(ev) {
+ ev.stopPropagation();
+ var e = this.parentNode.event;
+ if (e) {
+ calendar._eventDeleteDispatch(e);
+ }
+ };
+ del.style.display = "none";
+ div.deleteIcon = del;
+ div.appendChild(del);
+ }
+
+ // areas
+ var areas = data.areas ? DayPilot.Areas.copy(data.areas) : [];
+ DayPilot.Areas.attach(div, e, {"areas": areas});
+
+ if (typeof calendar.onAfterEventRender === 'function') {
+ var args = {};
+ args.e = div.event;
+ args.div = div;
+
+ calendar.onAfterEventRender(args);
+ }
+
+ this.elements.events.push(div);
+
+ this.nav.events.appendChild(div);
+ };
+
+
+ // returns DayPilot.Date object
+ this.lastVisibleDayOfMonth = function() {
+ return this.startDate.lastDayOfMonth();
+ };
+
+ this._prepareRows = function() {
+
+ if (typeof this.startDate === 'string') {
+ this.startDate = new DayPilot.Date(this.startDate);
+ }
+ this.startDate = this.startDate.firstDayOfMonth();
+
+ this.firstDate = this.startDate.firstDayOfWeek(this.getWeekStart());
+
+ var firstDayOfMonth = this.startDate;
+
+ var rowCount;
+
+ var lastVisibleDayOfMonth = this.lastVisibleDayOfMonth();
+ var count = DayPilot.DateUtil.daysDiff(this.firstDate, lastVisibleDayOfMonth) + 1;
+ rowCount = Math.ceil(count / 7);
+
+ this.days = rowCount * 7;
+
+ this.rows = [];
+ for (var x = 0; x < rowCount; x++) {
+ var r = {};
+ r.start = this.firstDate.addDays( x * 7); // start point
+ r.end = r.start.addDays(this.getColCount()); // end point
+ r.events = []; // collection of events
+ r.lines = []; // collection of lines
+ r.index = x; // row index
+ r.minHeight = this.cellHeight; // default, can be extended during events loading
+ r.calendar = this;
+
+ r.belongsHere = function(ev) {
+ if (ev.end().getTime() === ev.start().getTime() && ev.start().getTime() === this.start.getTime()) {
+ return true;
+ }
+ return !(ev.end().getTime() <= this.start.getTime() || ev.start().getTime() >= this.end.getTime());
+ };
+
+ r.getPartStart = function(ev) {
+ return DayPilot.DateUtil.max(this.start, ev.start());
+ };
+
+ r.getPartEnd = function(ev) {
+ return DayPilot.DateUtil.min(this.end, ev.end());
+ };
+
+ r.getStartColumn = function(ev) {
+ var partStart = this.getPartStart(ev);
+ return DayPilot.DateUtil.daysDiff(this.start, partStart);
+ };
+
+ r.getWidth = function(ev) {
+ return DayPilot.DateUtil.daysSpan(this.getPartStart(ev), this.getPartEnd(ev)) + 1;
+ };
+
+ r.putIntoLine = function(ev, colStart, colWidth, row) {
+ var thisRow = this;
+
+ for (var i = 0; i < this.lines.length; i++) {
+ var line = this.lines[i];
+ if (line.isFree(colStart, colWidth)) {
+ line.addEvent(ev, colStart, colWidth, row, i);
+ return i;
+ }
+ }
+
+ var line = [];
+ line.isFree = function(colStart, colWidth) {
+ var free = true;
+
+ for (var i = 0; i < this.length; i++) {
+ if (!(colStart + colWidth - 1 < this[i].part.colStart || colStart > this[i].part.colStart + this[i].part.colWidth - 1)) {
+ free = false;
+ }
+ }
+
+ return free;
+ };
+
+ line.addEvent = function(ep, colStart, colWidth, row, index) {
+ //var eventPart = {};
+ //eventPart.event = ev;
+ ep.part.colStart = colStart;
+ ep.part.colWidth = colWidth;
+ ep.part.row = row;
+ ep.part.line = index;
+ ep.part.startsHere = thisRow.start.getTime() <= ep.start().getTime();
+ //if (confirm('r.start: ' + thisRow.start + ' ev.Start: ' + ev.Start)) thisRow = null;
+ ep.part.endsHere = thisRow.end.getTime() >= ep.end().getTime();
+
+ this.push(ep);
+ };
+
+ line.addEvent(ev, colStart, colWidth, row, this.lines.length);
+
+ this.lines.push(line);
+
+ return this.lines.length - 1;
+ };
+
+ r.getStart = function() {
+ var start = 0;
+ for (var i = 0; i < calendar.rows.length && i < this.index; i++) {
+ start += calendar.rows[i].getHeight();
+ }
+ };
+
+ r.getHeight = function() {
+ return Math.max(this.lines.length * calendar.lineHeight() + calendar.cellHeaderHeight + calendar.cellMarginBottom, this.calendar.cellHeight);
+ };
+
+ this.rows.push(r);
+ }
+
+ //this.endDate = DayPilot.Date.addDays(this.firstDate, rowCount * 7);
+ this.endDate = this.firstDate.addDays(rowCount * 7);
+ };
+
+ this.visibleStart = function() {
+ return calendar.firstDate;
+ };
+
+ this.visibleEnd = function() {
+ return calendar.endDate;
+ };
+
+ this.getHeight = function() {
+ var height = this.headerHeight;
+ for (var i = 0; i < this.rows.length; i++) {
+ height += this.rows[i].getHeight();
+ }
+ return height;
+ };
+
+ this.getWidth = function(start, end) {
+ var diff = (end.y * 7 + end.x) - (start.y * 7 + start.x);
+ return diff + 1;
+ };
+
+ this.getMinCoords = function(first, second) {
+ if ((first.y * 7 + first.x) < (second.y * 7 + second.x)) {
+ return first;
+ }
+ else {
+ return second;
+ }
+ };
+
+ this._prefixCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ if (prefix) {
+ return prefix + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this._drawTop = function() {
+ var relative = this.nav.top;
+ //this.nav.top = relative;
+ relative.setAttribute("unselectable", "on");
+ relative.style.MozUserSelect = 'none';
+ relative.style.KhtmlUserSelect = 'none';
+ relative.style.WebkitUserSelect = 'none';
+ relative.style.position = 'relative';
+ if (this.width) {
+ relative.style.width = this.width;
+ }
+ relative.style.height = this.getHeight() + 'px';
+ relative.onselectstart = function(e) { return false; }; // prevent text cursor in Chrome during drag&drop
+
+
+ if (this.hideUntilInit) {
+ relative.style.visibility = 'hidden';
+ }
+
+ if (!this.visible) {
+ relative.style.display = "none";
+ }
+
+ relative.className = this._prefixCssClass("_main");
+
+ var cells = document.createElement("div");
+ this.nav.cells = cells;
+ cells.style.position = "absolute";
+ cells.style.left = "0px";
+ cells.style.right = "0px";
+ cells.setAttribute("unselectable", "on");
+ relative.appendChild(cells);
+
+ var events = document.createElement("div");
+ this.nav.events = events;
+ events.style.position = "absolute";
+ events.style.left = "0px";
+ events.style.right = "0px";
+ events.setAttribute("unselectable", "on");
+ relative.appendChild(events);
+
+ relative.onmousemove = function(ev) {
+
+ if (DayPilotMonth.resizingEvent) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ if (!coords) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+ calendar.clearShadow();
+ var resizing = DayPilotMonth.resizingEvent;
+
+ var original = resizing.start;
+ var width, start;
+
+ if (resizing.direction === 'w-resize') {
+ start = cell;
+
+ var endDate = resizing.event.end();
+ //if (DayPilot.Date.getDate(endDate).getTime() === endDate.getTime()) {
+ if (endDate.getDatePart() === endDate) {
+ endDate = endDate.addDays(-1);
+ }
+
+ var end = calendar.getCellFromDate(endDate);
+ width = calendar.getWidth(cell, end);
+ }
+ else {
+ start = calendar.getCellFromDate(resizing.event.start());
+ width = calendar.getWidth(start, cell);
+ }
+
+ if (width < 1) {
+ width = 1;
+ }
+
+ calendar.drawShadow(start.x, start.y, 0, width);
+
+ }
+ else if (DayPilotMonth.movingEvent) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ if (!coords) {
+ return;
+ }
+
+ // not actually moved, Chrome bug
+ if (coords.x === DayPilotMonth.movingEvent.coords.x && coords.y === DayPilotMonth.movingEvent.coords.y) {
+ return;
+ }
+
+ var minDistance = 3;
+ var distance = Math.abs(coords.x - DayPilotMonth.movingEvent.coords.x) + Math.abs(coords.y - DayPilotMonth.movingEvent.coords.y);
+ if (distance <= minDistance) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+
+ calendar.clearShadow();
+
+ var event = DayPilotMonth.movingEvent.event;
+ var offset = DayPilotMonth.movingEvent.offset;
+ var width = calendar.cellMode ? 1 : DayPilot.DateUtil.daysSpan(event.start(), event.end()) + 1;
+
+ if (width < 1) {
+ width = 1;
+ }
+ calendar.drawShadow(cell.x, cell.y, 0, width, offset, event);
+ }
+ else if (DayPilotMonth.timeRangeSelecting) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ if (!coords) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+
+ calendar.clearShadow();
+
+ var start = DayPilotMonth.timeRangeSelecting;
+
+ var startIndex = start.y * 7 + start.x;
+ var cellIndex = cell.y * 7 + cell.x;
+
+ var width = Math.abs(cellIndex - startIndex) + 1;
+
+ if (width < 1) {
+ width = 1;
+ }
+
+ var shadowStart = startIndex < cellIndex ? start : cell;
+
+ DayPilotMonth.timeRangeSelecting.from = { x: shadowStart.x, y: shadowStart.y };
+ DayPilotMonth.timeRangeSelecting.width = width;
+ DayPilotMonth.timeRangeSelecting.moved = true;
+
+ calendar.drawShadow(shadowStart.x, shadowStart.y, 0, width, 0, null);
+
+ }
+
+ };
+
+ //this.nav.top.appendChild(this.vsph);
+ };
+
+ this._updateHeight = function() {
+ this.nav.top.style.height = this.getHeight() + 'px';
+
+ for (var x = 0; x < this.cells.length; x++) {
+ for (var y = 0; y < this.cells[x].length; y++) {
+ this.cells[x][y].style.top = this.getRowTop(y) + 'px';
+ this.cells[x][y].style.height = this.rows[y].getHeight() + 'px';
+ }
+ }
+ };
+
+ this.getCellBelowPoint = function(x, y) {
+ var columnWidth = Math.floor(this.nav.top.clientWidth / this.getColCount());
+ var column = Math.min(Math.floor(x / columnWidth), this.getColCount() - 1);
+
+ var row = null;
+
+ var height = this.headerHeight;
+ var relativeY = 0;
+ for (var i = 0; i < this.rows.length; i++) {
+ var baseHeight = height;
+ height += this.rows[i].getHeight();
+ if (y < height) {
+ relativeY = y - baseHeight;
+ row = i;
+ break;
+ }
+ }
+ if (row === null) {
+ row = this.rows.length - 1; // might be a pixel below the last line
+ }
+
+ var cell = {};
+ cell.x = column;
+ cell.y = row;
+ cell.relativeY = relativeY;
+
+ return cell;
+ };
+
+ this.getCellFromDate = function(date) {
+ var width = DayPilot.DateUtil.daysDiff(this.firstDate, date);
+ var cell = { x: 0, y: 0 };
+ while (width >= 7) {
+ cell.y++;
+ width -= 7;
+ }
+ cell.x = width;
+ return cell;
+ };
+
+ this._drawTable = function() {
+
+ var table = document.createElement("div");
+ table.oncontextmenu = function() { return false; };
+ this.nav.cells.appendChild(table);
+
+ this.cells = [];
+
+ for (var x = 0; x < this.getColCount(); x++) {
+
+ this.cells[x] = [];
+ var headerProperties = null;
+
+ var header = document.createElement("div");
+ header.setAttribute("unselectable", "on");
+ header.style.position = 'absolute';
+
+ header.style.left = (this.getCellWidth() * x) + '%';
+ header.style.width = (this.getCellWidth()) + '%';
+ header.style.top = '0px';
+ header.style.height = (this.headerHeight) + 'px';
+
+ var dayIndex = x + this.getWeekStart();
+ if (dayIndex > 6) {
+ dayIndex -= 7;
+ }
+
+ header.className = this._prefixCssClass("_header");
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.innerHTML = resolved.locale().dayNames[dayIndex];
+
+ header.appendChild(inner);
+
+ inner.style.position = "absolute";
+ inner.style.top = "0px";
+ inner.style.bottom = "0px";
+ inner.style.left = "0px";
+ inner.style.right = "0px";
+ inner.className = this._prefixCssClass("_header_inner");
+
+ inner.innerHTML = resolved.locale().dayNames[dayIndex];
+
+ table.appendChild(header);
+
+ for (var y = 0; y < this.rows.length; y++) {
+ this._drawCell(x, y, table);
+ }
+
+ }
+
+ };
+
+ this._clearTable = function() {
+
+ // clear event handlers
+ for (var x = 0; x < this.cells.length; x++) {
+ for (var y = 0; y < this.cells[x].length; y++) {
+ this.cells[x][y].onclick = null;
+ }
+ }
+
+ this.nav.cells.innerHTML = '';
+
+ };
+
+ this._api2 = function() {
+ return calendar.api === 2;
+ };
+ this._drawCell = function(x, y, table) {
+
+ var row = this.rows[y];
+ var d = this.firstDate.addDays(y * 7 + x);
+
+ var date = d.getDay();
+ var headerHtml = null;
+ if (date === 1) {
+ headerHtml = resolved.locale().monthNames[d.getMonth()] + ' ' + date;
+ }
+ else {
+ headerHtml = "" + date;
+ }
+ var business = !calendar.isWeekend(d);
+
+ var args = {};
+ args.control = calendar;
+ args.cell = {};
+ args.cell.start = d;
+ args.cell.end = d.addDays(1);
+ args.cell.properties = {
+ "headerHtml": headerHtml,
+ "backColor": null,
+ "business": business,
+ "html": null
+ };
+ if (typeof calendar.onBeforeCellRender === "function") {
+ calendar.onBeforeCellRender(args);
+ }
+
+ var props = args.cell.properties;
+
+ var cell = document.createElement("div");
+ cell.setAttribute("unselectable", "on");
+ cell.style.position = 'absolute';
+ cell.style.cursor = 'default';
+ cell.style.left = (this.getCellWidth() * x) + '%';
+ cell.style.width = (this.getCellWidth()) + '%';
+ cell.style.top = (this.getRowTop(y)) + 'px';
+ cell.style.height = (row.getHeight()) + 'px';
+ cell.className = this._prefixCssClass("_cell");
+
+ if (props.business) {
+ var business = this._prefixCssClass("_cell_business");
+ DayPilot.Util.addClass(cell, business);
+ }
+
+ var previousMonth = this.startDate.addMonths(-1).getMonth();
+ var nextMonth = this.startDate.addMonths(1).getMonth();
+
+ var thisMonth = this.startDate.getMonth();
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ cell.appendChild(inner);
+
+ inner.style.position = "absolute";
+ inner.style.left = "0px";
+ inner.style.right = "0px";
+ inner.style.top = "0px";
+ inner.style.bottom = "0px";
+ inner.className = this._prefixCssClass("_cell_inner");
+
+ if (props.backColor) {
+ inner.style.backgroundColor = args.cell.properties.backColor;
+ }
+
+ cell.onmousedown = function(e) {
+ if (calendar.timeRangeSelectedHandling !== 'Disabled') {
+ calendar.clearShadow();
+ DayPilotMonth.timeRangeSelecting = { "root": calendar, "x": x, "y": y, "from": { x: x, y: y }, "width": 1 };
+ }
+ };
+
+ cell.onclick = function() {
+
+ var single = function(d) {
+ var start = new DayPilot.Date(d);
+ var end = start.addDays(1);
+ calendar._timeRangeSelectedDispatch(start, end);
+ };
+
+ if (calendar.timeRangeSelectedHandling !== 'Disabled') {
+ single(d);
+ return;
+ }
+
+ };
+
+ var day = document.createElement("div");
+ day.setAttribute("unselectable", "on");
+ day.style.height = this.cellHeaderHeight + "px";
+ day.className = this._prefixCssClass("_cell_header");
+
+ day.onclick = function(ev) {
+
+ if (calendar.cellHeaderClickHandling !== "Enabled") {
+ return;
+ }
+
+ ev.stopPropagation();
+
+ var args = {};
+ args.control = calendar;
+ args.start = d;
+ args.end = d.addDays(1);
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onCellHeaderClick === "function") {
+ calendar.onCellHeaderClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ if (typeof calendar.onCellHeaderClicked === "function") {
+ calendar.onCellHeaderClicked(args);
+ }
+
+ };
+
+ day.innerHTML = props.headerHtml;
+
+ inner.appendChild(day);
+
+ if (props.html) {
+ var html = document.createElement("div");
+ html.style.height = (row.getHeight() - this.cellHeaderHeight) + 'px';
+ html.style.overflow = 'hidden';
+ html.innerHTML = props.html;
+ inner.appendChild(html);
+ }
+
+ this.cells[x][y] = cell;
+
+ table.appendChild(cell);
+ };
+
+ this.getWeekStart = function() {
+ if (calendar.weekStarts === 'Auto') {
+ var locale = resolved.locale();
+ if (locale) {
+ return locale.weekStarts;
+ }
+ else {
+ return 0; // Sunday
+ }
+ }
+ else {
+ return calendar.weekStarts || 0;
+ }
+ return resolved.locale().weekStarts;
+ };
+
+ this.getColCount = function() {
+ return 7;
+ };
+
+ this.getCellWidth = function() {
+ return 14.285;
+ };
+
+ this.getRowTop = function(index) {
+ var top = this.headerHeight;
+ for (var i = 0; i < index; i++) {
+ top += this.rows[i].getHeight();
+ }
+ return top;
+ };
+
+ this._callBack2 = function(action, data, parameters) {
+
+ var envelope = {};
+
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + DayPilot.JSON.stringify(envelope);
+
+ if (this.backendUrl) {
+ DayPilot.request(this.backendUrl, this._callBackResponse, commandstring, this.ajaxError);
+ }
+ };
+
+ this._callBackResponse = function(response) {
+ calendar._updateView(response.responseText);
+ };
+
+ this._getCallBackHeader = function() {
+ var h = {};
+ h.control = "dpm";
+ h.id = this.id;
+ h.v = this.v;
+
+ h.visibleStart = new DayPilot.Date(this.firstDate);
+ h.visibleEnd = h.visibleStart.addDays(this.days);
+
+ h.startDate = calendar.startDate;
+ // h.headerBackColor = this.headerBackColor;
+ // h.backColor = this.backColor;
+ // h.nonBusinessBackColor = this.nonBusinessBackColor;
+ h.timeFormat = this.timeFormat;
+ h.weekStarts = this.weekStarts;
+
+ return h;
+ };
+
+ this.eventClickCallBack = function(e, data) {
+ this._callBack2('EventClick', data, e);
+ };
+
+ this._eventClickDispatch = function(e) {
+
+ DayPilotMonth.movingEvent = null;
+ DayPilotMonth.resizingEvent = null;
+
+ var div = this;
+
+ var e = e || window.event;
+ var ctrlKey = e.ctrlKey;
+
+ e.cancelBubble = true;
+ if (e.stopPropagation) {
+ e.stopPropagation();
+ }
+
+ calendar.eventClickSingle(div, e);
+ };
+
+
+ this.eventClickSingle = function(div, ev) {
+ var e = div.event;
+ if (!e) {
+ return;
+ }
+ if (!e.client.clickEnabled()) {
+ return;
+ }
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.control = calendar;
+ args.div = div;
+ args.originalEvent = ev;
+ args.meta = ev.metaKey;
+ args.ctrl = ev.ctrlKey;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventClick === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClick(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventClickHandling) {
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(e);
+ }
+ }
+ break; }
+
+ if (typeof calendar.onEventClicked === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClicked(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventClickHandling) {
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventClick(e);
+ break;
+ }
+ }
+ };
+
+ this._onEventContextMenu = function() {
+ var e = this;
+ calendar._eventRightClickDispatch(e.event);
+ return false;
+ };
+
+ this._eventRightClickDispatch = function(e) {
+
+ this.event = e;
+
+ if (!e.client.rightClickEnabled()) {
+ return false;
+ }
+
+ var args = {};
+ args.e = e;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventRightClick === 'function') {
+ calendar.onEventRightClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventRightClickHandling) {
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(this.event);
+ }
+ }
+ break;
+ }
+
+ if (typeof calendar.onEventRightClicked === 'function') {
+ calendar.onEventRightClicked(args);
+ }
+
+
+ return false;
+ };
+
+
+ this._eventDeleteDispatch = function (e) {
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventDelete === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDelete(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventDeleteHandling) {
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'Update':
+ calendar.events.remove(e);
+ break;
+ }
+
+ if (typeof calendar.onEventDeleted === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDeleted(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventDeleteHandling) {
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventDelete(e);
+ break;
+ }
+ }
+
+ };
+
+ this.eventDeleteCallBack = function(e, data) {
+ this._callBack2('EventDelete', data, e);
+ };
+
+ this.eventDeletePostBack = function(e, data) {
+ this._postBack2('EventDelete', data, e);
+ };
+
+ this.eventMoveCallBack = function(e, newStart, newEnd, data, position) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+ params.position = position;
+
+ this._callBack2('EventMove', data, params);
+ };
+
+ this._eventMoveDispatch = function(e, x, y, offset, ev, position) {
+
+ var startOffset = e.start().getTimePart();
+
+ var endDate = e.end().getDatePart();
+ if (endDate.getTime() !== e.end().getTime()) {
+ endDate = endDate.addDays(1);
+ }
+ var endOffset = DayPilot.DateUtil.diff(e.end(), endDate);
+
+ var boxStart = this.getDateFromCell(x, y);
+ boxStart = boxStart.addDays(-offset);
+ var width = DayPilot.DateUtil.daysSpan(e.start(), e.end()) + 1;
+
+ var boxEnd = boxStart.addDays(width);
+
+ var newStart = boxStart.addTime(startOffset);
+ var newEnd = boxEnd.addTime(endOffset);
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.control = calendar;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.ctrl = ev.ctrlKey;
+ args.shift = ev.shiftKey;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventMove === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMove(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventMoveHandling) {
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventMoved === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMoved(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventMoveHandling) {
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd);
+ break;
+ case 'JavaScript':
+ calendar.onEventMove(e, newStart, newEnd);
+ break;
+ }
+ }
+ };
+
+ this.eventResizeCallBack = function(e, newStart, newEnd, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._callBack2('EventResize', data, params);
+ };
+
+ this._eventResizeDispatch = function(e, start, width) {
+ var startOffset = e.start().getTimePart();
+
+ var endDate = e.end().getDatePart();
+ if (endDate.getTime() !== e.end().getTime()) {
+ endDate = endDate.addDays(1);
+ }
+ var endOffset = DayPilot.DateUtil.diff(e.end(), endDate);
+
+ var boxStart = this.getDateFromCell(start.x, start.y);
+ //var width = DayPilot.Date.daysSpan(e.start(), e.end()) + 1;
+ var boxEnd = boxStart.addDays(width);
+
+ var newStart = boxStart.addTime(startOffset);
+ var newEnd = boxEnd.addTime(endOffset);
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.control = calendar;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventResize === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResize(args);
+ });
+
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventResizeHandling) {
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventResized === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResized(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventResizeHandling) {
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'JavaScript':
+ calendar.onEventResize(e, newStart, newEnd);
+ break;
+ }
+ }
+
+ };
+
+
+ this.timeRangeSelectedCallBack = function(start, end, data) {
+
+ var range = {};
+ range.start = start;
+ range.end = end;
+
+ this._callBack2('TimeRangeSelected', data, range);
+ };
+
+ this._timeRangeSelectedDispatch = function(start, end) {
+ if (this._api2()) {
+ var args = {};
+ args.control = calendar;
+ args.start = start;
+ args.end = end;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onTimeRangeSelect === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelect(args);
+ });
+
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ }
+
+ if (typeof calendar.onTimeRangeSelected === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelected(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ case 'JavaScript':
+ calendar.onTimeRangeSelected(start, end);
+ break;
+ }
+ }
+ };
+
+ this._angular = {};
+ this._angular.scope = null;
+ this._angular.notify = function() {
+ if (calendar._angular.scope) {
+ calendar._angular.scope["$apply"]();
+ }
+ };
+ this._angular.apply = function(f) {
+ f();
+
+ /*
+ if (calendar.angularAutoApply && calendar._angular.scope) {
+ calendar._angular.scope["$apply"](f);
+ }
+ else {
+ f();
+ }*/
+ };
+
+ this.clearSelection = function() {
+ calendar.clearShadow();
+ };
+
+ this.commandCallBack = function(command, data) {
+
+ var params = {};
+ params.command = command;
+
+ this._callBack2('Command', data, params);
+ };
+
+ this.isWeekend = function(date) {
+ date = new DayPilot.Date(date);
+
+ var sunday = 0;
+ var saturday = 6;
+
+ if (date.dayOfWeek() === sunday) {
+ return true;
+ }
+ if (date.dayOfWeek() === saturday) {
+ return true;
+ }
+ return false;
+ };
+
+ this._resolved = {};
+ this._resolved.locale = function() {
+ var found = DayPilot.Locale.find(calendar.locale);
+ if (!found) {
+ return DayPilot.Locale.US;
+ }
+ return found;
+ };
+
+ this._resolved._xssProtectionEnabled = function() {
+ return calendar.xssProtection !== "Disabled";
+ };
+
+
+ var resolved = this._resolved;
+
+ this.debug = function(msg, append) {
+ if (!this.debuggingEnabled) {
+ return;
+ }
+
+ if (!calendar.debugMessages) {
+ calendar.debugMessages = [];
+ }
+ calendar.debugMessages.push(msg);
+
+ if (typeof console !== 'undefined') {
+ console.log(msg);
+ }
+ };
+
+ this.dispose = function() {
+ var c = calendar;
+
+ if (c._disposed) {
+ return;
+ }
+ c._disposed = true;
+
+ c._deleteEvents();
+
+ c.nav.top.removeAttribute("style");
+ c.nav.top.removeAttribute("class");
+ c.nav.top.innerHTML = '';
+ c.nav.top.dp = null;
+
+ c.nav.top.onmousemove = null;
+ c.nav.top = null;
+
+ };
+
+ this.disposed = function() {
+ return this._disposed;
+ };
+
+ this._registerGlobalHandlers = function() {
+ if (!DayPilotMonth.globalHandlers) {
+ DayPilotMonth.globalHandlers = true;
+ DayPilot.re(document, 'mouseup', DayPilotMonth.gMouseUp);
+ }
+ };
+
+ this.loadFromServer = function() {
+ // make sure it has a place to ask
+ if (this.backendUrl || typeof WebForm_DoCallback === 'function') {
+ return (typeof calendar.events.list === 'undefined') || (!calendar.events.list);
+ }
+ else {
+ return false;
+ }
+
+ };
+
+ this._show = function() {
+ if (this.nav.top.style.visibility === 'hidden') {
+ this.nav.top.style.visibility = 'visible';
+ }
+ };
+
+ this.show = function() {
+ calendar.visible = true;
+ calendar.nav.top.style.display = '';
+ };
+
+ this.hide = function() {
+ calendar.visible = false;
+ calendar.nav.top.style.display = 'none';
+ };
+
+ this._loadTop = function() {
+ if (this.id && this.id.tagName) {
+ this.nav.top = this.id;
+ }
+ else if (typeof this.id === "string") {
+ this.nav.top = document.getElementById(this.id);
+ if (!this.nav.top) {
+ throw "DayPilot.Month: The placeholder element not found: '" + id + "'.";
+ }
+ }
+ else {
+ throw "DayPilot.Month() constructor requires the target element or its ID as a parameter";
+ }
+ };
+
+ this._initShort = function() {
+
+ this._prepareRows();
+ this._drawTop();
+ this._drawTable();
+ this._registerGlobalHandlers();
+ this._callBack2('Init'); // load events
+ };
+
+ this._xssTextHtml = function(text, html) {
+
+ if (calendar._resolved._xssProtectionEnabled()) {
+ return DayPilot.Util.escapeTextHtml(text, html);
+ }
+
+ if (!DayPilot.Util.isNullOrUndefined(html)) {
+ return html;
+ }
+ if (DayPilot.Util.isNullOrUndefined(text)) {
+ return "";
+ }
+ return text;
+ };
+
+ this.internal = {};
+ this.internal.loadOptions = this._loadOptions;
+ this.internal.xssTextHtml = calendar._xssTextHtml;
+
+ this.init = function() {
+ this._loadTop();
+
+ var loadFromServer = this.loadFromServer();
+
+ if (loadFromServer) {
+ this._initShort();
+ return;
+ }
+
+ this._prepareRows();
+ this._loadEvents();
+ this._drawTop();
+ this._drawTable();
+ this._show();
+ this._drawEvents();
+
+ this._registerGlobalHandlers();
+
+ this.fireAfterRenderDetached(null, false);
+
+ this._initialized = true;
+
+ return this;
+ };
+
+ this.Init = this.init;
+
+ // API compatibility (common)
+ Object.defineProperty(this, 'durationBarVisible', { get: function() { return calendar.eventBarVisible; } });
+
+ this._loadOptions(options);
+ };
+
+ DayPilotMonth.gMouseUp = function(ev) {
+
+ if (DayPilotMonth.movingEvent) {
+ var src = DayPilotMonth.movingEvent;
+
+ if (!src.event) {
+ return;
+ }
+ if (!src.event.calendar) {
+ return;
+ }
+ if (!src.event.calendar.shadow) {
+ return;
+ }
+ if (!src.event.calendar.shadow.start) {
+ return;
+ }
+
+ // load ref
+ var calendar = DayPilotMonth.movingEvent.event.calendar;
+ var e = DayPilotMonth.movingEvent.event;
+ var start = calendar.shadow.start;
+ var position = calendar.shadow.position;
+ var offset = DayPilotMonth.movingEvent.offset;
+
+ // cleanup
+ calendar.clearShadow();
+ DayPilotMonth.movingEvent = null;
+
+ var ev = ev || window.event;
+
+ // fire the event
+ calendar._eventMoveDispatch(e, start.x, start.y, offset, ev, position);
+
+ ev.cancelBubble = true;
+ if (ev.stopPropagation) {
+ ev.stopPropagation();
+ }
+ DayPilotMonth.movingEvent = null;
+ return false;
+ }
+ else if (DayPilotMonth.resizingEvent) {
+ var src = DayPilotMonth.resizingEvent;
+
+ if (!src.event) {
+ return;
+ }
+ if (!src.event.calendar) {
+ return;
+ }
+ if (!src.event.calendar.shadow) {
+ return;
+ }
+ if (!src.event.calendar.shadow.start) {
+ return;
+ }
+
+ // load ref
+ var calendar = DayPilotMonth.resizingEvent.event.calendar;
+
+ var e = DayPilotMonth.resizingEvent.event;
+ var start = calendar.shadow.start;
+ var width = calendar.shadow.width;
+
+ // cleanup
+ calendar.clearShadow();
+ DayPilotMonth.resizingEvent = null;
+
+ // fire the event
+ calendar._eventResizeDispatch(e, start, width);
+
+ ev.cancelBubble = true;
+ DayPilotMonth.resizingEvent = null;
+ return false;
+ }
+ else if (DayPilotMonth.timeRangeSelecting) {
+ if (DayPilotMonth.timeRangeSelecting.moved) {
+ var sel = DayPilotMonth.timeRangeSelecting;
+ var calendar = sel.root;
+
+ var start = new DayPilot.Date(calendar.getDateFromCell(sel.from.x, sel.from.y));
+ var end = start.addDays(sel.width);
+ calendar._timeRangeSelectedDispatch(start, end);
+
+ calendar.clearShadow();
+ }
+ DayPilotMonth.timeRangeSelecting = null;
+ }
+ };
+
+ // publish the API
+
+ // current
+ DayPilot.Month = DayPilotMonth.Month;
+
+ // experimental jQuery bindings
+ if (typeof jQuery !== 'undefined') {
+ (function($) {
+ $.fn.daypilotMonth = function(options) {
+ var first = null;
+ var j = this.each(function() {
+ if (this.daypilot) { // already initialized
+ return;
+ };
+
+ var daypilot = new DayPilot.Month(this.id);
+ this.daypilot = daypilot;
+ for (name in options) {
+ daypilot[name] = options[name];
+ }
+ daypilot.Init();
+ if (!first) {
+ first = daypilot;
+ }
+ });
+ if (this.length === 1) {
+ return first;
+ }
+ else {
+ return j;
+ }
+ };
+ })(jQuery);
+ }
+
+ (function registerAngularModule() {
+
+ var app = DayPilot.am();
+
+ if (!app) {
+ return;
+ }
+
+ app.directive("daypilotMonth", ['$parse', function($parse) {
+ return {
+ "restrict": "E",
+ "template": "",
+ "replace": true,
+ "link": function (scope, element, attrs) {
+
+ var calendar = new DayPilot.Month(element[0]);
+ calendar._angular.scope = scope;
+ calendar.init();
+
+ var oattr = attrs["id"];
+ if (oattr) {
+ scope[oattr] = calendar;
+ }
+
+ // save DayPilot.Calendar object in the specified variable
+ var pas = attrs["publishAs"];
+ if (pas) {
+ var getter = $parse(pas);
+ var setter = getter.assign;
+ setter(scope, calendar);
+ }
+
+ // bind event handlers from attributes starting with "on"
+ for (var name in attrs) {
+ if (name.indexOf("on") === 0) { // event handler
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ scope["$apply"](function() {
+ f(scope, {"args": args});
+ });
+ };
+ })(name);
+ }
+ }
+
+ var watch = scope["$watch"];
+ var config = attrs["config"] || attrs["daypilotConfig"];
+ var events = attrs["events"] || attrs["daypilotEvents"];
+
+ //var watch = scope["$watch"];
+
+ watch.call(scope, config, function (value) {
+ for (var name in value) {
+ calendar[name] = value[name];
+ }
+ calendar.update();
+ }, true);
+
+ watch.call(scope, events, function(value) {
+ calendar.events.list = value;
+ calendar.update();
+ }, true);
+
+ }
+ };
+ }]);
+ })();
+
+
+ if (typeof Sys !== 'undefined' && Sys.Application && Sys.Application.notifyScriptLoaded) {
+ Sys.Application.notifyScriptLoaded();
+ }
+
+
+})();
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function() {
+
+ if (typeof DayPilot.Navigator !== 'undefined' && DayPilot.Navigator.nav) {
+ return;
+ }
+
+ var DayPilotNavigator = {};
+ DayPilot.Navigator = function(id, options) {
+ this.v = '2024.3.539-lite';
+ var calendar = this;
+ this.id = id;
+ this.api = 2;
+ this.isNavigator = true;
+
+ this.autoFocusOnClick = true;
+ this.weekStarts = 'Auto'; // 0 = Sunday, 1 = Monday, ... 'Auto' = according to locale
+ this.selectMode = 'Day'; // day/week/month/none
+ this.titleHeight = 30;
+ this.dayHeaderHeight = 30;
+ this.bound = null;
+ this.cellWidth = 30;
+ this.cellHeight = 30;
+ this.cssClassPrefix = "navigator_default";
+ this.freeHandSelectionEnabled = false;
+ this.selectionStart = new DayPilot.Date().getDatePart(); // today
+ this.selectionEnd = null;
+ this.selectionDay = null;
+ this.showMonths = 1;
+ this.skipMonths = 1;
+ this.command = "navigate";
+ this.year = new DayPilot.Date().getYear();
+ this.month = new DayPilot.Date().getMonth() + 1;
+ this.showWeekNumbers = false;
+ this.weekNumberAlgorithm = 'Auto';
+ this.rowsPerMonth = 'Six'; // Six, Auto
+ this.orientation = "Vertical";
+ this.locale = "en-us";
+ this.rtl = false;
+ this.visible = true;
+
+ this.timeRangeSelectedHandling = "Bind";
+ this.visibleRangeChangedHandling = "Enabled";
+
+ this.onVisibleRangeChange = null;
+ this.onVisibleRangeChanged = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+
+ this.nav = {};
+
+ this._cache = {};
+
+ this._prepare = function() {
+
+ this.root.dp = this;
+
+ this.root.className = this._prefixCssClass('_main');
+
+ if (this.orientation === "Horizontal") {
+ this.root.style.width = this.showMonths * (resolved.cellWidth() * 7 + this._weekNumberWidth()) + 'px';
+ this.root.style.height = (this.cellHeight*6 + this.titleHeight + this.dayHeaderHeight) + 'px';
+ }
+ else {
+ this.root.style.width = (resolved.cellWidth() * 7 + this._weekNumberWidth()) + 'px';
+ }
+ //this.root.style.height = (this.showMonths*(this.cellHeight*6 + this.titleHeight + this.dayHeaderHeight)) + 'px';
+
+ if (this.rtl) {
+ this.root.style.direction = "rtl";
+ }
+
+ this.root.style.position = "relative";
+
+ if (!this.visible) {
+ this.root.style.display = "none";
+ }
+
+ var vsph = document.createElement("input");
+ vsph.type = 'hidden';
+ vsph.name = calendar.id + "_state";
+ vsph.id = vsph.name;
+ //vsph.value = result.VsUpdate;
+ this.root.appendChild(vsph);
+ this.state = vsph;
+
+ if (!this.startDate) {
+ this.startDate = DayPilot.Date.fromYearMonthDay(this.year, this.month);
+ }
+ else { // make sure it's the first day
+ this.startDate = new DayPilot.Date(this.startDate).firstDayOfMonth();
+ }
+
+ this.calendars = [];
+ this.selected = [];
+ this.months = [];
+ };
+
+ this._api2 = function() {
+ return calendar.api === 2;
+ };
+
+ this._clearTable = function() {
+ // TODO do something smarter here
+ this.root.innerHTML = '';
+ };
+
+ this._prefixCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ if (prefix) {
+ return prefix + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this._addClass = function(object, name) {
+ var fullName = this._prefixCssClass("_" + name);
+ DayPilot.Util.addClass(object, fullName);
+ };
+
+ this._removeClass = function(object, name) {
+ var fullName = this._prefixCssClass("_" + name);
+ DayPilot.Util.removeClass(object, fullName);
+ };
+
+ this._drawTable = function(j, showLinks) {
+ var month = {};
+ month.cells = [];
+ month.days = [];
+ month.weeks = [];
+
+ var startDate = this.startDate.addMonths(j);
+
+ var showBefore = showLinks.before;
+ var showAfter = showLinks.after;
+
+ var firstOfMonth = startDate.firstDayOfMonth();
+ var first = firstOfMonth.firstDayOfWeek(resolved.weekStarts());
+
+ var last = firstOfMonth.addMonths(1);
+ var days = DayPilot.DateUtil.daysDiff(first, last);
+
+ var rowCount = (this.rowsPerMonth === "Auto") ? Math.ceil(days / 7) : 6;
+ month.rowCount = rowCount;
+ var today = (new DayPilot.Date()).getDatePart();
+
+ var width = resolved.cellWidth() * 7 + this._weekNumberWidth();
+ month.width = width;
+ var height = this.cellHeight * rowCount + this.titleHeight + this.dayHeaderHeight;
+ month.height = height;
+
+ var main = document.createElement("div");
+ main.style.width = (width) + 'px';
+ main.style.height = (height) + 'px';
+
+
+ if (this.orientation === "Horizontal") {
+ main.style.position = "absolute";
+ main.style.left = (width * j) + "px";
+ main.style.top = "0px";
+
+ month.top = 0;
+ month.left = width * j;
+ }
+ else {
+ main.style.position = 'relative';
+
+ var above = j > 0 ? calendar.months[j - 1].top + calendar.months[j - 1].height : 0;
+ month.top = above;
+
+ month.left = 0;
+ }
+
+ main.className = this._prefixCssClass('_month');
+ main.style.cursor = 'default';
+ main.style.MozUserSelect = 'none';
+ main.style.KhtmlUserSelect = 'none';
+ main.style.WebkitUserSelect = 'none';
+
+ main.month = month;
+
+ this.root.appendChild(main);
+
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+
+ // title left
+ var tl = document.createElement("div");
+ tl.style.position = 'absolute';
+ tl.style.left = '0px';
+ tl.style.right = '0px'; // rtl
+ tl.style.top = '0px';
+ tl.style.width = resolved.cellWidth() + 'px';
+ tl.style.height = this.titleHeight + 'px';
+ tl.style.lineHeight = this.titleHeight + 'px';
+ //tl.style.textAlign = 'left';
+ tl.setAttribute("unselectable", "on");
+ tl.className = this._prefixCssClass('_titleleft');
+ if (showLinks.left) {
+ tl.style.cursor = 'pointer';
+ tl.innerHTML = "<";
+ tl.onclick = this._clickLeft;
+ }
+ main.appendChild(tl);
+ this.tl = tl;
+
+ // title center
+ var ti = document.createElement("div");
+ ti.style.position = 'absolute';
+ ti.style.left = resolved.cellWidth() + 'px';
+ ti.style.top = '0px';
+ ti.style.width = (resolved.cellWidth() * 5 + this._weekNumberWidth()) + 'px';
+ ti.style.height = this.titleHeight + 'px';
+ ti.style.lineHeight = this.titleHeight + 'px';
+ //ti.style.textAlign = 'center';
+ ti.setAttribute("unselectable", "on");
+ ti.className = this._prefixCssClass('_title');
+ ti.innerHTML = resolved.locale().monthNames[startDate.getMonth()] + ' ' + startDate.getYear();
+ main.appendChild(ti);
+ this.ti = ti;
+
+ // title right
+ var tr = document.createElement("div");
+ tr.style.position = 'absolute';
+ tr.style.left = (resolved.cellWidth() * 6 + this._weekNumberWidth()) + 'px';
+ tr.style.right = (resolved.cellWidth() * 6 + this._weekNumberWidth()) + 'px'; // rtl
+ tr.style.top = '0px';
+ tr.style.width = resolved.cellWidth() + 'px';
+ tr.style.height = this.titleHeight + 'px';
+ tr.style.lineHeight = this.titleHeight + 'px';
+ //tr.style.textAlign = 'right';
+ tr.setAttribute("unselectable", "on");
+ tr.className = this._prefixCssClass('_titleright');
+ if (showLinks.right) {
+ tr.style.cursor = 'pointer';
+ tr.innerHTML = ">";
+ tr.onclick = this._clickRight;
+ }
+ main.appendChild(tr);
+ this.tr = tr;
+
+
+ var xOffset = this._weekNumberWidth();
+ if (this.showWeekNumbers) {
+ for (var y = 0; y < rowCount; y++) {
+ var day = first.addDays(y * 7);
+ var weekNumber = null;
+ switch (this.weekNumberAlgorithm) {
+ case "Auto":
+ weekNumber = (resolved.weekStarts() === 1) ? day.weekNumberISO() : day.weekNumber();
+ break;
+ case "US":
+ weekNumber = day.weekNumber();
+ break;
+ case "ISO8601":
+ weekNumber = day.weekNumberISO();
+ break;
+ default:
+ throw "Unknown weekNumberAlgorithm value.";
+ }
+
+ var dh = document.createElement("div");
+ dh.style.position = 'absolute';
+ dh.style.left = (0) + 'px';
+ dh.style.right = (0) + 'px';
+ dh.style.top = (y * this.cellHeight + totalHeaderHeight) + 'px';
+ dh.style.width = resolved.cellWidth() + 'px';
+ dh.style.height = this.cellHeight + 'px';
+ dh.style.lineHeight = this.cellHeight + 'px';
+ //dh.style.textAlign = 'right';
+ dh.setAttribute("unselectable", "on");
+ dh.className = this._prefixCssClass('_weeknumber');
+ //dh.innerHTML = "" + weekNumber + "";
+ dh.innerHTML = "" + weekNumber + "";
+ main.appendChild(dh);
+ month.weeks.push(dh);
+ }
+ }
+
+
+ for (var x = 0; x < 7; x++) {
+ month.cells[x] = [];
+
+ // day header
+ var dh = document.createElement("div");
+ dh.style.position = 'absolute';
+ dh.style.left = (x * resolved.cellWidth() + xOffset) + 'px';
+ dh.style.right = (x * resolved.cellWidth() + xOffset) + 'px'; // rtl
+ dh.style.top = this.titleHeight + 'px';
+ dh.style.width = resolved.cellWidth() + 'px';
+ dh.style.height = this.dayHeaderHeight + 'px';
+ dh.style.lineHeight = this.dayHeaderHeight + 'px';
+ //dh.style.textAlign = 'right';
+ dh.setAttribute("unselectable", "on");
+ dh.className = this._prefixCssClass('_dayheader');
+ dh.innerHTML = "" + this._getDayName(x) + "";
+ main.appendChild(dh);
+ month.days.push(dh);
+
+ for (var y = 0; y < rowCount; y++) {
+ var day = first.addDays(y * 7 + x);
+
+ var isSelected = this._isSelected(day) && this._selectModeLowerCase() !== 'none';
+ var isCurrentMonth = day.firstDayOfMonth() === startDate;
+ var isPrevMonth = day < startDate;
+ var isNextMonth = day >= startDate.addMonths(1);
+
+ if (this._selectModeLowerCase() === "month") {
+ isSelected = isSelected && isCurrentMonth;
+ }
+ else if (this._selectModeLowerCase() === "day") {
+ isSelected = isSelected && (isCurrentMonth || (showBefore && isPrevMonth) || (showAfter && isNextMonth));
+ }
+ else if (this._selectModeLowerCase() === "week") {
+ //var sd = this.selectionDay || this.selectionStart;
+ var isSelectionCurrentMonth = day.firstDayOfMonth() === startDate;
+ isSelected = isSelected && (isSelectionCurrentMonth || (showBefore && isPrevMonth) || (showAfter && isNextMonth));
+ }
+
+ var dc = document.createElement("div");
+ month.cells[x][y] = dc;
+
+ var cellPos = calendar._cellRelativeCoords(x, y);
+ var left = cellPos.x;
+ var top = cellPos.y;
+
+ dc.day = day;
+ dc.x = x;
+ dc.y = y;
+ dc.left = left;
+ dc.top = top;
+ dc.isCurrentMonth = isCurrentMonth;
+ dc.isNextMonth = isNextMonth;
+ dc.isPrevMonth = isPrevMonth;
+ dc.showBefore = showBefore;
+ dc.showAfter = showAfter;
+ dc.className = this._prefixCssClass((isCurrentMonth ? '_day' : '_dayother'));
+ calendar._addClass(dc, "cell");
+ if (day.getTime() === today.getTime() && isCurrentMonth) {
+ this._addClass(dc, 'today');
+ }
+ if (day.dayOfWeek() === 0 || day.dayOfWeek() === 6) {
+ this._addClass(dc, 'weekend');
+ }
+
+ dc.style.position = 'absolute';
+ dc.style.left = (left) + 'px';
+ dc.style.right = (left) + 'px'; // rtl
+ dc.style.top = (top) + 'px';
+ dc.style.width = resolved.cellWidth() + 'px';
+ dc.style.height = this.cellHeight + 'px';
+ dc.style.lineHeight = this.cellHeight + 'px'; // vertical alignment
+ //dc.style.textAlign = 'right';
+ //dc.style.border = '1px solid white';
+
+ var inner = document.createElement("div");
+ inner.style.position = 'absolute';
+ inner.className = (day.getTime() === today.getTime() && isCurrentMonth) ? this._prefixCssClass('_todaybox') : this._prefixCssClass('_daybox');
+ calendar._addClass(inner, "cell_box");
+ //inner.style.boxSizing = "border-box";
+ inner.style.left = '0px';
+ inner.style.top = '0px';
+ inner.style.right = '0px';
+ inner.style.bottom = '0px';
+ //inner.style.width = (this.cellWidth - 2) + 'px';
+ //inner.style.height = (this.cellHeight - 2) + 'px';
+ dc.appendChild(inner);
+
+ /*
+ if (isCurrentMonth) {
+ dc.style.cursor = 'pointer';
+ }
+ */
+
+ var cell = null;
+ if (this.cells && this.cells[day.toStringSortable()]) {
+ cell = this.cells[day.toStringSortable()];
+ }
+
+ if (typeof calendar.onBeforeCellRender === "function") {
+ var args = {};
+ args.cell = cell || {};
+ args.cell.day = day;
+ args.cell.isCurrentMonth = isCurrentMonth;
+ args.cell.isToday = day.getTime() === today.getTime() && isCurrentMonth;
+ args.cell.isWeekend = day.dayOfWeek() === 0 || day.dayOfWeek() === 6;
+ if (cell) {
+ args.cell.html = cell.html || day.getDay();
+ args.cell.cssClass = cell.css;
+ }
+ else {
+ args.cell.html = day.getDay();
+ args.cell.cssClass = null;
+ }
+
+ calendar.onBeforeCellRender(args);
+
+ cell = args.cell;
+ }
+
+ if (cell) {
+ DayPilot.Util.addClass(dc, cell.cssClass || cell.css);
+ }
+
+ //var span = null;
+ if (isCurrentMonth || (showBefore && isPrevMonth) || (showAfter && isNextMonth)) {
+ var text = document.createElement("div");
+ text.innerHTML = day.getDay();
+ text.style.position = "absolute";
+ text.style.left = '0px';
+ text.style.top = '0px';
+ text.style.right = '0px';
+ text.style.bottom = '0px';
+ calendar._addClass(text, "cell_text");
+
+ // dc.style.cursor = 'pointer';
+ dc.isClickable = true;
+
+ if (cell && cell.html) {
+ text.innerHTML = cell.html;
+ }
+
+ dc.appendChild(text);
+ }
+
+
+ dc.setAttribute("unselectable", "on");
+
+ dc.onclick = this._cellClick;
+
+ main.appendChild(dc);
+
+ if (isSelected) {
+ calendar._cellSelect(main, x, y);
+ this.selected.push(dc);
+ }
+
+ }
+ }
+
+ var line = document.createElement("div");
+ line.style.position = 'absolute';
+ line.style.left = '0px';
+ line.style.top = (totalHeaderHeight - 2) + 'px';
+ line.style.width = (resolved.cellWidth() * 7 + this._weekNumberWidth()) + 'px';
+ line.style.height = '1px';
+ line.style.fontSize = '1px';
+ line.style.lineHeight = '1px';
+ line.className = this._prefixCssClass("_line");
+
+ main.appendChild(line);
+ this.months.push(month);
+ };
+
+ this._cellRelativeCoords = function(x, y) {
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+ var xOffset = this._weekNumberWidth();
+ var left = x * resolved.cellWidth() + xOffset;
+ var top = y * this.cellHeight + totalHeaderHeight;
+
+ return {
+ "x": left,
+ "y": top
+ };
+
+ };
+
+ this._cellSelect = function(main, x, y) {
+ var div = main.month.cells[x][y];
+
+ calendar._addClass(div, 'select');
+
+ /*
+ if (div.selectDiv) {
+ return;
+ }
+
+ var xOffset = this._weekNumberWidth();
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+
+ // overlay select
+ var od = document.createElement("div");
+ od.style.position = 'absolute';
+ od.style.left = (x * this.cellWidth + xOffset) + 'px';
+ od.style.top = (y * this.cellHeight + totalHeaderHeight) + 'px';
+ od.style.width = this.cellWidth + 'px';
+ od.style.height = this.cellHeight + 'px';
+ od.className = calendar._prefixCssClass("_selected_overlay");
+
+ div.selectDiv = od;
+
+ main.appendChild(od);
+ */
+ };
+
+ this._cellUnselect = function(main, x, y) {
+ var div = main.month.cells[x][y];
+
+ calendar._removeClass(div, 'select');
+
+ /*
+ DayPilot.de(div.selectDiv);
+ div.selectDiv = null;
+ */
+ };
+
+ this._weekNumberWidth = function() {
+ if (this.showWeekNumbers) {
+ return resolved.cellWidth();
+ }
+ return 0;
+ };
+
+ this._updateFreeBusy = function() {
+ if (!this.items) {
+ return;
+ }
+
+ for (var j = 0; j < this.showMonths; j++) {
+ for (var x = 0; x < 7; x++) {
+ for (var y = 0; y < 6; y++) {
+ var cell = this.months[j].cells[x][y];
+ if (!cell) {
+ continue;
+ }
+ if (this.items[cell.day.toStringSortable()] === 1) {
+ this._addClass(cell, 'busy');
+ this._removeClass(cell, 'free');
+ }
+ else {
+ this._removeClass(cell, 'busy');
+ this._addClass(cell, 'free');
+ }
+ }
+ }
+ }
+ };
+
+ this._saveState = function() {
+ var s = {};
+ s.startDate = calendar.startDate;
+ s.selectionStart = calendar.selectionStart;
+ s.selectionEnd = calendar.selectionEnd.addDays(1);
+ calendar.state.value = JSON.stringify(s);
+ };
+
+ this._selectModeLowerCase = function() {
+ var selectMode = this.selectMode || "";
+ return selectMode.toLowerCase();
+ };
+
+ this._adjustSelection = function() {
+
+ var input = this.selectionDay || this.selectionStart; // selectionDay is preferred
+ if (!input) {
+ input = DayPilot.Date.today();
+ }
+ input = new DayPilot.Date(input); // make sure it's DayPilot.Date
+
+ // ignores selectionEnd
+ // uses selectMode
+ switch (this._selectModeLowerCase()) {
+ case 'day':
+ this.selectionStart = input;
+ this.selectionDay = input;
+ this.selectionEnd = input;
+ break;
+ case 'week':
+ this.selectionDay = input;
+ this.selectionStart = input.firstDayOfWeek(resolved.weekStarts());
+ this.selectionEnd = this.selectionStart.addDays(6);
+ break;
+ case 'month':
+ this.selectionDay = input;
+ this.selectionStart = input.firstDayOfMonth();
+ this.selectionEnd = this.selectionStart.lastDayOfMonth();
+ break;
+ case 'none':
+ this.selectionEnd = input;
+ break;
+ default:
+ throw "Unknown selectMode value.";
+ }
+
+ };
+
+ this._postponedSelect = null;
+
+ // options.dontFocus, options.dontNotify
+ this.select = function(a1, a2, a3) {
+
+ var a2IsDate = a2 && (a2 instanceof DayPilot.Date || typeof a2 === "string");
+ var a2IsOptions = (a2 && typeof a2 === "object") || typeof a2 === "boolean";
+
+ var date1 = a1;
+ var date2 = a2IsDate ? a2 : null;
+ var options = a2IsOptions ? a2 : a3;
+
+ if (!this._initialized) {
+ this._postponedSelect = {
+ "date1": date1,
+ "date2": date2,
+ "options": options
+ };
+ return;
+ }
+
+ var focus = true;
+ var notify = true; // fire the timeRangeSelected event
+
+ if (options && typeof options === "object") {
+ if (options.dontFocus) {
+ focus = false;
+ }
+ if (options.dontNotify) {
+ notify = false;
+ }
+ }
+ else if (typeof options === "boolean") {
+ focus = !options;
+ }
+
+ var originalStart = this.selectionStart;
+ var originalEnd = this.selectionEnd;
+
+ this.selectionStart = new DayPilot.Date(date1).getDatePart();
+ this.selectionDay = this.selectionStart;
+
+ var startChanged = false;
+ if (focus) {
+
+ var newStart = this.startDate;
+ if (this.selectionStart < this._activeStart() || this.selectionStart >= this._activeEnd()) {
+ newStart = this.selectionStart.firstDayOfMonth();
+ }
+
+ if (newStart.toStringSortable() !== this.startDate.toStringSortable()) {
+ startChanged = true;
+ }
+
+ this.startDate = newStart;
+ }
+
+ if (date2 && calendar.freeHandSelectionEnabled) {
+ calendar.selectionEnd = new DayPilot.Date(date2);
+ }
+ else {
+ this._adjustSelection();
+ }
+
+ // redraw
+ this._clearTable();
+ this._prepare();
+ this._drawMonths();
+ this._updateFreeBusy();
+ this._saveState();
+
+ if (notify && (!originalStart.equals(this.selectionStart) || !originalEnd.equals(this.selectionEnd))) {
+ //alert('time range');
+ this._timeRangeSelectedDispatch();
+ }
+
+ if (startChanged) {
+ //alert('visible range');
+ this._visibleRangeChangedDispatch();
+ }
+ };
+
+ this.update = function(options) {
+
+ calendar._loadOptions(options);
+
+/*
+ if (!calendar._initialized) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Navigator instance that hasn't been initialized yet.");
+ }
+*/
+ if (!this._initialized) {
+ return;
+ }
+
+ if (calendar._disposed) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Navigator instance that has been disposed.");
+ }
+
+ calendar._clearCache();
+
+ var os = {
+ "day": calendar.selectionDay,
+ "start": calendar.selectionStart,
+ "end": calendar.selectionEnd
+ };
+
+ calendar._update();
+
+ if (os.start !== calendar.selectionStart || os.end != calendar.selectionEnd || os.day !== calendar.selectionDay) {
+ calendar._timeRangeSelectedDispatch();
+ }
+ };
+
+ this._update = function () {
+ // redraw
+ this._clearTable();
+ this._prepare();
+ this._adjustSelection();
+ this._drawMonths();
+ this._loadEvents();
+ this._updateFreeBusy();
+ this._saveState();
+
+ if (this.visible) {
+ this.show();
+ }
+ else {
+ this.hide();
+ }
+ };
+
+ this._clearCache = function() {
+ calendar._cache = {};
+ };
+
+ this._specialHandling = null;
+ this._loadOptions = function(options) {
+ if (!options) {
+ return;
+ }
+ var specialHandling = {
+ "events": {
+ "preInit": function() {
+ var events = this.data || [];
+ if (DayPilot.isArray(events.list)) {
+ calendar.events.list = events.list;
+ }
+ else {
+ calendar.events.list = events;
+ }
+ }
+ }
+ };
+ this._specialHandling = specialHandling;
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ var item = specialHandling[name];
+ item.data = options[name];
+ if (item.preInit) {
+ item.preInit();
+ }
+ }
+ else {
+ calendar[name] = options[name];
+ }
+ }
+
+ };
+
+ this._postInit = function() {
+ var specialHandling = this._specialHandling;
+ for (var name in specialHandling) {
+ var item = specialHandling[name];
+ if (item.postInit) {
+ item.postInit();
+ }
+ }
+ };
+
+
+ this._callBack2 = function(action, data, parameters) {
+ var envelope = {};
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + JSON.stringify(envelope);
+
+ var context = null;
+ if (this.backendUrl) {
+ DayPilot.request(this.backendUrl, this._callBackResponse, commandstring, this._ajaxError);
+ }
+ else {
+ WebForm_DoCallback(this.uniqueID, commandstring, this._updateView, context, this.callbackError, true);
+ }
+
+ };
+
+ this._ajaxError = function(req) {
+ if (typeof calendar.onAjaxError === 'function') {
+ var args = {};
+ args.request = req;
+ calendar.onAjaxError(args);
+ }
+ else if (typeof calendar.ajaxError === 'function') { // backwards compatibility
+ calendar.ajaxError(req);
+ }
+ };
+
+ this._callBackResponse = function(response) {
+ calendar._updateView(response.responseText);
+ };
+
+ this._postBack2 = function(action, data, parameters) {
+ var envelope = {};
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + JSON.stringify(envelope);
+ __doPostBack(calendar.uniqueID, commandstring);
+ };
+
+ this._getCallBackHeader = function() {
+ var h = {};
+ h.v = this.v;
+ h.startDate = this.startDate;
+ h.selectionStart = this.selectionStart;
+ h.showMonths = this.showMonths;
+ return h;
+ };
+
+ this._listen = function(action, data) {
+ if (action === 'refresh') {
+ this._visibleRangeChangedDispatch();
+ }
+ };
+
+ this._getDayName = function(i) {
+ var x = i + resolved.weekStarts();
+ if (x > 6) {
+ x -= 7;
+ }
+ return resolved.locale().dayNamesShort[x];
+
+ };
+
+ this._isSelected = function(date) {
+ if (this.selectionStart === null || this.selectionEnd === null) {
+ return false;
+ }
+
+ if (this.selectionStart.getTime() <= date.getTime() && date.getTime() <= this.selectionEnd.getTime()) {
+ return true;
+ }
+
+ return false;
+ };
+
+ this._getMonthFromCoords = function(coords) {
+ var month = 0;
+ //debugger;
+ for (var i = 0; i < calendar.months.length; i++) {
+ var m = calendar.months[i];
+ if (!m) {
+ return null;
+ }
+ if (coords.x < m.left || m.width < coords.x ) {
+ return null;
+ }
+
+ var height = calendar.months[i].height;
+ if (m.top <= coords.y && coords.y < m.top + m.height) {
+ return i;
+ }
+ }
+ return null;
+
+ };
+
+ this._getPosition = function(ev) {
+
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ var monthIndex = calendar._getMonthFromCoords(coords);
+ if (monthIndex === null) {
+ return null;
+ }
+ var month = calendar.months[monthIndex];
+
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+ if (month.top <= coords.y && coords.y < month.top + totalHeaderHeight) {
+ return {
+ "month": monthIndex,
+ "x": 0,
+ "y": 0,
+ "coords": coords,
+ "header": true
+ };
+ }
+
+ for (var x = 0; x < month.cells.length; x++) {
+ for (var y = 0; y < month.cells[x].length; y++) {
+ var cell = month.cells[x][y];
+ var top = cell.top + month.top;
+ var left = cell.left + month.left;
+ if (left <= coords.x && coords.x < left + calendar.cellWidth) {
+ if (top <= coords.y && coords.y < top + calendar.cellHeight) {
+ return {
+ "month": monthIndex,
+ "x": x,
+ "y": y,
+ "coords": coords
+ };
+ }
+ }
+ }
+ }
+
+ return null;
+ };
+
+ this._onTopMouseDown = function(ev) {
+ var freeHandSelection = calendar.freeHandSelectionEnabled;
+ if (!freeHandSelection) {
+ return;
+ }
+ var start = calendar._getPosition(ev);
+ if (start && !start.header) {
+ ps.start = start;
+ }
+ var cell = calendar.months[start.month].cells[start.x][start.y];
+
+ ev.preventDefault();
+
+ };
+
+ this._onTopMouseMove = function(ev) {
+ if (!ps.start) {
+ return;
+ }
+ var end = calendar._getPosition(ev);
+ if (ps.end) {
+ ps.end = end;
+ }
+ else if (end) {
+ var requiredDistance = 3;
+ var distance = DayPilot.distance(ps.start.coords, end.coords);
+
+ if (distance > requiredDistance) {
+ ps.end = end;
+ }
+ }
+
+ if (ps.end) {
+ ps.clear();
+ ps.draw();
+ }
+ //ps.end = end;
+ };
+
+ this._preselection = {};
+ var ps = this._preselection;
+ ps.start = null;
+
+ ps.drawCell = function(pos) {
+ var month = calendar.months[pos.month];
+
+ var cellPos = calendar._cellRelativeCoords(pos.x, pos.y);
+
+ var top = month.top + cellPos.y;
+ var left = month.left + cellPos.x;
+
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+ div.style.left = left + "px";
+ div.style.top = top + "px";
+ div.style.height = calendar.cellHeight + "px";
+ div.style.width = calendar.cellWidth + "px";
+ div.style.backgroundColor = "#ccc";
+ div.style.opacity = 0.5;
+ div.style.cursor = "default";
+
+ calendar.nav.preselection.appendChild(div);
+ ps.cells.push(div);
+ };
+
+ ps.clear = function() {
+ if (!ps.cells) {
+ return;
+ }
+
+ for(var i = 0; i < ps.cells.length; i++) {
+ calendar.nav.preselection.removeChild(ps.cells[i]);
+ }
+ ps.cells = [];
+ };
+
+ ps.draw = function() {
+ var ordered = ps.ordered();
+
+ var position = new Position(ordered.start);
+ var end = ordered.end;
+
+ if (!end) {
+ return;
+ }
+
+ if (end === ps.end && end.header) {
+ if (end.month > 0) {
+ end.month -= 1;
+ var month = calendar.months[end.month];
+ end.x = 6;
+ end.y = month.rowCount - 1;
+ }
+ }
+
+ ps.cells = [];
+
+ while (!position.is(end)) {
+ ps.drawCell(position);
+ var next = new Position(position).next();
+ if (!next) {
+ return;
+ }
+ position.month = next.month;
+ position.x = next.x;
+ position.y = next.y;
+ }
+
+ // the last one
+ ps.drawCell(position);
+
+ };
+
+ ps.ordered = function() {
+ //var position = new Position(start);
+ var start = ps.start;
+ var end = ps.end;
+
+ var result = {};
+ if (!end || new Position(start).before(end)) {
+ result.start = start;
+ result.end = end;
+ }
+ else {
+ result.start = end;
+ result.end = start;
+ }
+ return result;
+ };
+
+ var Position = function(month, x, y) {
+
+ if (month instanceof Position) {
+ return month;
+ }
+ if (typeof month === "object") {
+ var ref = month;
+ this.month = ref.month;
+ this.x = ref.x;
+ this.y = ref.y;
+ }
+ else {
+ this.month = month;
+ this.x = x;
+ this.y = y;
+ }
+ this.is = function(ref) {
+ return this.month === ref.month && this.x === ref.x && this.y === ref.y;
+ };
+
+ this.next = function() {
+ var start = this;
+ if (start.x < 6) {
+ return {
+ "month": start.month,
+ "x": start.x + 1,
+ "y": start.y
+ };
+ }
+ var month = calendar.months[start.month];
+ if (start.y < month.rowCount - 1) {
+ return {
+ "month": start.month,
+ "x": 0,
+ "y": start.y + 1
+ };
+ }
+ if (start.month < calendar.months.length - 1) {
+ return {
+ "month": start.month + 1,
+ "x": 0,
+ "y": 0
+ };
+ }
+ return null;
+ };
+
+ this.visible = function() {
+ var cell = this.cell();
+ if (cell.isCurrentMonth) {
+ return true;
+ }
+ if (cell.isPrevMonth && cell.showBefore) {
+ return true;
+ }
+ if (cell.isNextMonth && cell.showAfter) {
+ return true;
+ }
+ return false;
+ };
+
+ this.nextVisible = function() {
+ var pos = this;
+ while (!pos.visible()) {
+ var next = pos.next();
+ if (!next) {
+ return null;
+ }
+ pos = new Position(next);
+ }
+ return pos;
+ };
+
+ this.previous = function() {
+ var start = this;
+ if (start.x > 0) {
+ return {
+ "month": start.month,
+ "x": start.x - 1,
+ "y": start.y
+ };
+ }
+ var month = calendar.months[start.month];
+ if (start.y > 0) {
+ return {
+ "month": start.month,
+ "x": 6,
+ "y": start.y - 1
+ };
+ }
+ if (start.month > 0) {
+ var m = calendar.months[start.month - 1];
+ return {
+ "month": start.month - 1,
+ "x": 6,
+ "y": m.rowCount - 1
+ };
+ }
+ return null;
+
+ };
+
+ this.previousVisible = function() {
+ var pos = this;
+ while (!pos.visible()) {
+ var previous = pos.previous();
+ if (!previous) {
+ return null;
+ }
+ pos = new Position(previous);
+ }
+ return pos;
+ };
+
+ this.cell = function() {
+ return calendar.months[this.month].cells[this.x][this.y];
+ };
+
+ this.date = function() {
+ return this.cell().day;
+ };
+
+ this.before = function(ref) {
+ var thisDate = this.date();
+ var refDate = new Position(ref).date();
+ return thisDate < refDate;
+ };
+ };
+
+
+ this._cellClick = function(ev) {
+ var main = this.parentNode;
+ var month = this.parentNode.month;
+
+ var x = this.x;
+ var y = this.y;
+ var day = month.cells[x][y].day;
+
+ if (!month.cells[x][y].isClickable) {
+ return;
+ }
+
+ calendar.clearSelection();
+
+ calendar.selectionDay = day;
+
+ var day = calendar.selectionDay;
+ switch (calendar._selectModeLowerCase()) {
+ case 'none':
+ //var s = month.cells[x][y];
+ calendar.selectionStart = day;
+ calendar.selectionEnd = day;
+ break;
+ case 'day':
+ if (calendar.autoFocusOnClick) {
+ var start = day;
+ if (day < calendar._activeStart() ||
+ day >= calendar._activeEnd()) {
+ calendar.select(day);
+ return;
+ }
+ }
+
+ var s = month.cells[x][y];
+ calendar._cellSelect(main, x, y);
+ //calendar._addClass(s, 'select');
+ calendar.selected.push(s);
+ calendar.selectionStart = s.day;
+ calendar.selectionEnd = s.day;
+
+ break;
+ case 'week':
+ if (calendar.autoFocusOnClick) {
+ var start = month.cells[0][y].day;
+ var end = month.cells[6][y].day;
+ if (start.firstDayOfMonth() === end.firstDayOfMonth()) {
+ if (start < calendar._activeStart() ||
+ end >= calendar._activeEnd()) {
+ calendar.select(day);
+ return;
+ }
+ }
+ }
+ for (var j = 0; j < 7; j++) {
+ calendar._cellSelect(main, j, y);
+ //calendar._addClass(month.cells[j][y], 'select');
+ calendar.selected.push(month.cells[j][y]);
+ }
+ calendar.selectionStart = month.cells[0][y].day;
+ calendar.selectionEnd = month.cells[6][y].day;
+
+ break;
+ case 'month':
+ if (calendar.autoFocusOnClick) {
+ var start = day;
+ if (day < calendar._activeStart() ||
+ day >= calendar._activeEnd()) {
+ calendar.select(day);
+ return;
+ }
+ }
+
+ var start = null;
+ var end = null;
+ for (var y = 0; y < 6; y++) {
+ for (var x = 0; x < 7; x++) {
+ var s = month.cells[x][y];
+ if (!s) {
+ continue;
+ }
+ if (s.day.getYear() === day.getYear() && s.day.getMonth() === day.getMonth()) {
+ calendar._cellSelect(main, x, y);
+ //calendar._addClass(s, 'select');
+ calendar.selected.push(s);
+ if (start === null) {
+ start = s.day;
+ }
+ end = s.day;
+ }
+ }
+ }
+ calendar.selectionStart = start;
+ calendar.selectionEnd = end;
+ break;
+ default:
+ throw 'unknown selectMode';
+ }
+
+ calendar._saveState();
+
+ calendar._timeRangeSelectedDispatch();
+ };
+
+ this._timeRangeSelectedDispatch = function(options) {
+ var start = calendar.selectionStart;
+ var end = calendar.selectionEnd.addDays(1);
+ var days = DayPilot.DateUtil.daysDiff(start, end);
+ var day = calendar.selectionDay;
+
+ options = options || {};
+
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.day = day;
+ args.days = days;
+ args.mode = options.mode || calendar.selectMode;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onTimeRangeSelect === 'function') {
+ calendar.onTimeRangeSelect(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'Bind':
+ if (typeof bound === "object") {
+ var selection = {};
+ selection.start = start;
+ selection.end = end;
+ selection.days = days;
+ selection.day = day;
+ bound.commandCallBack(calendar.command, selection);
+ }
+ break;
+ case 'None':
+ break;
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end, day);
+ break;
+ }
+
+ if (typeof calendar.onTimeRangeSelected === 'function') {
+ calendar.onTimeRangeSelected(args);
+ }
+
+ }
+ else {
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'Bind':
+ if (typeof bound === "object") {
+ var selection = {};
+ selection.start = start;
+ selection.end = end;
+ selection.days = days;
+ selection.day = day;
+ bound.commandCallBack(calendar.command, selection);
+ }
+ break;
+ case 'JavaScript':
+ calendar.onTimeRangeSelected(start, end, day);
+ break;
+ case 'None':
+ break;
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end, day);
+ break;
+ }
+ }
+
+
+ };
+
+ this.timeRangeSelectedPostBack = function(start, end, data, day) {
+ var params = {};
+ params.start = start;
+ params.end = end;
+ params.day = day;
+
+ this._postBack2('TimeRangeSelected', data, params);
+ };
+
+ this._clickRight = function(ev) {
+ calendar._moveMonth(calendar.skipMonths);
+ };
+
+ this._clickLeft = function(ev) {
+ calendar._moveMonth(-calendar.skipMonths);
+ };
+
+ this._moveMonth = function(i) {
+ this.startDate = this.startDate.addMonths(i);
+ this._clearTable();
+ this._prepare();
+ this._drawMonths();
+
+ this._saveState();
+
+ this._visibleRangeChangedDispatch();
+ this._updateFreeBusy();
+ };
+
+ this._activeStart = function() {
+ return calendar.startDate.firstDayOfMonth();
+ };
+
+ this._activeEnd = function() {
+ return calendar.startDate.firstDayOfMonth().addMonths(this.showMonths);
+ };
+
+ this.visibleStart = function() {
+ return calendar.startDate.firstDayOfMonth().firstDayOfWeek(resolved.weekStarts());
+ };
+
+ this.visibleEnd = function() {
+ return calendar.startDate.firstDayOfMonth().addMonths(this.showMonths - 1).firstDayOfWeek(resolved.weekStarts()).addDays(42);
+ };
+
+ this._visibleRangeChangedDispatch = function() {
+ var start = this.visibleStart();
+ var end = this.visibleEnd();
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onVisibleRangeChange === 'function') {
+ calendar.onVisibleRangeChange(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (this.visibleRangeChangedHandling) {
+ case "CallBack":
+ this.visibleRangeChangedCallBack(null);
+ break;
+ case "PostBack":
+ this.visibleRangeChangedPostBack(null);
+ break;
+ case "Disabled":
+ break;
+ }
+
+ if (typeof calendar.onVisibleRangeChanged === 'function') {
+ calendar.onVisibleRangeChanged(args);
+ }
+
+ }
+ else {
+ switch (this.visibleRangeChangedHandling) {
+ case "CallBack":
+ this.visibleRangeChangedCallBack(null);
+ break;
+ case "PostBack":
+ this.visibleRangeChangedPostBack(null);
+ break;
+ case "JavaScript":
+ this.onVisibleRangeChanged(start, end);
+ break;
+ case "Disabled":
+ break;
+ }
+ }
+
+ /*
+ switch (this.visibleRangeChangedHandling) {
+ case "CallBack":
+ this.visibleRangeChangedCallBack(null);
+ break;
+ case "PostBack":
+ this.visibleRangeChangedPostBack(null);
+ break;
+ case "JavaScript":
+ this.onVisibleRangeChanged(start, end);
+ break;
+ case "Disabled":
+ break;
+ }
+ */
+ };
+
+
+ this.visibleRangeChangedCallBack = function(data) {
+ var parameters = {};
+ this._callBack2("Visible", data, parameters);
+ };
+
+ this.visibleRangeChangedPostBack = function(data) {
+ var parameters = {};
+ this._postBack2("Visible", data, parameters);
+ };
+
+ this._updateView = function(result, context) {
+ var result = JSON.parse(result);
+ calendar.items = result.Items;
+ calendar.cells = result.Cells;
+
+ if (calendar.cells) {
+ calendar.update();
+ }
+ else {
+ calendar._updateFreeBusy();
+ }
+ };
+
+ this._drawMonths = function() {
+ for (var j = 0; j < this.showMonths; j++) {
+ var showLinks = this._getShowLinks(j);
+ this._drawTable(j, showLinks);
+ }
+
+ this.root.style.height = this._getHeight() + "px";
+
+ this.nav.preselection = document.createElement("div");
+ this.nav.preselection.style.position = "absolute";
+ this.nav.preselection.style.left = "0px";
+ this.nav.preselection.style.top = "0px";
+ this.root.appendChild(this.nav.preselection);
+
+ /*
+ var div = document.createElement("div");
+ div.style.clear = "left";
+ div.style.height = "0px";
+ div.style.width = "0px";
+ this.root.appendChild(div);
+ */
+
+ };
+
+
+ this._getHeight = function() {
+ if (this.orientation === "Horizontal") {
+ var max = 0;
+ for (var i = 0; i < this.months.length; i++) {
+ var month = this.months[i];
+ if (month.height > max) {
+ max = month.height;
+ }
+ }
+ return max;
+ }
+ else {
+ var total = 0;
+ for (var i = 0; i < this.months.length; i++) {
+ var month = this.months[i];
+ //total += this.showMonths*(this.cellHeight*month.rowCount + this.titleHeight + this.dayHeaderHeight);
+ total += month.height;
+ }
+ return total;
+ }
+ };
+
+ this._getShowLinks = function(j) {
+ if (this.internal.showLinks) {
+ return this.internal.showLinks;
+ }
+
+ var showLinks = {};
+ showLinks.left = (j === 0);
+ showLinks.right = (j === 0);
+ showLinks.before = j === 0;
+ showLinks.after = j === this.showMonths - 1;
+
+ if (this.orientation === "Horizontal") {
+ showLinks.right = (j === this.showMonths - 1);
+ }
+
+ return showLinks;
+ };
+
+ // not used at the moment - no internal changes to data
+
+ this._angular = {};
+ this._angular.scope = null;
+ this._angular.notify = function() {
+ if (calendar._angular.scope) {
+ calendar._angular.scope["$apply"]();
+ }
+ };
+
+ this.internal = {};
+ this.internal.loadOptions = calendar._loadOptions;
+ // ASP.NET
+ this.internal.initialized = function() {
+ return calendar._initialized;
+ };
+
+ this._resolved = {};
+ var resolved = this._resolved;
+
+ resolved.locale = function() {
+ return DayPilot.Locale.find(calendar.locale);
+ };
+
+ resolved.weekStarts = function() {
+ if (calendar.weekStarts === 'Auto') {
+ var locale = resolved.locale();
+ if (locale) {
+ return locale.weekStarts;
+ }
+ else {
+ return 0; // Sunday
+ }
+ }
+ else {
+ return calendar.weekStarts;
+ }
+ };
+
+ resolved.cellWidth = function() {
+ if (calendar._cache.cellWidth) {
+ return calendar._cache.cellWidth;
+ }
+ var width = calendar._getDimensionsFromCss("_cell_dimensions").width;
+ if (!width) {
+ width = calendar.cellWidth;
+ }
+ calendar._cache.cellWidth = width;
+ return width;
+ };
+
+ this.clearSelection = function() {
+ for (var j = 0; j < this.selected.length; j++) {
+ //this._removeClass(this.selected[j], 'select');
+ var div = this.selected[j];
+ calendar._cellUnselect(div.parentNode, div.x, div.y);
+ }
+ this.selected = [];
+ };
+
+ this._isShortInit = function() {
+ // make sure it has a place to ask
+ if (this.backendUrl) {
+ return (typeof calendar.items === 'undefined') || (!calendar.items);
+ }
+ else {
+ return false;
+ }
+ };
+
+ this.events = {};
+
+ this._loadEvents = function() {
+ if (!DayPilot.isArray(this.events.list)) {
+ return;
+ }
+
+ this.items = {};
+
+ for(var i = 0; i < this.events.list.length; i++) {
+ var e = this.events.list[i];
+ if (e.hidden) {
+ continue;
+ }
+ var days = this._eventDays(e);
+ for(var name in days) {
+ this.items[name] = 1;
+ }
+ }
+ };
+
+ this._getDimensionsFromCss = function(className) {
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+ div.style.top = "-2000px";
+ div.style.left = "-2000px";
+ div.className = this._prefixCssClass(className);
+
+ var container = calendar.root || document.body;
+
+ container.appendChild(div);
+ var height = div.offsetHeight;
+ var width = div.offsetWidth;
+ container.removeChild(div);
+
+ var result = {};
+ result.height = height;
+ result.width = width;
+ return result;
+ };
+
+ this._eventDays = function(e) {
+ var start = new DayPilot.Date(e.start);
+ var end = new DayPilot.Date(e.end);
+
+ var days = {};
+
+ var d = start.getDatePart();
+ while (d.getTime() <= end.getTime()) {
+ days[d.toStringSortable()] = 1;
+ d = d.addDays(1);
+ }
+
+ return days;
+ };
+
+ this.show = function() {
+ calendar.visible = true;
+ calendar.root.style.display = '';
+ };
+
+ this.hide = function() {
+ calendar.visible = false;
+ calendar.root.style.display = 'none';
+ };
+
+ this._loadTop = function() {
+ if (this.id && this.id.tagName) {
+ this.nav.top = this.id;
+ }
+ else if (typeof this.id === "string") {
+ this.nav.top = document.getElementById(this.id);
+ if (!this.nav.top) {
+ throw "DayPilot.Navigator: The placeholder element not found: '" + id + "'.";
+ }
+ }
+ else {
+ throw "DayPilot.Navigator() constructor requires the target element or its ID as a parameter";
+ }
+
+ this.root = this.nav.top;
+
+ };
+
+ this.init = function() {
+ this._loadTop();
+
+ if (this.root.dp) { // already initialized
+ return;
+ }
+
+ this._adjustSelection();
+ this._prepare();
+ this._drawMonths();
+ this._loadEvents();
+ this._updateFreeBusy();
+ this._registerDispose();
+ this._registerTopHandlers();
+ this._registerGlobalHandlers();
+
+ //this.select(this.selectionStart);
+
+ var loadFromServer = this._isShortInit();
+ if (loadFromServer) {
+ this._visibleRangeChangedDispatch(); // TODO change to "Init"?
+ }
+ this._initialized = true;
+ this._postInit();
+
+ if (this._postponedSelect) {
+ var params = this._postponedSelect;
+ this.select(params.date1, params.date2, params.options);
+ this._postponedSelect = null;
+ }
+
+ return this;
+ };
+
+ this._registerTopHandlers = function() {
+ calendar.nav.top.onmousedown = this._onTopMouseDown;
+ calendar.nav.top.onmousemove = this._onTopMouseMove;
+ };
+
+ this._registerGlobalHandlers = function() {
+ DayPilot.re(document, 'mouseup', calendar._gMouseUp);
+ };
+
+ this._gMouseUp = function(ev) {
+ if (ps.start && ps.end) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ // click, cancel
+ if (coords.x === ps.start.coords.x && coords.y === ps.start.coords.y) {
+ ps.start = null;
+ ps.clear();
+ return;
+ }
+
+ ps.clear();
+
+ var ordered = ps.ordered();
+
+ ordered.start = new Position(ordered.start).nextVisible();
+ ordered.end = new Position(ordered.end).previousVisible();
+
+
+ calendar.selectionDay = new Position(ordered.start).date();
+ calendar.selectionStart = calendar.selectionDay;
+ calendar.selectionEnd = new Position(ordered.end).date();
+
+ ps.start = null;
+ ps.end = null;
+
+ // redraw
+ calendar._clearTable();
+ calendar._prepare();
+ calendar._drawMonths();
+ calendar._updateFreeBusy();
+ calendar._saveState();
+
+ var notify = true;
+ if (notify) {
+ calendar._timeRangeSelectedDispatch({"mode": "FreeHand"});
+ }
+
+ }
+
+ // clear in either case
+ ps.start = null;
+ ps.end = null;
+
+ };
+
+ this.dispose = function() {
+ var c = calendar;
+
+ if (!c.root) {
+ return;
+ }
+
+ c.root.removeAttribute("style");
+ c.root.removeAttribute("class");
+ c.root.dp = null;
+ c.root.innerHTML = null;
+ c.root = null;
+
+ c._disposed = true;
+
+ };
+
+ this._registerDispose = function() {
+ //var root = document.getElementById(id);
+ this.root.dispose = this.dispose;
+ };
+
+ this.Init = this.init;
+
+ this._loadOptions(options);
+ };
+
+ // jQuery plugin
+ if (typeof jQuery !== 'undefined') {
+ (function($) {
+ $.fn.daypilotNavigator = function(options) {
+ var first = null;
+ var j = this.each(function() {
+ if (this.daypilot) { // already initialized
+ return;
+ };
+
+ var daypilot = new DayPilot.Navigator(this.id);
+ this.daypilot = daypilot;
+ for (var name in options) {
+ daypilot[name] = options[name];
+ }
+ daypilot.Init();
+ if (!first) {
+ first = daypilot;
+ }
+ });
+ if (this.length === 1) {
+ return first;
+ }
+ else {
+ return j;
+ }
+ };
+ })(jQuery);
+ }
+
+ // AngularJS plugin
+ (function registerAngularModule() {
+ var app = DayPilot.am();
+
+ if (!app) {
+ return;
+ }
+
+
+ app.directive("daypilotNavigator", ['$parse', function($parse) {
+ return {
+ "restrict": "E",
+ "template": "",
+ "compile": function compile(element, attrs) {
+ element.replaceWith(this["template"].replace("{{id}}", attrs["id"]));
+
+ return function link(scope, element, attrs) {
+ var calendar = new DayPilot.Navigator(element[0]);
+ calendar._angular.scope = scope;
+ calendar.init();
+
+ var oattr = attrs["id"];
+ if (oattr) {
+ scope[oattr] = calendar;
+ }
+
+ // save DayPilot.Calendar object in the specified variable
+ var pas = attrs["publishAs"];
+ if (pas) {
+ var getter = $parse(pas);
+ var setter = getter.assign;
+ setter(scope, calendar);
+ }
+
+ // bind event handlers from attributes starting with "on"
+ for (var name in attrs) {
+ if (name.indexOf("on") === 0) { // event handler
+ var apply = DayPilot.Util.shouldApply(name);
+
+ if (apply) {
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ scope["$apply"](function() {
+ f(scope, {"args": args});
+ });
+ };
+ })(name);
+ }
+ else {
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ f(scope, {"args": args});
+ };
+ })(name);
+ }
+
+ }
+ }
+
+ var watch = scope["$watch"];
+ var config = attrs["config"] || attrs["daypilotConfig"];
+ var events = attrs["events"] || attrs["daypilotEvents"];
+
+ watch.call(scope, config, function (value, oldVal) {
+ for (var name in value) {
+ calendar[name] = value[name];
+ }
+ calendar.update();
+ }, true);
+
+ watch.call(scope, events, function(value) {
+ calendar.events.list = value;
+ calendar._loadEvents();
+ calendar._updateFreeBusy();
+ }, true);
+
+ };
+ }
+ };
+ }]);
+
+ })();
+
+ if (typeof Sys !== 'undefined' && Sys.Application && Sys.Application.notifyScriptLoaded) {
+ Sys.Application.notifyScriptLoaded();
+ }
+
+
+})();
diff --git a/static/src/js/src/daypilot-calendar.src.js b/static/src/js/src/daypilot-calendar.src.js
new file mode 100644
index 0000000..553a6ba
--- /dev/null
+++ b/static/src/js/src/daypilot-calendar.src.js
@@ -0,0 +1,3979 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function() {
+
+ var doNothing = function() {};
+
+ if (typeof DayPilot.Calendar !== 'undefined' && DayPilot.Calendar.events) {
+ return;
+ }
+
+ var DayPilotCalendar = {};
+
+ // internal selecting
+ DayPilotCalendar.selectedCells = [];
+ DayPilotCalendar.topSelectedCell = null;
+ DayPilotCalendar.bottomSelectedCell = null;
+ DayPilotCalendar.selecting = false;
+ DayPilotCalendar.column = null;
+ DayPilotCalendar.firstSelected = null;
+ DayPilotCalendar.firstMousePos = null;
+
+ // internal resizing
+ DayPilotCalendar.originalMouse = null;
+ DayPilotCalendar.originalHeight = null;
+ DayPilotCalendar.originalTop = null;
+ DayPilotCalendar.resizing = null;
+ DayPilotCalendar.globalHandlers = false;
+
+ // internal moving
+ DayPilotCalendar.moving = null;
+
+ // helpers
+ DayPilotCalendar.register = function(calendar) {
+ if (!DayPilotCalendar.registered) {
+ DayPilotCalendar.registered = [];
+ }
+ var r = DayPilotCalendar.registered;
+
+ for (var i = 0; i < r.length; i++) {
+ if (r[i] === calendar) {
+ return;
+ }
+ }
+ r.push(calendar);
+ };
+
+ DayPilotCalendar.unregister = function (calendar) {
+ var a = DayPilotCalendar.registered;
+ if (!a) {
+ return;
+ }
+
+ var i = DayPilot.indexOf(a, calendar);
+ if (i === -1) {
+ return;
+ }
+ a.splice(i, 1);
+ };
+
+ DayPilotCalendar.getCellsAbove = function(cell) {
+ var array = [];
+ var c = DayPilotCalendar.getColumn(cell);
+
+ var tr = cell.parentNode;
+
+ var select = null;
+ while (tr && select !== DayPilotCalendar.firstSelected) {
+ select = tr.getElementsByTagName("td")[c];
+ array.push(select);
+ tr = tr.previousSibling;
+ while (tr && tr.tagName !== "TR") {
+ tr = tr.previousSibling;
+ }
+ }
+ return array;
+ };
+
+ DayPilotCalendar.getCellsBelow = function(cell) {
+ var array = [];
+ var c = DayPilotCalendar.getColumn(cell);
+ var tr = cell.parentNode;
+
+ var select = null;
+ while (tr && select !== DayPilotCalendar.firstSelected) {
+ select = tr.getElementsByTagName("td")[c];
+ array.push(select);
+ tr = tr.nextSibling;
+ while (tr && tr.tagName !== "TR") {
+ tr = tr.nextSibling;
+ }
+ }
+ return array;
+ };
+
+ DayPilotCalendar.getColumn = function(cell) {
+ var i = 0;
+ while (cell.previousSibling) {
+ cell = cell.previousSibling;
+ if (cell.tagName === "TD") {
+ i++;
+ }
+ }
+ return i;
+ };
+
+ DayPilotCalendar.gUnload = function (ev) {
+
+ if (!DayPilotCalendar.registered) {
+ return;
+ }
+ var r = DayPilotCalendar.registered;
+
+ for (var i = 0; i < r.length; i++) {
+ var c = r[i];
+ c.dispose();
+
+ DayPilotCalendar.unregister(c);
+ }
+
+ };
+
+ DayPilotCalendar.gMouseUp = function (ev){
+
+ if (DayPilotCalendar.resizing) {
+ if (!DayPilotCalendar.resizingShadow) {
+ DayPilotCalendar.resizing.style.cursor = 'default';
+ document.body.style.cursor = 'default';
+ DayPilotCalendar.resizing = null;
+ DayPilot.Global.resizing = null;
+ return;
+ }
+
+ var dpEvent = DayPilotCalendar.resizing.event;
+ var height = DayPilotCalendar.resizingShadow.clientHeight + 4;
+ var top = DayPilotCalendar.resizingShadow.offsetTop;
+ var border = DayPilotCalendar.resizing.dpBorder;
+
+ // stop resizing on the client
+ DayPilotCalendar.deleteShadow(DayPilotCalendar.resizingShadow);
+ DayPilotCalendar.resizingShadow = null;
+ DayPilotCalendar.resizing.style.cursor = 'default';
+ dpEvent.calendar.nav.top.style.cursor = 'auto';
+ //document.body.style.cursor = 'default';
+ DayPilotCalendar.resizing.onclick = null; // will be re-created anyway
+
+ DayPilotCalendar.resizing = null;
+ DayPilot.Global.resizing = null;
+
+ dpEvent.calendar._eventResizeDispatch(dpEvent, height, top, border);
+ }
+ else if (DayPilotCalendar.moving) {
+ if (!DayPilotCalendar.movingShadow) {
+ DayPilotCalendar.moving = null;
+ DayPilot.Global.moving = null;
+ document.body.style.cursor = 'default';
+ return;
+ }
+
+ var top = DayPilotCalendar.movingShadow.offsetTop;
+ var dpEvent = DayPilotCalendar.moving.event;
+
+ DayPilotCalendar.deleteShadow(DayPilotCalendar.movingShadow);
+ DayPilot.Util.removeClass(DayPilotCalendar.moving, dpEvent.calendar._prefixCssClass("_event_moving_source"));
+
+ var newColumnIndex = DayPilotCalendar.movingShadow.column;
+
+ // stop moving on the client
+ DayPilotCalendar.moving = null;
+ DayPilot.Global.moving = null;
+ DayPilotCalendar.movingShadow = null;
+ //document.body.style.cursor = 'default';
+ dpEvent.calendar.nav.top.style.cursor = 'auto';
+
+ dpEvent.calendar._eventMoveDispatch(dpEvent, newColumnIndex, top, ev);
+ }
+ else if (DayPilotCalendar.selecting && DayPilotCalendar.topSelectedCell !== null) {
+ var calendar = DayPilotCalendar.selecting.calendar;
+ DayPilotCalendar.selecting = false;
+
+ var sel = calendar.getSelection();
+
+ calendar._timeRangeSelectedDispatch(sel.start, sel.end, sel.resource);
+ if (calendar.timeRangeSelectedHandling !== "Hold" && calendar.timeRangeSelectedHandling !== "HoldForever") {
+ doNothing();
+ }
+ }
+ else {
+ DayPilotCalendar.selecting = false;
+ }
+
+
+ };
+
+ DayPilotCalendar.deleteShadow = function(shadow) {
+ if (!shadow) {
+ return;
+ }
+ if (!shadow.parentNode) {
+ return;
+ }
+
+ shadow.parentNode.removeChild(shadow);
+ };
+
+ DayPilotCalendar.moveShadow = function(column) {
+ var shadow = DayPilotCalendar.movingShadow;
+ var parent = shadow.parentNode;
+
+ parent.style.display = 'none';
+
+ shadow.parentNode.removeChild(shadow);
+ column.firstChild.appendChild(shadow);
+ shadow.style.left = '0px';
+
+ parent.style.display = '';
+
+ shadow.style.width = (DayPilotCalendar.movingShadow.parentNode.offsetWidth + 1) + 'px';
+ };
+
+ DayPilotCalendar.Calendar = function(id, options) {
+
+ var isConstructor = false;
+ if (this instanceof DayPilotCalendar.Calendar && !this.__constructor) {
+ isConstructor = true;
+ this.__constructor = true;
+ }
+
+ if (!isConstructor) {
+ throw "DayPilot.Calendar() is a constructor and must be called as 'var c = new DayPilot.Calendar(id);'";
+ }
+
+ var calendar = this;
+ this.uniqueID = null;
+ this.isCalendar = true;
+
+ this.v = '2024.3.539-lite';
+ this.id = id;
+ this.clientName = id;
+
+ this.cache = {};
+ this.cache.pixels = {};
+
+ this.elements = {};
+ this.elements.events = [];
+ this.elements.selection = [];
+
+ this.nav = {};
+
+ this.afterRender = function() {};
+
+ this.fasterDispose = true;
+
+ this.angularAutoApply = false;
+ this.api = 2;
+ this.businessBeginsHour = 9;
+ this.businessEndsHour = 18;
+ this.cellHeight = 30;
+ this.columnMarginRight = 5;
+ this.columnsLoadMethod = "GET";
+ this.contextMenu = null;
+ this.days = 1;
+ this.durationBarVisible = true;
+ this.eventsLoadMethod = "GET";
+ this.headerDateFormat = null; // uses locale.dateFormat by default
+ this.headerHeight = 30;
+ this.headerTextWrappingEnabled = false;
+ this.height = 300;
+ this.heightSpec = 'BusinessHours';
+ this.hideUntilInit = true;
+ this.hourWidth = 60;
+ this.initScrollPos = 'Auto';
+ this.loadingLabelHtml = null;
+ this.loadingLabelText = "Loading...";
+ this.loadingLabelVisible = true;
+ this.locale = "en-us";
+ this.showToolTip = true;
+ this.startDate = new DayPilot.Date().getDatePart();
+ this.cssClassPrefix = "calendar_default";
+ this.theme = null;
+ this.timeFormat = 'Auto';
+ this.viewType = "Days";
+ this.visible = true;
+ this.xssProtection = "Enabled";
+
+ this.headerClickHandling = "Enabled";
+ this.eventClickHandling = 'Enabled';
+ this.eventResizeHandling = 'Update';
+ this.eventRightClickHandling = 'ContextMenu';
+ this.eventMoveHandling = 'Update';
+ this.eventDeleteHandling = "Disabled";
+ this.timeRangeSelectedHandling = 'Enabled';
+
+ this.onBeforeCellRender = null;
+ this.onBeforeEventRender = null;
+
+ this.onEventClick = null;
+ this.onEventClicked = null;
+ this.onEventDelete = null;
+ this.onEventDeleted = null;
+ this.onEventMove = null;
+ this.onEventMoved = null;
+ this.onEventResize = null;
+ this.onEventResized = null;
+ this.onEventRightClick = null;
+ this.onEventRightClicked = null;
+ this.onHeaderClick = null;
+ this.onHeaderClicked = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+
+ this._disposed = false;
+
+ this.clearSelection = function() {
+ DayPilotCalendar.topSelectedCell = null;
+ DayPilotCalendar.bottomSelectedCell = null;
+ this._hideSelection();
+ };
+
+ this._hideSelection = function() {
+ DayPilot.de(calendar.elements.selection);
+ calendar.elements.selection = [];
+ calendar.nav.activeSelection = null;
+ };
+
+ this.cleanSelection = this.clearSelection;
+
+ this._postBack2 = function(action, data, parameters) {
+ var envelope = {};
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + DayPilot.JSON.stringify(envelope);
+ __doPostBack(calendar.uniqueID, commandstring);
+ };
+
+ this._callBack2 = function(action, data, parameters) {
+
+ if (this.callbackTimeout) {
+ window.clearTimeout(this.callbackTimeout);
+ }
+
+ this.callbackTimeout = window.setTimeout(function() {
+ calendar.loadingStart();
+ }, 100);
+
+ var envelope = {};
+
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + DayPilot.JSON.stringify(envelope);
+ if (this.backendUrl) {
+ DayPilot.request(this.backendUrl, this._callBackResponse, commandstring, this.ajaxError);
+ }
+ else if (typeof WebForm_DoCallback === 'function') {
+ WebForm_DoCallback(this.uniqueID, commandstring, this._updateView, this.clientName, this.onCallbackError, true);
+ }
+ };
+
+ this.onCallbackError = function(result, context) {
+ alert("Error!\r\nResult: " + result + "\r\nContext:" + context);
+ };
+
+ this.dispose = function() {
+
+
+ var c = calendar;
+
+ if (c._disposed) {
+ return;
+ }
+ c._disposed = true;
+
+ clearInterval(c._visibilityInterval);
+
+ c._deleteEvents();
+ c.nav.scroll.root = null;
+
+ DayPilot.pu(c.nav.loading);
+
+ c._disposeMain();
+ c._disposeHeader();
+
+ c.nav.select = null;
+ c.nav.cornerRight = null;
+ c.nav.scrollable = null;
+ c.nav.zoom = null;
+ c.nav.loading = null;
+ c.nav.header = null;
+ c.nav.hourTable = null;
+ c.nav.scrolltop = null;
+ c.nav.scroll.onscroll = null;
+ c.nav.scroll = null;
+ c.nav.main = null;
+ c.nav.message = null;
+ c.nav.messageClose = null;
+ c.nav.top = null;
+
+ DayPilotCalendar.unregister(c);
+ };
+
+ this.disposed = function() {
+ return this._disposed;
+ };
+
+ this._registerDispose = function() {
+ this.nav.top.dispose = this.dispose;
+ };
+
+ this._callBackResponse = function(response) {
+ calendar._updateView(response.responseText);
+ };
+
+ this._getCallBackHeader = function() {
+ var h = {};
+
+ h.control = "dpc";
+ h.id = this.id;
+ h.v = this.v;
+
+ h.days = calendar.days;
+ h.startDate = calendar.startDate;
+ h.heightSpec = calendar.heightSpec;
+ h.businessBeginsHour = calendar.businessBeginsHour;
+ h.businessEndsHour = calendar.businessEndsHour;
+ h.hashes = calendar.hashes;
+
+ // h.backColor = calendar.cellBackColor;
+ h.timeFormat = calendar.timeFormat;
+ h.viewType = calendar.viewType;
+ h.locale = calendar.locale;
+
+ return h;
+ };
+
+ this._createShadow = function(object, copyText) {
+ var parentTd = object.parentNode;
+ while (parentTd && parentTd.tagName !== "TD") {
+ parentTd = parentTd.parentNode;
+ }
+
+ var shadow = document.createElement('div');
+ shadow.setAttribute('unselectable', 'on');
+ shadow.style.position = 'absolute';
+ shadow.style.width = (object.offsetWidth) + 'px';
+ shadow.style.height = (object.offsetHeight) + 'px';
+ shadow.style.left = (object.offsetLeft) + 'px';
+ shadow.style.top = (object.offsetTop) + 'px';
+ shadow.style.boxSizing = "border-box";
+ shadow.style.zIndex = 101;
+
+ // shadow.style.backgroundColor = "#aaaaaa";
+ // shadow.style.opacity = 0.5;
+ // shadow.style.filter = "alpha(opacity=50)";
+ // shadow.style.border = '2px solid #aaaaaa';
+
+ shadow.className = calendar._prefixCssClass("_shadow");
+
+ var inner = document.createElement("div");
+ inner.className = calendar._prefixCssClass("_shadow_inner");
+ shadow.appendChild(inner);
+
+ if (copyText && false) { // disabled
+ shadow.style.overflow = 'hidden';
+ shadow.style.fontSize = object.style.fontSize;
+ shadow.style.fontFamily = object.style.fontFamily;
+ shadow.style.color = object.style.color;
+ shadow.innerHTML = object.data.client.html();
+ }
+
+ /*
+ shadow.style.MozBorderRadius = "5px";
+ shadow.style.webkitBorderRadius = "5px";
+ shadow.style.borderRadius = "5px";
+ */
+
+ parentTd.firstChild.appendChild(shadow);
+
+ return shadow;
+ };
+
+
+ this._resolved = {};
+ this._resolved.locale = function() {
+ var found = DayPilot.Locale.find(calendar.locale);
+ if (!found) {
+ return DayPilot.Locale.US;
+ }
+ return found;
+ };
+
+ this._resolved.timeFormat = function() {
+ if (calendar.timeFormat !== 'Auto') {
+ return calendar.timeFormat;
+ }
+ return this.locale().timeFormat;
+ };
+
+ this._resolved._xssProtectionEnabled = function() {
+ return calendar.xssProtection !== "Disabled";
+ };
+
+ this._resolved._weekStarts = function() {
+ if (calendar.weekStarts === 'Auto') {
+ var locale = resolved.locale();
+ if (locale) {
+ return locale.weekStarts;
+ }
+ else {
+ return 0; // Sunday
+ }
+ }
+ else {
+ return calendar.weekStarts || 0;
+ }
+ };
+
+ var resolved = this._resolved;
+
+ this._updateView = function(result, context) {
+
+ if (result && result.indexOf("$$$") === 0) {
+ if (window.console) {
+ console.log("Error received from the server side: " + result);
+ }
+ else {
+ throw "Error received from the server side: " + result;
+ }
+ return;
+ }
+
+ var result = JSON.parse(result);
+
+ if (result.CallBackRedirect) {
+ document.location.href = result.CallBackRedirect;
+ return;
+ }
+
+ if (result.UpdateType === "None") {
+ calendar.loadingStop();
+ calendar._show();
+ return;
+ }
+
+ calendar._deleteEvents();
+
+ if (result.UpdateType === "Full") {
+
+ calendar.columns = result.Columns;
+
+ // properties
+ calendar.days = result.Days;
+ calendar.startDate = new DayPilot.Date(result.StartDate);
+ calendar.heightSpec = result.HeightSpec ? result.HeightSpec : calendar.heightSpec;
+ calendar.businessBeginsHour = result.BusinessBeginsHour ? result.BusinessBeginsHour : calendar.businessBeginsHour;
+ calendar.businessEndsHour = result.BusinessEndsHour ? result.BusinessEndsHour : calendar.businessEndsHour;
+ calendar.headerDateFormat = result.HeaderDateFormat ? result.HeaderDateFormat : calendar.headerDateFormat;
+ calendar.viewType = result.ViewType; //
+ calendar.backColor = result.BackColor ? result.BackColor : calendar.backColor;
+ calendar.eventHeaderVisible = result.EventHeaderVisible ? result.EventHeaderVisible : calendar.eventHeaderVisible;
+ calendar.timeFormat = result.TimeFormat ? result.TimeFormat : calendar.timeFormat;
+ calendar.locale = result.Locale ? result.Locale : calendar.locale;
+
+ calendar._prepareColumns();
+ }
+
+ // hashes
+ if (result.Hashes) {
+ for (var key in result.Hashes) {
+ calendar.hashes[key] = result.Hashes[key];
+ }
+ }
+
+ calendar.events.list = result.Events;
+ calendar._loadEvents();
+ calendar._updateHeaderHeight();
+
+ if (result.UpdateType === "Full") {
+ calendar._drawHeader();
+ calendar._drawMain();
+ calendar._drawHourTable();
+ calendar._updateHeight();
+ }
+
+ calendar._show();
+
+ calendar._drawEvents();
+ calendar.clearSelection();
+
+ calendar.afterRender(result.CallBackData, true);
+
+ calendar.loadingStop();
+
+ };
+
+ this._durationHours = function() {
+ return this._duration() / (3600 * 1000);
+ };
+
+ this._businessHoursSpan = function() {
+ if (this.businessBeginsHour > this.businessEndsHour) {
+ return 24 - this.businessBeginsHour + this.businessEndsHour;
+ }
+ else {
+ return this.businessEndsHour - this.businessBeginsHour;
+ }
+ };
+
+ this._rowCount = function() {
+ return this._duration() / (60 * 60 * 1000 / 2);
+ };
+
+ // in ticks
+ this._duration = function() {
+ var dHours = 0;
+
+ if (this.heightSpec === 'BusinessHoursNoScroll') {
+ dHours = this._businessHoursSpan();
+ }
+ else {
+ dHours = 24;
+ }
+ return dHours * 60 * 60 * 1000; // return ticks
+ };
+
+ this._visibleStart = function() {
+ if (this.heightSpec === 'BusinessHoursNoScroll') {
+ return this.businessBeginsHour;
+ }
+ return 0;
+ };
+
+
+
+ this._api2 = function() {
+ return calendar.api === 2;
+ };
+
+ this.eventClickCallBack = function(e, data) {
+ this._callBack2('EventClick', data, e);
+ };
+
+ this.eventClickPostBack = function(e, data) {
+ this._postBack2('EventClick', data, e);
+ };
+
+ this._eventClickDispatch = function (ev) {
+ var thisDiv = this;
+
+ var e = thisDiv.event;
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.originalEvent = ev;
+ args.meta = ev.metaKey;
+ args.ctrl = ev.ctrlKey;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventClick === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClick(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventClickHandling) {
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'PostBack':
+ calendar.eventClickPostBack(e);
+ break;
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(e);
+ }
+ }
+ break;
+ }
+
+ if (typeof calendar.onEventClicked === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClicked(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventClickHandling) {
+ case 'PostBack':
+ calendar.eventClickPostBack(e);
+ break;
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventClick(e);
+ break;
+ }
+ }
+
+ };
+
+ this._eventRightClickDispatch = function(ev) {
+
+ var e = this.event;
+
+ if (ev.stopPropagation) {
+ ev.stopPropagation();
+ }
+
+ if (!e.client.rightClickEnabled()) {
+ return false;
+ }
+
+ var args = {};
+ args.e = e;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventRightClick === 'function') {
+ calendar.onEventRightClick(args);
+ if (args.preventDefault.value) {
+ return false;
+ }
+ }
+
+ switch (calendar.eventRightClickHandling) {
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(this.event);
+ }
+ }
+ break;
+ }
+
+ if (typeof calendar.onEventRightClicked === 'function') {
+ calendar.onEventRightClicked(args);
+ }
+
+
+ if (ev.preventDefault) {
+ ev.preventDefault();
+ }
+ return false;
+ };
+
+
+ this.eventDeleteCallBack = function(e, data) {
+ this._callBack2('EventDelete', data, e);
+ };
+
+ this.eventDeletePostBack = function(e, data) {
+ this._postBack2('EventDelete', data, e);
+ };
+
+ this._eventDeleteDispatch = function (e) {
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventDelete === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDelete(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventDeleteHandling) {
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'Update':
+ calendar.events.remove(e);
+ break;
+ }
+
+ if (typeof calendar.onEventDeleted === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDeleted(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventDeleteHandling) {
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventDelete(e);
+ break;
+ }
+ }
+
+ };
+
+ this.eventResizeCallBack = function(e, newStart, newEnd, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._callBack2('EventResize', data, params);
+ };
+
+ this.eventResizePostBack = function(e, newStart, newEnd, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._postBack2('EventResize', data, params);
+ };
+
+ this._eventResizeDispatch = function (e, shadowHeight, shadowTop, border ) {
+ var _startOffset = 1;
+
+ var newStart = new Date();
+ var newEnd = new Date();
+
+ var start = e.start();
+ var end = e.end();
+
+ if (border === 'top') {
+ var day = start.getDatePart();
+ var step = Math.floor((shadowTop - _startOffset) / calendar.cellHeight);
+ var minutes = step * 30;
+ var ts = minutes * 60 * 1000;
+ var visibleStartOffset = calendar._visibleStart() * 60 * 60 * 1000;
+
+ newStart = day.addTime(ts + visibleStartOffset);
+ newEnd = e.end();
+
+ }
+ else if (border === 'bottom') {
+ var day = end.getDatePart();
+ var step = Math.floor((shadowTop + shadowHeight - _startOffset) / calendar.cellHeight);
+ var minutes = step * 30;
+ var ts = minutes * 60 * 1000;
+ var visibleStartOffset = calendar._visibleStart() * 60 * 60 * 1000;
+
+ newStart = start;
+ newEnd = day.addTime(ts + visibleStartOffset);
+ }
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.control = calendar;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventResize === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResize(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventResizeHandling) {
+ case 'PostBack':
+ calendar.eventResizePostBack(e, newStart, newEnd);
+ break;
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventResized === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResized(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventResizeHandling) {
+ case 'PostBack':
+ calendar.eventResizePostBack(e, newStart, newEnd);
+ break;
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'JavaScript':
+ calendar.onEventResize(e, newStart, newEnd);
+ break;
+ }
+ }
+ };
+
+ this.eventMovePostBack = function(e, newStart, newEnd, newResource, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._postBack2('EventMove', data, params);
+ };
+
+ this.eventMoveCallBack = function(e, newStart, newEnd, newResource, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._callBack2('EventMove', data, params);
+ };
+
+ this._eventMoveDispatch = function (e, newColumnIndex, shadowTop, ev) {
+ var _startOffset = 1;
+ var step = Math.floor((shadowTop - _startOffset) / calendar.cellHeight);
+
+ var boxStart = step * 30 * 60 * 1000;
+ var start = e.start();
+ var end = e.end();
+ var day = new Date();
+
+ if (start instanceof DayPilot.Date) {
+ start = start.toDate();
+ }
+ day.setTime(Date.UTC(start.getUTCFullYear(), start.getUTCMonth(), start.getUTCDate()));
+
+ var startOffset = start.getTime() - (day.getTime() + start.getUTCHours() * 3600 *1000 + Math.floor(start.getUTCMinutes()/30)*30*60*1000 );
+ var length = end.getTime() - start.getTime();
+
+ var newColumn = this._columns[newColumnIndex];
+ var newResource = newColumn.id;
+
+ var date = newColumn.start.getTime();
+ var newStartUTC = new Date();
+ newStartUTC.setTime(date + boxStart + startOffset);
+
+ var newStart = new DayPilot.Date(newStartUTC);
+
+ var newEnd = newStart.addTime(length);
+
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.newResource = newResource;
+ args.ctrl = ev.ctrlKey;
+ args.shift = ev.shiftKey;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventMove === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMove(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventMoveHandling) {
+ case 'PostBack':
+ calendar.eventMovePostBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ e.resource(newResource);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventMoved === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMoved(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventMoveHandling) {
+ case 'PostBack':
+ calendar.eventMovePostBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd, newColumn.id);
+ break;
+ case 'JavaScript':
+ calendar.onEventMove(e, newStart, newEnd, newColumn.id, false);
+ break;
+ }
+ }
+
+ };
+
+ this.timeRangeSelectedPostBack = function(start, end, resource, data) {
+
+ var range = {};
+ range.start = start;
+ range.end = end;
+
+ this._postBack2('TimeRangeSelected', data, range);
+ };
+
+ this.timeRangeSelectedCallBack = function(start, end, resource, data) {
+
+ var range = {};
+ range.start = start;
+ range.end = end;
+
+ this._callBack2('TimeRangeSelected', data, range);
+ };
+
+ this._timeRangeSelectedDispatch = function (start, end, resource) {
+
+ // make sure it's DayPilot.Date
+ start = new DayPilot.Date(start);
+ end = new DayPilot.Date(end);
+
+ if (this._api2()) {
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.resource = resource;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onTimeRangeSelect === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelect(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end);
+ break;
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ }
+
+ if (typeof calendar.onTimeRangeSelected === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelected(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end);
+ break;
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ case 'JavaScript':
+ calendar.onTimeRangeSelected(start, end);
+ break;
+ }
+ }
+ };
+
+ this._onCellMousedown = function(ev) {
+
+ if (DayPilotCalendar.selecting) {
+ return;
+ }
+
+ if (calendar.timeRangeSelectedHandling === "Disabled") {
+ return;
+ }
+
+ var button = ev.which;
+ if (button !== 1 && button !== 0) { // Khtml says first button is 0
+ return;
+ }
+
+ DayPilotCalendar.firstMousePos = DayPilot.mc(ev);
+ DayPilotCalendar.selecting = {};
+ DayPilotCalendar.selecting.calendar = calendar;
+ if (DayPilotCalendar.selectedCells) {
+ calendar.clearSelection();
+ DayPilotCalendar.selectedCells = [];
+ }
+ DayPilotCalendar.column = DayPilotCalendar.getColumn(this);
+ DayPilotCalendar.selectedCells.push(this);
+ DayPilotCalendar.firstSelected = this;
+
+ DayPilotCalendar.topSelectedCell = this;
+ DayPilotCalendar.bottomSelectedCell = this;
+
+ calendar._activateSelection();
+
+ return false;
+ };
+
+ this._activateSelection = function() {
+ var selection = this.getSelection();
+
+ (function activateSelectionNew() {
+
+ var first = DayPilotCalendar.topSelectedCell;
+ var last = DayPilotCalendar.bottomSelectedCell;
+
+ // var columnIndex = first.parentNode.cells.indexOf(first);
+
+ var columnIndex = (function() {
+ // it's the new div cell
+ if (first.data) {
+ return first.data.x;
+ }
+
+ var cells = first.parentNode.cells;
+ for (var i = 0; i < cells.length; i++) {
+ if (cells[i] === first) {
+ return i;
+ }
+ }
+ return -1;
+ })();
+
+ var col = calendar._columns[columnIndex];
+
+ if (!col) {
+ return;
+ }
+
+ var colStart = col.start;
+
+ var top = calendar.getPixels(first.start, colStart).boxTop;
+ var bottom = calendar.getPixels(last.end, colStart).boxBottom;
+ var height = bottom - top;
+
+ var div = (function() {
+ if (calendar.nav.activeSelection) {
+ return calendar.nav.activeSelection;
+ }
+ var div = document.createElement("div");
+ div.setAttribute("unselectable", "on");
+ // div.style.zIndex = calendar._shadowZindex;
+ div.style.position = "absolute";
+ div.style.left = "0px";
+ div.style.width = "100%";
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = calendar._prefixCssClass("_shadow_inner");
+
+ div.appendChild(inner);
+
+ calendar.nav.events.rows[0].cells[columnIndex].selection.appendChild(div);
+ calendar.elements.selection.push(div);
+ calendar.nav.activeSelection = div;
+
+ return div;
+ })();
+
+ // reset
+ div.className = calendar._prefixCssClass("_shadow");
+ div.firstChild.innerHTML = "";
+
+ // position
+ div.style.top = top + "px";
+ div.style.height = height + "px";
+
+ })();
+ };
+
+ this._mousemove = function(ev) {
+
+ if (typeof (DayPilotCalendar) === 'undefined') {
+ return;
+ }
+
+ if (!DayPilotCalendar.selecting) {
+ return;
+ }
+
+ var mousePos = DayPilot.mc(ev);
+
+ var thisColumn = DayPilotCalendar.getColumn(this);
+ if (thisColumn !== DayPilotCalendar.column) {
+ return;
+ }
+
+ // clean
+ calendar.clearSelection();
+
+ // new selected cells
+ if (mousePos.y < DayPilotCalendar.firstMousePos.y) {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsBelow(this);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.selectedCells[0];
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.firstSelected;
+ }
+ else {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsAbove(this);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.firstSelected;
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.selectedCells[0];
+ }
+
+ calendar._activateSelection();
+ };
+
+ this.getSelection = function() {
+ var start = DayPilotCalendar.topSelectedCell.start;
+ var end = DayPilotCalendar.bottomSelectedCell.end;
+ var resource = DayPilotCalendar.topSelectedCell.resource;
+
+ return new DayPilot.Selection(start, end, resource, calendar);
+ };
+
+ this._getColumnForPixels = function(x) {
+
+ if (x < 0) {
+ return null;
+ }
+
+ var i = 0;
+ var cells = calendar.nav.events.rows[0].cells;
+
+ for (var j = 0; j < cells.length; j++) {
+ var cell = cells[j];
+ var width = cell.offsetWidth;
+ i += width;
+ if (x < i) {
+ return j;
+ }
+ }
+ return null;
+ };
+
+ this._table = {};
+ this._table.getCellCoords = function() {
+ var result = {};
+ result.x = 0;
+ result.y = 0;
+
+ if (!calendar.coords) {
+ return null;
+ }
+
+ result.x = calendar._getColumnForPixels(calendar.coords.x);
+
+ var _startOffset = 0;
+ var row = Math.floor((calendar.coords.y - _startOffset) / calendar.cellHeight);
+ result.y = row;
+
+ if (result.x < 0) {
+ return null;
+ }
+
+ return result;
+ };
+
+ this.columns = {};
+ this.columns.list = [];
+
+ this.columns.load = function(url, success, error) {
+
+ if (!url) {
+ throw new DayPilot.Exception("columns.load(): 'url' parameter required");
+ }
+
+ var onError = function(args) {
+ var largs = {};
+ largs.exception = args.exception;
+ largs.request = args.request;
+
+ if (typeof error === 'function') {
+ error(largs);
+ }
+ };
+
+ var onSuccess = function(args) {
+ var r = args.request;
+ var data;
+
+ // it's supposed to be JSON
+ try {
+ data = JSON.parse(r.responseText);
+ }
+ catch (e) {
+ var fargs = {};
+ fargs.exception = e;
+ onError(fargs);
+ return;
+ }
+
+ if (DayPilot.isArray(data)) {
+ var sargs = {};
+ sargs.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+ sargs.data = data;
+ if (typeof success === "function") {
+ success(sargs);
+ }
+
+ if (sargs.preventDefault.value) {
+ return;
+ }
+
+ calendar.columns.list = data;
+ if (calendar._initialized) {
+ calendar.update();
+ }
+ }
+ };
+
+ var usePost = calendar.columnsLoadMethod && calendar.columnsLoadMethod.toUpperCase() === "POST";
+
+ if (usePost) {
+ DayPilot.ajax({
+ "method": "POST",
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ else {
+ DayPilot.ajax({
+ "method": "GET",
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ };
+
+
+ this._prepareColumns = function() {
+ var columns;
+ if (calendar.viewType !== "Resources") {
+ columns = this._createDaysViewColumns();
+ }
+ else {
+ columns = calendar.columns.list;
+ }
+
+ this._columns = [];
+ for (var i = 0; i < columns.length; i++) {
+ var c = this._activateColumn(columns[i]);
+ this._columns.push(c);
+ }
+ };
+
+ this._activateColumn = function(column) {
+
+/*
+ if (column.Start) {
+ column.start = column.Start;
+ column.html = column.InnerHTML;
+ column.name = column.Name;
+ }
+*/
+ var result = {};
+ result.name = column.name;
+ result.html = column.html;
+ result.id = column.id;
+ result.toolTip = column.toolTip;
+ result.data = column;
+
+ if (column.start) {
+ result.start = new DayPilot.Date(column.start);
+ }
+ else {
+ result.start = new DayPilot.Date(calendar.startDate);
+ }
+
+ result.putIntoBlock = function(ep) {
+
+ for (var i = 0; i < this.blocks.length; i++) {
+ var block = this.blocks[i];
+ if (block.overlapsWith(ep.part.top, ep.part.height)) {
+ block.events.push(ep);
+ block.min = Math.min(block.min, ep.part.top);
+ block.max = Math.max(block.max, ep.part.top + ep.part.height);
+ return i;
+ }
+ }
+
+ // no suitable block found, create a new one
+ var block = [];
+ block.lines = [];
+ block.events = [];
+
+ block.overlapsWith = function(start, width) {
+ var end = start + width - 1;
+
+ if (!(end < this.min || start > this.max - 1)) {
+ return true;
+ }
+
+ return false;
+ };
+ block.putIntoLine = function (ep) {
+ var thisCol = this;
+
+ for (var i = 0; i < this.lines.length; i++) {
+ var line = this.lines[i];
+ if (line.isFree(ep.part.top, ep.part.height)) {
+ line.push(ep);
+ return i;
+ }
+ }
+
+ var line = [];
+ line.isFree = function(start, width) {
+ var end = start + width - 1;
+ var max = this.length;
+
+ for (var i = 0; i < max; i++) {
+ var e = this[i];
+ if (!(end < e.part.top || start > e.part.top + e.part.height - 1)) {
+ return false;
+ }
+ }
+
+ return true;
+ };
+
+ line.push(ep);
+
+ this.lines.push(line);
+
+ return this.lines.length - 1;
+
+ };
+
+ block.events.push(ep);
+ block.min = ep.part.top;
+ block.max = ep.part.top + ep.part.height;
+
+ this.blocks.push(block);
+
+ return this.blocks.length - 1;
+
+ };
+
+ result.putIntoLine = function(ep) {
+ var thisCol = this;
+
+ for (var i = 0; i < this.lines.length; i++) {
+ var line = this.lines[i];
+ if (line.isFree(ep.part.top, ep.part.height)) {
+ line.push(ep);
+ return i;
+ }
+ }
+
+ var line = [];
+ line.isFree = function(start, width) {
+ var end = start + width - 1;
+ var max = this.length;
+
+ for (var i = 0; i < max; i++) {
+ var e = this[i];
+ if (!(end < e.part.top || start > e.part.top + e.part.height - 1)) {
+ return false;
+ }
+ }
+
+ return true;
+ };
+
+ line.push(ep);
+
+ this.lines.push(line);
+
+ return this.lines.length - 1;
+ };
+
+ return result;
+
+ };
+
+ this._createDaysViewColumns = function() {
+ var columns = [];
+
+ var start = this.startDate.getDatePart();
+
+ var days = this.days;
+ switch (this.viewType) {
+ case "Day":
+ days = 1;
+ break;
+ case "Week":
+ days = 7;
+ var weekStarts = resolved._weekStarts();
+ start = start.firstDayOfWeek(weekStarts);
+ break;
+ case "WorkWeek":
+ days = 5;
+ start = start.firstDayOfWeek(1); // Monday
+ break;
+ }
+
+ if (this.heightSpec === 'BusinessHoursNoScroll') {
+ start = start.addHours(this.businessBeginsHour);
+ }
+
+ for (var i = 0; i < days; i++) {
+
+ var format = calendar.headerDateFormat ? calendar.headerDateFormat : resolved.locale().datePattern;
+
+ var column = {};
+ column.start = start.addDays(i);
+ column.name = column.start.toString(format, resolved.locale());
+
+ columns.push(column);
+ }
+
+ return columns;
+ };
+
+ this.visibleStart = function() {
+ if (calendar.viewType === "Resources") {
+ if (calendar._columns.length === 0) {
+ return DayPilot.Date.today();
+ }
+ var dates = calendar._columns.map(function(column) {
+ return column.start.getTime();
+ });
+ var min = Math.min.apply(null, dates);
+ return new DayPilot.Date(min);
+ }
+ return this._columns[0].start;
+ };
+
+ this.visibleEnd = function() {
+ if (calendar.viewType === "Resources") {
+ if (calendar._columns.length === 0) {
+ return DayPilot.Date.today().addDays(1);
+ }
+ var dates = calendar._columns.map(function(column) {
+ return column.start.getTime();
+ });
+ var max = Math.max.apply(null, dates);
+ return new DayPilot.Date(max).addDays(1);
+ }
+ var max = this._columns.length - 1;
+ return this._columns[max].start.addDays(1);
+ };
+
+
+ this._prefixCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ if (prefix) {
+ return prefix + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this._deleteEvents = function() {
+
+ if (this.elements.events) {
+
+ for (var i = 0; i < this.elements.events.length; i++) {
+ var div = this.elements.events[i];
+
+ var object = div.event;
+ if (object) {
+ object.calendar = null;
+ }
+
+ div.onclick = null;
+ div.onclickSave = null;
+ div.onmouseover = null;
+ div.onmouseout = null;
+ div.onmousemove = null;
+ div.onmousedown = null;
+
+ if (div.firstChild && div.firstChild.firstChild && div.firstChild.firstChild.tagName && div.firstChild.firstChild.tagName.toUpperCase() === 'IMG') {
+ var img = div.firstChild.firstChild;
+ img.onmousedown = null;
+ img.onmousemove = null;
+ img.onclick = null;
+
+ }
+
+ div.helper = null;
+ div.data = null;
+ div.event = null;
+
+ DayPilot.de(div);
+ }
+ }
+ this.elements.events = [];
+ };
+
+ this._drawEvent = function(e) {
+ /*
+
+ supported e.data properties:
+ * text
+ * html
+ * cssClass
+ * backColor
+ * barColor
+ * barBackColor
+ * toolTip
+ * barHidden
+ * borderColor
+ * fontColor
+
+ */
+
+
+ var data = e.cache || e.data;
+
+ var main = this.nav.events;
+
+ var rounded = true;
+ var radius = true;
+ var pixels = false;
+
+ // var borderColor = this.eventBorderColor;
+
+ var div = document.createElement("div");
+ div.style.position = 'absolute';
+ div.style.left = e.part.left + '%';
+ div.style.top = (e.part.top) + 'px';
+ div.style.width = e.part.width + '%';
+ div.style.height = Math.max(e.part.height, 2) + 'px';
+ div.style.overflow = 'hidden';
+
+ // TODO unify
+ div.data = e;
+ div.event = e;
+
+ div.unselectable = 'on';
+ div.style.MozUserSelect = 'none';
+ div.style.KhtmlUserSelect = 'none';
+
+ div.className = this._prefixCssClass("_event");
+
+ if (data.cssClass) {
+ DayPilot.Util.addClass(div, data.cssClass);
+ }
+
+ if (calendar.showToolTip && e.client.toolTip()) {
+ div.title = e.client.toolTip();
+ }
+
+ div.isFirst = e.part.start.getTime() === e.start().getTime();
+ div.isLast = e.part.end.getTime() === e.end().getTime();
+
+ div.onclick = this._eventClickDispatch;
+ DayPilot.re(div, "contextmenu", this._eventRightClickDispatch);
+
+ div.onmouseout = function(ev) {
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "none";
+ }
+ };
+
+ div.onmousemove = function (ev) {
+ // const
+ var resizeMargin = 5;
+
+ if (typeof(DayPilotCalendar) === 'undefined') {
+ return;
+ }
+
+ // position
+ var offset = DayPilot.mo3(div, ev);
+ if (!offset) {
+ return;
+ }
+
+ if (DayPilotCalendar.resizing || DayPilotCalendar.moving) {
+ return;
+ }
+
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "";
+ }
+
+ var isLastPart = this.isLast;
+
+ if (offset.y <= resizeMargin && e.client.resizeEnabled()) {
+ this.style.cursor = "n-resize";
+ this.dpBorder = 'top';
+ }
+ else if (this.offsetHeight - offset.y <= resizeMargin && e.client.resizeEnabled()) {
+ if (isLastPart) {
+ this.style.cursor = "s-resize";
+ this.dpBorder = 'bottom';
+ }
+ else {
+ this.style.cursor = 'not-allowed';
+ }
+ }
+ else if (!DayPilotCalendar.resizing && !DayPilotCalendar.moving) {
+ if (calendar.eventClickHandling !== 'Disabled') {
+ this.style.cursor = 'pointer';
+ }
+ else {
+ this.style.cursor = 'default';
+ }
+ }
+
+ };
+
+ div.onmousedown = function (ev) {
+ var button = ev.which || ev.button;
+
+ if ((this.style.cursor === 'n-resize' || this.style.cursor === 's-resize') && button === 1) {
+ // set
+ DayPilotCalendar.resizing = this;
+ DayPilot.Global.resizing = this;
+ DayPilotCalendar.originalMouse = DayPilot.mc(ev);
+ DayPilotCalendar.originalHeight = this.offsetHeight;
+ DayPilotCalendar.originalTop = this.offsetTop;
+
+ calendar.nav.top.style.cursor = this.style.cursor;
+
+ }
+ else if (button === 1 && e.client.moveEnabled()) {
+ DayPilotCalendar.moving = this;
+ DayPilot.Global.moving = this;
+ DayPilotCalendar.moving.event = this.event;
+ var helper = DayPilotCalendar.moving.helper = {};
+ helper.oldColumn = calendar._columns[this.data.part.dayIndex].id;
+ DayPilotCalendar.originalMouse = DayPilot.mc(ev);
+ DayPilotCalendar.originalTop = this.offsetTop;
+
+ var offset = DayPilot.mo3(this, ev);
+ if (offset) {
+ DayPilotCalendar.moveOffsetY = offset.y;
+ }
+ else {
+ DayPilotCalendar.moveOffsetY = 0;
+ }
+
+ // cursor
+ //document.body.style.cursor = 'move';
+ calendar.nav.top.style.cursor = 'move';
+ }
+
+ return false;
+ };
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = calendar._prefixCssClass("_event_inner");
+ inner.innerHTML = e.client.html();
+
+ if (data.borderColor === "darker" && data.backColor) {
+ inner.style.borderColor = DayPilot.ColorUtil.darker(data.backColor, 2);
+ }
+ else {
+ inner.style.borderColor = data.borderColor;
+ }
+
+ if (data.backColor) {
+ inner.style.background = data.backColor;
+ }
+
+ if (data.fontColor) {
+ inner.style.color = data.fontColor;
+ }
+
+ div.appendChild(inner);
+
+ // TODO
+ if (e.client.barVisible()) {
+ var height = e.part.height - 2;
+ var barTop = 100 * e.part.barTop / height; // %
+ var barHeight = Math.ceil(100 * e.part.barHeight / height); // %
+
+ var bar = document.createElement("div");
+ bar.setAttribute("unselectable", "on");
+ bar.className = this._prefixCssClass("_event_bar");
+ bar.style.position = "absolute";
+
+ if (data.barBackColor) {
+ bar.style.backgroundColor = data.barBackColor;
+ }
+
+ var barInner = document.createElement("div");
+ barInner.setAttribute("unselectable", "on");
+ barInner.className = this._prefixCssClass("_event_bar_inner");
+ barInner.style.top = barTop + "%";
+ if (0 < barHeight && barHeight <= 1) {
+ barInner.style.height = "1px";
+ }
+ else {
+ barInner.style.height = barHeight + "%";
+ }
+
+ if (data.barColor) {
+ barInner.style.backgroundColor = data.barColor;
+ }
+
+ bar.appendChild(barInner);
+ div.appendChild(bar);
+ }
+
+ if (e.client.deleteEnabled()) {
+ var del = document.createElement("div");
+ del.style.position = "absolute";
+ del.style.right = "2px";
+ del.style.top = "2px";
+ del.style.width = "17px";
+ del.style.height = "17px";
+ del.className = calendar._prefixCssClass("_event_delete");
+ del.onmousedown = function(ev) {
+ ev.stopPropagation();
+ };
+ del.onclick = function(ev) {
+ ev.stopPropagation();
+ var e = this.parentNode.event;
+ if (e) {
+ calendar._eventDeleteDispatch(e);
+ }
+ };
+ del.style.display = "none";
+ div.deleteIcon = del;
+ div.appendChild(del);
+ }
+
+ // areas
+ var areas = data.areas ? DayPilot.Areas.copy(data.areas) : [];
+
+/* var col = calendar._columnsBottom[e.part.dayIndex];
+
+ areas.forEach(function(area) {
+ if (area.start) {
+ area.top = calendar._getPixels(new DayPilot.Date(area.start), col.start).top - e.part.top - calendar._autoHiddenPixels();
+ }
+ if (area.end) {
+ area.bottom = e.part.top + e.part.height - calendar._getPixels(new DayPilot.Date(area.end), col.start).top - calendar._autoHiddenPixels();
+ }
+ });
+
+ if (e.client.deleteEnabled()) {
+ areas.push({"action":"JavaScript","v":"Hover","w":17,"h":17,"top":2,"right":2, "css": calendar._prefixCssClass("_event_delete"),"js":function(e) { calendar._eventDeleteDispatch(e); } });
+ }*/
+
+ DayPilot.Areas.attach(div, e, {"areas": areas});
+
+ if (main.rows[0].cells[e.part.dayIndex]) { // temporary fix for multirow header, but won't hurt later
+ var wrapper = main.rows[0].cells[e.part.dayIndex].firstChild;
+ wrapper.appendChild(div);
+
+ calendar._makeChildrenUnselectable(div);
+
+ //var e = new DayPilotCalendar.Event(div, calendar);
+ }
+
+ if (typeof calendar.onAfterEventRender === 'function') {
+ var args = {};
+ args.e = div.event;
+ args.div = div;
+
+ calendar.onAfterEventRender(args);
+ }
+
+ calendar.elements.events.push(div);
+ };
+
+ this._makeChildrenUnselectable = function(el) {
+ var c = (el && el.childNodes) ? el.childNodes.length : 0;
+ for (var i = 0; i < c; i++) {
+ try {
+ var child = el.childNodes[i];
+ if (child.nodeType === 1) {
+ child.unselectable = 'on';
+ this._makeChildrenUnselectable(child);
+ }
+ }
+ catch (e) {
+ }
+ }
+ };
+
+ this._drawEvents = function() {
+
+ //var start = new Date();
+
+ for (var i = 0; i < this._columns.length; i++) {
+ var col = this._columns[i];
+ if (!col.blocks) {
+ continue;
+ }
+
+ for (var m = 0; m < col.blocks.length; m++) {
+ var block = col.blocks[m];
+ for (var j = 0; j < block.lines.length; j++) {
+ var line = block.lines[j];
+
+ for(var k = 0; k < line.length; k++) {
+ var e = line[k];
+
+ e.part.width = 100 / block.lines.length;
+ e.part.left = e.part.width * j;
+
+ var isLastBlock = (j === block.lines.length - 1);
+ if (!isLastBlock) {
+ e.part.width = e.part.width * 1.5;
+ }
+
+ this._drawEvent(e);
+ }
+ }
+ }
+ }
+
+ //var end = new Date();
+ //var diff = end.getTime() - start.getTime();
+ };
+
+ this._drawTop = function() {
+
+ //this.nav.top = document.getElementById(this.id);
+ this.nav.top.innerHTML = '';
+
+ DayPilot.Util.addClass(this.nav.top, this._prefixCssClass("_main"));
+
+ this.nav.top.style.MozUserSelect = 'none';
+ this.nav.top.style.KhtmlUserSelect = 'none';
+ this.nav.top.style.position = 'relative';
+ this.nav.top.style.width = this.width ? this.width : '100%';
+
+ if (this.hideUntilInit) {
+ this.nav.top.style.visibility = 'hidden';
+ }
+
+ if (!this.visible) {
+ this.nav.top.style.display = "none";
+ }
+
+ this.nav.scroll = document.createElement("div");
+ this.nav.scroll.style.height = this._getScrollableHeight() + "px";
+
+ if (this.heightSpec === 'BusinessHours') {
+ this.nav.scroll.style.overflow = "auto";
+ }
+ else
+ {
+ this.nav.scroll.style.overflow = "hidden";
+ }
+
+ this.nav.scroll.style.position = "relative";
+
+ var header = this._drawTopHeaderDiv();
+ this.nav.top.appendChild(header);
+
+ // fixing the column alignment bug
+ // solved thanks to http://stackoverflow.com/questions/139000/div-with-overflowauto-and-a-100-wide-table-problem
+ this.nav.scroll.style.zoom = 1;
+
+ var wrap = this._drawScrollable();
+ this.nav.scrollable = wrap.firstChild;
+ this.nav.scroll.appendChild(wrap);
+ this.nav.top.appendChild(this.nav.scroll);
+
+ this.nav.scrollLayer = document.createElement("div");
+ this.nav.scrollLayer.style.position = 'absolute';
+ this.nav.scrollLayer.style.top = '0px';
+ this.nav.scrollLayer.style.left = '0px';
+ this.nav.top.appendChild(this.nav.scrollLayer);
+
+ this.nav.loading = document.createElement("div");
+ this.nav.loading.style.position = 'absolute';
+ this.nav.loading.style.top = '0px';
+ this.nav.loading.style.left = (this.hourWidth + 5) + "px";
+ // this.nav.loading.style.backgroundColor = this.loadingLabelBackColor;
+ // this.nav.loading.style.fontSize = this.loadingLabelFontSize;
+ // this.nav.loading.style.fontFamily = this.loadingLabelFontFamily;
+ // this.nav.loading.style.color = this.loadingLabelFontColor;
+ // this.nav.loading.style.padding = '2px';
+ this.nav.loading.innerHTML = calendar._xssTextHtml(calendar.loadingLabelText, calendar.loadingLabelHtml);
+ this.nav.loading.style.display = 'none';
+
+ this.nav.top.appendChild(this.nav.loading);
+
+ };
+
+ // used during full update
+ this._drawHourTable = function() {
+ // clear old hour table
+ if (!this.fasterDispose) DayPilot.pu(this.nav.hourTable);
+ this.nav.scrollable.rows[0].cells[0].innerHTML = '';
+ this.nav.hourTable = this._createHourTable();
+ this.nav.scrollable.rows[0].cells[0].appendChild(this.nav.hourTable);
+ };
+
+ // used during initial load only
+ this._drawScrollable = function() {
+ var zoom = document.createElement("div");
+ zoom.style.zoom = 1;
+ zoom.style.position = 'relative';
+ // zoom.onmousemove = calendar._mousemove;
+
+ var table = document.createElement("table");
+
+ table.cellSpacing = "0";
+ table.cellPadding = "0";
+ table.border = "0";
+ table.style.border = "0px none";
+ table.style.width = "100%";
+ table.style.position = 'absolute';
+
+ var r = table.insertRow(-1);
+
+ var c;
+ c = r.insertCell(-1);
+ c.valign = "top";
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ this.nav.hourTable = this._createHourTable();
+ c.appendChild(this.nav.hourTable);
+
+ c = r.insertCell(-1);
+ c.valign = "top";
+ c.width = "100%";
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ var wrap = document.createElement("div");
+ wrap.style.position = "relative";
+ c.appendChild(wrap);
+
+ wrap.appendChild(this._createEventsAndCells());
+ wrap.appendChild(this._createEventsTable());
+
+ zoom.appendChild(table);
+
+ this.nav.zoom = zoom;
+
+ return zoom;
+ };
+
+ this._createEventsAndCells = function() {
+ var table = document.createElement("table");
+
+ table.cellPadding = "0";
+ table.cellSpacing = "0";
+ table.border = "0";
+ table.style.width = "100%";
+ table.style.border = "0px none";
+ table.style.tableLayout = 'fixed';
+
+ this.nav.main = table;
+ this.nav.events = table;
+
+ return table;
+ };
+
+ this._createEventsTable = function() {
+ var table = document.createElement("table");
+
+ //table.style.position = "absolute";
+ table.style.top = "0px";
+ table.cellPadding = "0";
+ table.cellSpacing = "0";
+ table.border = "0";
+ table.style.position = "absolute";
+ table.style.width = "100%";
+ table.style.border = "0px none";
+ table.style.tableLayout = 'fixed';
+ //table.setAttribute("events", "true");
+
+ this.nav.events = table;
+ var create = true;
+ var columns = this._columns;
+ var cl = columns.length;
+
+ var r = (create) ? table.insertRow(-1) : table.rows[0];
+
+ for (var j = 0; j < cl; j++) {
+ var c = (create) ? r.insertCell(-1) : r.cells[j];
+
+ if (create) {
+
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+ c.style.height = '0px';
+ c.style.overflow = 'visible';
+ if (!calendar.rtl) {
+ c.style.textAlign = 'left';
+ }
+
+ // withpct
+ //c.style.width = (100.0 / columns.length) + "%";
+
+ var div = document.createElement("div");
+ div.style.marginRight = calendar.columnMarginRight + "px";
+ div.style.position = 'relative';
+ div.style.height = '1px';
+ div.style.marginTop = '-1px';
+
+ var selection = document.createElement("div");
+ c.selection = selection;
+
+ c.appendChild(div);
+ c.appendChild(selection);
+
+ }
+ }
+
+ return table;
+ };
+
+ this._createHourTable = function() {
+ var table = document.createElement("table");
+ table.cellSpacing = "0";
+ table.cellPadding = "0";
+ table.border = "0";
+ table.style.border = '0px none';
+ table.style.width = this.hourWidth + "px";
+ table.oncontextmenu = function() { return false; };
+
+ var hours = calendar._durationHours();
+ for (var i = 0; i < hours; i++) {
+ this._createHourRow(table, i);
+ }
+
+ return table;
+
+ };
+
+ this._createHourRow = function(table, i) {
+ var height = (this.cellHeight * 2);
+
+ var r = table.insertRow(-1);
+ r.style.height = height + "px";
+
+ var c = r.insertCell(-1);
+ c.valign = "bottom";
+ c.unselectable = "on";
+ c.style.cursor = "default";
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ var frame = document.createElement("div");
+ frame.style.position = "relative";
+ frame.className = this._prefixCssClass("_rowheader");
+ frame.style.width = this.hourWidth + "px";
+ frame.style.height = (height) + "px";
+ frame.style.overflow = 'hidden';
+ frame.unselectable = 'on';
+
+ var block = document.createElement("div");
+ block.className = this._prefixCssClass("_rowheader_inner");
+ block.unselectable = "on";
+
+ var text = document.createElement("div");
+ text.unselectable = "on";
+
+ var start = this.startDate.addHours(i).addHours(calendar._visibleStart());
+ var hour = start.getHours();
+
+ var am = hour < 12;
+ var timeFormat = resolved.timeFormat();
+ if (timeFormat === "Clock12Hours") {
+ hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ }
+
+ text.innerHTML = hour;
+
+ var span = document.createElement("span");
+ span.unselectable = "on";
+ span.className = this._prefixCssClass("_rowheader_minutes");
+
+ var sup;
+ if (timeFormat === "Clock12Hours") {
+ if (am) {
+ sup = "AM";
+ }
+ else {
+ sup = "PM";
+ }
+ }
+ else {
+ sup = "00";
+ }
+
+ span.innerHTML = sup;
+
+ text.appendChild(span);
+
+ block.appendChild(text);
+
+ frame.appendChild(block);
+
+ c.appendChild(frame);
+ };
+
+ this._getScrollableHeight = function() {
+ switch (this.heightSpec) {
+ case "Full":
+ return (24 * 2 * this.cellHeight);
+ case "BusinessHours":
+ var dHours = this._businessHoursSpan();
+ return dHours * this.cellHeight * 2;
+ case "BusinessHoursNoScroll":
+ var dHours = this._businessHoursSpan();
+ return dHours * this.cellHeight * 2;
+ default:
+ throw "DayPilot.Calendar: Unexpected 'heightSpec' value.";
+
+ }
+ };
+
+ this._updateCorner = function() {
+ var parent = calendar.nav.corner ? calendar.nav.corner.parentNode : null;
+ if (!parent) {
+ return;
+ }
+
+ parent.innerHTML = '';
+
+ var corner = this._drawCorner();
+ parent.appendChild(corner);
+ calendar.nav.corner = corner;
+
+ };
+
+ this._drawTopHeaderDiv = function() {
+ var header = document.createElement("div");
+ header.style.overflow = "auto";
+
+ var table = document.createElement("table");
+ table.cellPadding = "0";
+ table.cellSpacing = "0";
+ table.border = "0";
+ table.style.width = "100%";
+ table.style.borderCollapse = 'separate';
+ table.style.border = "0px none";
+
+ var r = table.insertRow(-1);
+
+ // corner
+ var c = r.insertCell(-1);
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ var corner = this._drawCorner();
+ c.appendChild(corner);
+ this.nav.corner = corner;
+
+ // top header
+ c = r.insertCell(-1);
+
+ c.style.width = "100%";
+ c.valign = "top";
+ c.style.position = 'relative'; // ref point
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+
+ this.nav.header = document.createElement("table");
+ this.nav.header.cellPadding = "0";
+ this.nav.header.cellSpacing = "0";
+ this.nav.header.border = "0";
+ this.nav.header.width = "100%";
+ this.nav.header.style.tableLayout = "fixed";
+ this.nav.header.oncontextmenu = function() { return false; };
+
+ var scrollbar = this.nav.scroll.style.overflow !== 'hidden';
+ c.appendChild(this.nav.header);
+
+ if (scrollbar) {
+ c = r.insertCell(-1);
+ c.unselectable = "on";
+
+ var inside = document.createElement("div");
+ inside.unselectable = "on";
+ inside.style.position = "relative";
+ inside.style.width = "16px";
+ inside.style.height = this.headerHeight + "px";
+ inside.className = this._prefixCssClass("_cornerright");
+
+ var inner = document.createElement("div");
+ inner.className = this._prefixCssClass('_cornerright_inner');
+ inside.appendChild(inner);
+
+ c.appendChild(inside);
+
+ this.nav.cornerRight = inside;
+ }
+
+ header.appendChild(table);
+
+ return header;
+
+ };
+
+ this._drawCorner = function() {
+ var wrap = document.createElement("div");
+ wrap.style.position = 'relative';
+ wrap.className = this._prefixCssClass("_corner");
+ wrap.style.width = this.hourWidth + "px";
+ wrap.style.height = this.headerHeight + "px";
+ wrap.oncontextmenu = function() { return false; };
+
+ var corner = document.createElement("div");
+ corner.unselectable = "on";
+ corner.className = this._prefixCssClass("_corner_inner");
+
+ wrap.appendChild(corner);
+
+ return wrap;
+ };
+
+ this._disposeMain = function() {
+ var table = this.nav.main;
+ table.root = null;
+ table.onmouseup = null;
+
+ for (var y = 0; y < table.rows.length; y++) {
+ var r = table.rows[y];
+ for (var x = 0; x < r.cells.length; x++) {
+ var c = r.cells[x];
+ c.root = null;
+
+ c.onmousedown = null;
+ c.onmousemove = null;
+ c.onmouseout = null;
+ c.onmouseup = null;
+ }
+ }
+
+ if (!this.fasterDispose) DayPilot.pu(table);
+ };
+
+ // draw time cells
+ this._drawMain = function() {
+
+ //DayPilotCalendar.selectedCells = [];
+ var cols = [];
+ var dates = [];
+
+ var table = this.nav.main;
+ var step = 30 * 60 * 1000;
+ var rowCount = this._rowCount();
+
+ var columns = calendar._columns;
+ //var create = !this.tableCreated || columns.length !== table.rows[0].cells.length || rowCount !== table.rows.length; // redraw only if number of columns changes
+ var create = true;
+
+ if (table) {
+ this._disposeMain();
+ }
+
+ while (table && table.rows && table.rows.length > 0 && create) {
+ if (!this.fasterDispose) {
+ DayPilot.pu(table.rows[0]);
+ }
+ table.deleteRow(0);
+ }
+
+ this.tableCreated = true;
+
+ var cl = columns.length;
+
+ var events = this.nav.events;
+ while (events && events.rows && events.rows.length > 0 && create) {
+ if (!this.fasterDispose) {
+ DayPilot.pu(events.rows[0]);
+ }
+ events.deleteRow(0);
+ }
+
+ var cl = columns.length;
+
+ var r = (create) ? events.insertRow(-1) : events.rows[0];
+
+ for (var j = 0; j < cl; j++) {
+ var c = (create) ? r.insertCell(-1) : r.cells[j];
+
+ if (create) {
+
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+ c.style.height = '0px';
+ c.style.overflow = 'visible';
+ if (!calendar.rtl) {
+ c.style.textAlign = 'left';
+ }
+
+ // withpct
+ //c.style.width = (100.0 / columns.length) + "%";
+
+ var div = document.createElement("div");
+ div.style.marginRight = calendar.columnMarginRight + "px";
+ div.style.position = 'relative';
+ div.style.height = '1px';
+ div.style.marginTop = '-1px';
+
+ var selection = document.createElement("div");
+ selection.style.position = "relative";
+ c.selection = selection;
+
+ c.appendChild(div);
+ c.appendChild(selection);
+
+ }
+ }
+
+ for (var i = 0; i < rowCount; i++) {
+ var r = (create) ? table.insertRow(-1) : table.rows[i];
+
+ if (create) {
+ r.style.MozUserSelect = 'none';
+ r.style.KhtmlUserSelect = 'none';
+ }
+
+ for (var j = 0; j < cl; j++) {
+ var col = this._columns[j];
+
+ var c = (create) ? r.insertCell(-1) : r.cells[j];
+
+ // always update
+ c.start = col.start.addTime(i * step);
+ c.end = c.start.addTime(step);
+ c.resource = col.id;
+
+ c.onmousedown = this._onCellMousedown;
+ // c.onmousemove = this._mousemove;
+ c.onmouseup = function() {
+ return false;
+ };
+
+ c.onclick = function() {
+ return false;
+ };
+
+ if (create) {
+ c.root = this;
+
+ c.style.padding = '0px';
+ c.style.border = '0px none';
+ c.style.verticalAlign = 'top';
+ c.style.height = calendar.cellHeight + 'px';
+ c.style.overflow = 'hidden';
+ c.unselectable = 'on';
+
+ var div = document.createElement("div");
+ div.unselectable = 'on';
+ div.style.height = calendar.cellHeight + "px";
+ div.style.position = "relative";
+ div.className = this._prefixCssClass("_cell");
+
+ var business = this._isBusinessCell(c.start, c.end);
+
+ var properties = {
+ "business": business,
+ "text": null,
+ "html": null,
+ "cssClass": null,
+ "backColor": null,
+ "backImage": null,
+ "backRepeat": null,
+ };
+
+ (function() {
+ if (typeof calendar.onBeforeCellRender === 'function') {
+ var args = {};
+ args.cell = {
+ "start": c.start,
+ "end": c.end,
+ "resource": c.resource,
+ "properties": properties
+ };
+ calendar.onBeforeCellRender(args);
+ }
+ })();
+
+
+ if (properties.business) {
+ DayPilot.Util.addClass(div, calendar._prefixCssClass("_cell_business"));
+ }
+
+ if (properties.cssClass) {
+ DayPilot.Util.addClass(div, properties.cssClass);
+ }
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = this._prefixCssClass("_cell_inner");
+
+ var html = DayPilot.Util.escapeTextHtml(properties.text, properties.html);
+ if (html) {
+ inner.innerHTML = html;
+ }
+
+ if (properties.backColor) {
+ inner.style.backgroundColor = properties.backColor;
+ }
+
+ if (properties.backImage) {
+ inner.style.backgroundImage = "url(" + properties.backImage + ")";
+ }
+
+ if (properties.backRepeat) {
+ inner.style.backgroundRepeat = properties.backRepeat;
+ }
+
+ div.appendChild(inner);
+ c.appendChild(div);
+
+ c.appendChild(div);
+
+ }
+ }
+ }
+
+ table.root = this;
+
+ calendar.nav.scrollable.onmousemove = function(ev) {
+ var ref = calendar.nav.scrollable;
+ calendar.coords = DayPilot.mo3(ref, ev);
+
+ var mousePos = DayPilot.mc(ev);
+
+ if (DayPilotCalendar.resizing) {
+ if (!DayPilotCalendar.resizingShadow) {
+ DayPilotCalendar.resizingShadow = calendar._createShadow(DayPilotCalendar.resizing, false, calendar.shadow);
+ }
+
+ var _step = calendar.cellHeight;
+ var _startOffset = 1;
+ var delta = (mousePos.y - DayPilotCalendar.originalMouse.y);
+
+ // TODO: clear
+ if (DayPilotCalendar.resizing.dpBorder === 'bottom') {
+ var newHeight = Math.floor(((DayPilotCalendar.originalHeight + DayPilotCalendar.originalTop + delta) + _step / 2) / _step) * _step - DayPilotCalendar.originalTop + _startOffset;
+
+ if (newHeight < _step)
+ newHeight = _step;
+
+ var max = calendar.nav.main.clientHeight;
+ if (DayPilotCalendar.originalTop + newHeight > max)
+ newHeight = max - DayPilotCalendar.originalTop;
+
+ // DayPilotCalendar.resizingShadow.style.height = (newHeight - 4) + 'px';
+ DayPilotCalendar.resizingShadow.style.height = (newHeight) + 'px';
+ }
+ else if (DayPilotCalendar.resizing.dpBorder === 'top') {
+ var newTop = Math.floor(((DayPilotCalendar.originalTop + delta - _startOffset) + _step / 2) / _step) * _step + _startOffset;
+
+ if (newTop < _startOffset) {
+ newTop = _startOffset;
+ }
+
+ if (newTop > DayPilotCalendar.originalTop + DayPilotCalendar.originalHeight - _step) {
+ newTop = DayPilotCalendar.originalTop + DayPilotCalendar.originalHeight - _step;
+ }
+
+ // var newHeight = DayPilotCalendar.originalHeight - (newTop - DayPilotCalendar.originalTop) - 4;
+ var newHeight = DayPilotCalendar.originalHeight - (newTop - DayPilotCalendar.originalTop);
+
+ if (newHeight < _step) {
+ newHeight = _step;
+ }
+ else {
+ DayPilotCalendar.resizingShadow.style.top = newTop + 'px';
+ }
+
+ DayPilotCalendar.resizingShadow.style.height = (newHeight) + 'px';
+ }
+ }
+ else if (DayPilotCalendar.moving) {
+
+ if (!calendar.coords) {
+ return;
+ }
+
+ if (!DayPilotCalendar.movingShadow) {
+
+ var minDistance = 3;
+ var mousePos = DayPilot.mc(ev);
+ var distance = Math.abs(mousePos.x - DayPilotCalendar.originalMouse.x) + Math.abs(mousePos.y - DayPilotCalendar.originalMouse.y);
+ if (distance <= minDistance) {
+ return;
+ }
+ // fixes the ie8 bug (incorrect offsetX and offsetY causes flickering during move if there are inline elements in the event
+ DayPilotCalendar.movingShadow = calendar._createShadow(DayPilotCalendar.moving, true, calendar.shadow);
+ DayPilotCalendar.movingShadow.style.width = (DayPilotCalendar.movingShadow.parentNode.offsetWidth + 1) + 'px';
+ }
+
+ var _step = calendar.cellHeight;
+ var _startOffset = 1;
+
+ var offset = DayPilotCalendar.moveOffsetY;
+ if (!offset) {
+ offset = _step / 2; // for external drag
+ }
+
+ var newTop = Math.floor(((calendar.coords.y - offset - _startOffset) + _step / 2) / _step) * _step + _startOffset;
+
+ if (newTop < _startOffset) {
+ newTop = _startOffset;
+ }
+
+ var main = calendar.nav.events;
+ var max = calendar.nav.main.clientHeight + _startOffset;
+
+ var height = parseInt(DayPilotCalendar.movingShadow.style.height); // DayPilotCalendar.moving.data.height
+ if (newTop + height > max) {
+ newTop = max - height;
+ }
+
+ DayPilot.Util.addClass(DayPilotCalendar.moving, calendar._prefixCssClass("_event_moving_source"));
+
+ DayPilotCalendar.movingShadow.parentNode.style.display = 'none';
+ DayPilotCalendar.movingShadow.style.top = newTop + 'px';
+ DayPilotCalendar.movingShadow.parentNode.style.display = '';
+
+ var colWidth = main.clientWidth / main.rows[0].cells.length;
+ var column = Math.floor((calendar.coords.x - 45) / colWidth);
+
+ if (column < 0) {
+ column = 0;
+ }
+
+ if (column < main.rows[0].cells.length && column >= 0 && DayPilotCalendar.movingShadow.column !== column) {
+ DayPilotCalendar.movingShadow.column = column;
+ DayPilotCalendar.moveShadow(main.rows[0].cells[column]);
+ }
+ }
+ else if (DayPilotCalendar.selecting) {
+
+ var mousePos = DayPilot.mc(ev);
+
+ var cellcoords = calendar._table.getCellCoords();
+ var x = DayPilotCalendar.column;
+ var cell = calendar.nav.main.rows[cellcoords.y].cells[x];
+
+
+ // clean
+ // calendar.clearSelection();
+
+ // new selected cells
+ if (mousePos.y < DayPilotCalendar.firstMousePos.y) {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsBelow(cell);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.selectedCells[0];
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.firstSelected;
+ }
+ else {
+ DayPilotCalendar.selectedCells = DayPilotCalendar.getCellsAbove(cell);
+ DayPilotCalendar.topSelectedCell = DayPilotCalendar.firstSelected;
+ DayPilotCalendar.bottomSelectedCell = DayPilotCalendar.selectedCells[0];
+ }
+
+ calendar._activateSelection();
+
+ }
+ };
+ calendar.nav.scrollable.style.display = '';
+ };
+
+ this._isBusinessCell = function(start, end) {
+ if (this.businessBeginsHour < this.businessEndsHour)
+ {
+ return !(start.getHours() < this.businessBeginsHour || start.getHours() >= this.businessEndsHour || start.getDayOfWeek() === 6 || start.getDayOfWeek() === 0);
+ }
+
+ if (start.getHours() >= this.businessBeginsHour)
+ {
+ return true;
+ }
+
+ if (start.getHours() < this.businessEndsHour)
+ {
+ return true;
+ }
+
+ return false;
+ };
+
+ this._disposeHeader = function() {
+ var table = this.nav.header;
+ if (table && table.rows) {
+ for(var y = 0; y < table.rows.length; y++) {
+ var r = table.rows[y];
+ for (var x = 0; x < r.cells.length; x++) {
+ var c = r.cells[x];
+ c.onclick = null;
+ c.onmousemove = null;
+ c.onmouseout = null;
+ }
+ }
+ }
+ if (!this.fasterDispose) DayPilot.pu(table);
+ };
+
+ this._drawHeaderRow = function(create) {
+
+ // column headers
+ var r = (create) ? this.nav.header.insertRow(-1) : this.nav.header.rows[0];
+
+ var columns = this._columns;
+
+ var len = columns.length;
+
+ function drawHeaderCell(i) {
+ var data = columns[i];
+
+ var cell = (create) ? r.insertCell(-1) : r.cells[i];
+ cell.data = data;
+
+ cell.style.overflow = 'hidden';
+ cell.style.padding = '0px';
+ cell.style.border = '0px none';
+ cell.style.height = (calendar.headerHeight) + "px";
+
+ cell.onclick = calendar._headerClickDispatch;
+
+ var div = (create) ? document.createElement("div") : cell.firstChild;
+ var inner;
+
+ if (create) {
+ div.unselectable = 'on';
+ div.style.MozUserSelect = 'none';
+ div.style.cursor = 'default';
+ div.style.position = 'relative';
+ div.className = calendar._prefixCssClass('_colheader');
+ div.style.height = calendar.headerHeight + "px";
+
+ if (!calendar.headerTextWrappingEnabled) {
+ div.style.whiteSpace = 'nowrap';
+ }
+
+ inner = document.createElement("div");
+ inner.className = calendar._prefixCssClass('_colheader_inner');
+ inner.unselectable = 'on';
+
+ div.appendChild(inner);
+ cell.appendChild(div);
+ }
+ else {
+ inner = div.firstChild;
+ }
+
+ var args = {};
+ args.header = {};
+ args.header.cssClass = null;
+ args.header.verticalAlignment = "center";
+ args.column = calendar._createColumn(data, calendar);
+
+ if (typeof calendar.onBeforeHeaderRender === 'function') {
+ DayPilot.Util.copyProps(data, args.header, ['id', 'start', 'name', 'html', 'backColor', 'toolTip', 'areas']);
+ calendar.onBeforeHeaderRender(args);
+ DayPilot.Util.copyProps(args.header, data, ['html', 'backColor', 'toolTip', 'areas', 'cssClass', 'verticalAlignment']);
+ }
+
+ if (data.toolTip) {
+ inner.title = data.toolTip;
+ }
+
+ if (data.cssClass) {
+ DayPilot.Util.addClass(div, data.cssClass);
+ }
+
+ if (data.backColor) {
+ inner.style.background = data.backColor;
+ }
+
+ if (data.areas) {
+ DayPilot.Areas.attach(div, data);
+ }
+
+ var va = data.verticalAlignment;
+ if (va) {
+ inner.style.display = "flex";
+ switch (va) {
+ case "center":
+ inner.style.alignItems = "center";
+ break;
+ case "top":
+ inner.style.alignItems = "flex-start";
+ break;
+ case "bottom":
+ inner.style.alignItems = "flex-end";
+ break;
+ }
+ }
+
+ var text = div.firstChild;
+ text.innerHTML = calendar._xssTextHtml(data.name, data.html);
+ }
+
+ for (var i = 0; i < len; i++) {
+ drawHeaderCell(i);
+ }
+ };
+
+ this._headerClickDispatch = function(ev) {
+
+ var handling = calendar.headerClickHandling;
+
+ if (handling === "Disabled") {
+ return;
+ }
+
+ var data = this.data;
+ var c = calendar._createColumn(data);
+
+ var args = {};
+ args.header = {};
+ args.header.id = data.id;
+ args.header.name = data.name;
+ args.header.start = data.start;
+ args.column = c;
+ args.originalEvent = ev;
+ args.shift = ev.shiftKey;
+ args.ctrl = ev.ctrlKey;
+ args.meta = ev.metaKey;
+
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onHeaderClick === 'function') {
+ calendar.onHeaderClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ if (typeof calendar.onHeaderClicked === 'function') {
+ calendar.onHeaderClicked(args);
+ }
+
+ };
+
+ this._createColumn = function(data) {
+ return new DayPilot.CalendarColumn(data, calendar);
+ };
+
+ this._widthUnit = function() {
+ if (this.width && this.width.indexOf("px") !== -1) {
+ return "Pixel";
+ }
+ return "Percentage";
+ };
+
+ this._drawHeader = function() {
+
+ var header = this.nav.header;
+ var create = true;
+
+ var columns = this._columns;
+ var len = columns.length;
+
+ while (this.headerCreated && header && header.rows && header.rows.length > 0 && create) {
+ if (!this.fasterDispose) DayPilot.pu(header.rows[0]);
+ header.deleteRow(0);
+ }
+
+ this.headerCreated = true;
+
+ if (!create) {
+ // corner
+ var corner = calendar.nav.corner;
+ if (!this.fasterDispose) DayPilot.pu(corner.firstChild);
+ }
+
+ this._drawHeaderRow(create);
+ };
+
+ this.loadingStart = function() {
+ if (this.loadingLabelVisible) {
+ this.nav.loading.innerHTML = this.loadingLabelText;
+ this.nav.loading.style.top = (this.headerHeight + 5) + "px";
+ this.nav.loading.style.display = '';
+ }
+ };
+
+ this.commandCallBack = function(command, data) {
+ var params = {};
+ params.command = command;
+ this._callBack2('Command', data, params);
+ };
+
+ this.loadingStop = function(msg) {
+ if (this.callbackTimeout) {
+ window.clearTimeout(this.callbackTimeout);
+ }
+
+ this.nav.loading.style.display = 'none';
+ };
+
+ this._enableScrolling = function() {
+
+ var scrollDiv = this.nav.scroll;
+
+ if (!scrollDiv.onscroll) {
+ scrollDiv.onscroll = function() {
+ calendar._saveScrollHour();
+ };
+ }
+
+ var scrollpos = (typeof this._config.scrollpos !== 'undefined') ? this._config.scrollpos : this.initScrollPos;
+
+ if (!scrollpos) {
+ return;
+ }
+
+
+ if (scrollpos === 'Auto') {
+ if (this.heightSpec === "BusinessHours") {
+ scrollpos = 2 * this.cellHeight * this.businessBeginsHour;
+ }
+ else {
+ scrollpos = 0;
+ }
+ }
+
+ scrollDiv.root = this;
+
+ // initial position
+ if (scrollDiv.scrollTop === 0) {
+ scrollDiv.scrollTop = scrollpos;
+ }
+ };
+
+ this.callbackError = function (result, context) {
+ alert("Error!\r\nResult: " + result + "\r\nContext:" + context);
+ };
+
+ this._fixScrollHeader = function() {
+ var w = DayPilot.sw(this.nav.scroll);
+ var d = this.nav.cornerRight;
+ if (d) {
+ d.style.width = w + 'px';
+ }
+ };
+
+ this._registerGlobalHandlers = function() {
+ if (!DayPilotCalendar.globalHandlers) {
+ DayPilotCalendar.globalHandlers = true;
+ DayPilot.re(document, 'mouseup', DayPilotCalendar.gMouseUp);
+ DayPilot.re(window, 'unload', DayPilotCalendar.gUnload);
+ }
+ };
+
+ this.events = {};
+ //this.events.list = [];
+
+ this.events.add = function(e) {
+
+ var data = null;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ data = e;
+ }
+ else {
+ throw "DayPilot.Calendar.events.add() expects an object or DayPilot.Event instance.";
+ }
+
+ if (!calendar.events.list) {
+ calendar.events.list = [];
+ }
+
+ calendar.events.list.push(data);
+ calendar._update({"eventsOnly": true});
+ calendar._angular.notify();
+ };
+
+
+ this.events.find = function(id) {
+ if (!calendar.events.list) {
+ return null;
+ }
+
+ if (typeof id === "function") {
+ var fn = id;
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var e = new DayPilot.Event(calendar.events.list[i], calendar);
+ if (fn(e)) {
+ return e;
+ }
+ }
+ return null;
+ }
+
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var data = calendar.events.list[i];
+ if (data.id === id) {
+ return new DayPilot.Event(data, calendar);
+ }
+ }
+ return null;
+ };
+
+ this.events.update = function(e) {
+ if (e instanceof DayPilot.Event) {
+ e.commit();
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ var index = DayPilot.indexOf(calendar.events.list, target.data);
+ calendar.events.list.splice(index, 1, e);
+ }
+ }
+
+ calendar._update({"eventsOnly": true});
+ calendar._angular.notify();
+ };
+
+ this.events.remove = function(e) {
+ var data;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ data = target.data;
+ }
+ }
+ else if (typeof e === "string" || typeof e === "number") {
+ var target = calendar.events.find(e);
+ if (target) {
+ data = target.data;
+ }
+ }
+
+ var index = DayPilot.indexOf(calendar.events.list, data);
+ calendar.events.list.splice(index, 1);
+ calendar._update({"eventsOnly": true});
+ calendar._angular.notify();
+ };
+
+ this.events.load = function(url, success, error) {
+ var onError = function(args) {
+ var largs = {};
+ largs.exception = args.exception;
+ largs.request = args.request;
+
+ if (typeof error === 'function') {
+ error(largs);
+ }
+ };
+
+ var onSuccess = function(args) {
+ var r = args.request;
+ var data;
+
+ // it's supposed to be JSON
+ try {
+ data = JSON.parse(r.responseText);
+ }
+ catch (e) {
+ var fargs = {};
+ fargs.exception = e;
+ onError(fargs);
+ return;
+ }
+
+ if (DayPilot.isArray(data)) {
+ var sargs = {};
+ sargs.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+ sargs.data = data;
+ if (typeof success === "function") {
+ success(sargs);
+ }
+
+ if (sargs.preventDefault.value) {
+ return;
+ }
+
+ calendar.events.list = data;
+ if (calendar._initialized) {
+ calendar._update({"eventsOnly": true});
+ }
+ }
+ };
+
+ var usePost = calendar.eventsLoadMethod && calendar.eventsLoadMethod.toUpperCase() === "POST";
+
+ if (usePost) {
+ DayPilot.Http.ajax({
+ "method": "POST",
+ "data": { "start": calendar.visibleStart().toString(), "end": calendar.visibleEnd().toString()},
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ else {
+ var fullUrl = url;
+ var queryString = "start=" + calendar.visibleStart().toString() + "&end=" + calendar.visibleEnd().toString();
+ if (fullUrl.indexOf("?") > -1) {
+ fullUrl += "&" + queryString;
+ }
+ else {
+ fullUrl += "?" + queryString;
+ }
+
+ DayPilot.Http.ajax({
+ "method": "GET",
+ "url": fullUrl,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+
+ };
+
+ this._updateTheme = function() {
+ // manually update theme for elements that are not redrawn during update
+ //return;
+
+ var needsUpdate = calendar.nav.top.className !== calendar._prefixCssClass("_main");
+
+ if (!needsUpdate) {
+ return;
+ }
+
+ calendar.nav.top.className = calendar._prefixCssClass("_main");
+ var corner = calendar.nav.corner;
+ corner.className = calendar._prefixCssClass("_corner");
+ corner.firstChild.className = calendar._prefixCssClass("_corner_inner");
+
+ var cr = calendar.nav.cornerRight;
+ if (cr) {
+ cr.className = calendar._prefixCssClass("_cornerright");
+ cr.firstChild.className = calendar._prefixCssClass("_cornerright_inner");
+ }
+ };
+
+ this.update = function(options) {
+
+ if (calendar._disposed) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Calendar instance that has been disposed.");
+ }
+
+ calendar._loadOptions(options);
+ calendar._update();
+ };
+
+ this._update = function(args) {
+
+ if (!this._initialized) {
+ return;
+ }
+
+ var args = args || {};
+ var full = !args.eventsOnly;
+
+ calendar._prepareVariables();
+ calendar._deleteEvents();
+
+ // reset after drag and drop/deleting
+ calendar.nav.top.style.cursor = "auto";
+
+ // calendar._show();
+
+ if (full) {
+ calendar._prepareColumns();
+ calendar._drawHeader();
+ calendar._drawMain();
+ calendar._drawHourTable();
+ calendar._updateHeight();
+ calendar._updateCorner();
+ calendar._fixScrollHeader();
+ calendar._updateTheme();
+
+ calendar._restoreScrollHour();
+ }
+
+ calendar._loadEvents();
+ calendar._updateHeaderHeight();
+
+ calendar._drawEvents();
+ calendar.clearSelection();
+
+ if (this.visible) {
+ this.show();
+ }
+ else {
+ this.hide();
+ }
+ };
+
+
+ this._specialHandling = null;
+ this._loadOptions = function(options) {
+ if (!options) {
+ return;
+ }
+ var specialHandling = {
+ "events": {
+ "preInit": function() {
+ var events = this.data || [];
+ if (DayPilot.isArray(events.list)) {
+ calendar.events.list = events.list;
+ }
+ else {
+ calendar.events.list = events;
+ }
+ }
+ },
+ "columns": {
+ "preInit": function() {
+ calendar.columns.list = this.data;
+ }
+ }
+ };
+ this._specialHandling = specialHandling;
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ var item = specialHandling[name];
+ item.data = options[name];
+ if (item.preInit) {
+ item.preInit();
+ }
+ }
+ else {
+ calendar[name] = options[name];
+ }
+ }
+
+ };
+
+ this._postInit = function() {
+ var specialHandling = this._specialHandling;
+ for (var name in specialHandling) {
+ var item = specialHandling[name];
+ if (item.postInit) {
+ item.postInit();
+ }
+ }
+ };
+
+ this._loadTop = function() {
+ if (this.id && this.id.tagName) {
+ this.nav.top = this.id;
+ }
+ else if (typeof this.id === "string") {
+ this.nav.top = document.getElementById(this.id);
+ if (!this.nav.top) {
+ throw "DayPilot.Calendar: The placeholder element not found: '" + id + "'.";
+ }
+ }
+ else {
+ throw "DayPilot.Calendar() constructor requires the target element or its ID as a parameter";
+ }
+ };
+
+ this._cache = {};
+ this._cache.events = [];
+
+ this._doBeforeEventRender = function(i) {
+ var cache = this._cache.events;
+ var data = this.events.list[i];
+ var evc = {};
+
+ // make a copy
+ for (var name in data) {
+ evc[name] = data[name];
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ var args = {};
+ args.control = calendar;
+ args.data = evc;
+ this.onBeforeEventRender(args);
+ }
+
+ cache[i] = evc;
+
+ };
+
+ this._loadEvents = function() {
+
+ var events = this.events.list;
+
+ calendar._cache.events = [];
+
+ if (!events) {
+ return;
+ }
+
+ if (!DayPilot.isArray(events)) {
+ throw new DayPilot.Exception("DayPilot.Calendar.events.list expects an array object. You supplied: " + (typeof events));
+ }
+
+ var length = events.length;
+ var duration = 24 * 60 * 60 * 1000;
+
+ this.cache.pixels = {};
+
+ var loadCache = [];
+
+ this.scrollLabels = [];
+
+ this.minStart = 10000;
+ this.maxEnd = 0;
+
+ for (var i = 0; i < length; i++) {
+ var e = events[i];
+ var edata = e;
+
+ if (typeof edata !== "object") {
+ throw new DayPilot.Exception("Event data item must be an object");
+ }
+ if (!edata.start) {
+ throw new DayPilot.Exception("Event data item must specify 'start' property");
+ }
+ if (!edata.end) {
+ throw new DayPilot.Exception("Event data item must specify 'end' property");
+ }
+
+ if (edata instanceof DayPilot.Event) {
+ throw new DayPilot.Exception("DayPilot.Calendar: DayPilot.Event object detected in events.list array. Use raw event data instead.");
+ }
+
+ e.start = new DayPilot.Date(e.start);
+ e.end = new DayPilot.Date(e.end);
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ for (var i = 0; i < length; i++) {
+ this._doBeforeEventRender(i);
+ }
+ }
+
+ for(var i = 0; i < this._columns.length; i++) {
+
+ var scroll = {};
+ scroll.minEnd = 1000000;
+ scroll.maxStart = -1;
+ this.scrollLabels.push(scroll);
+
+ var col = this._columns[i];
+ col.events = [];
+ col.lines = [];
+ col.blocks = [];
+
+ var colStart = new DayPilot.Date(col.start);
+ var colStartTicks = colStart.getTime();
+ var colEnd = colStart.addTime(duration);
+ var colEndTicks = colEnd.getTime();
+
+ for (var j = 0; j < length; j++) {
+ if (loadCache[j]) {
+ continue;
+ }
+
+ var e = events[j];
+
+ var start = e.start;
+ var end = e.end;
+
+ var startTicks = start.getTime();
+ var endTicks = end.getTime();
+
+ if (endTicks < startTicks) { // skip invalid events
+ continue;
+ }
+
+ // belongs here
+ var belongsHere = !(endTicks <= colStartTicks || startTicks >= colEndTicks);
+ if (calendar.viewType === "Resources") {
+ belongsHere = belongsHere && col.id === e.resource;
+ }
+
+ if (belongsHere) {
+ var ep = new DayPilot.Event(e, calendar); // event part
+ ep.part.dayIndex = i;
+ ep.part.start = colStartTicks < startTicks ? e.start : colStart;
+ ep.part.end = colEndTicks > endTicks ? e.end : colEnd;
+
+ var partStartPixels = this.getPixels(ep.part.start, col.start);
+ var partEndPixels = this.getPixels(ep.part.end, col.start);
+
+ var top = partStartPixels.top;
+ var bottom = partEndPixels.top;
+
+ // events in the hidden areas
+ if (top === bottom && (partStartPixels.cut || partEndPixels.cut)) {
+ continue;
+ }
+
+ var boxBottom = partEndPixels.boxBottom;
+
+ ep.part.top = Math.floor(top / this.cellHeight) * this.cellHeight + 1;
+ ep.part.height = Math.max(Math.ceil(boxBottom / this.cellHeight) * this.cellHeight - ep.part.top, this.cellHeight - 1) + 1;
+ ep.part.barTop = Math.max(top - ep.part.top - 1, 0); // minimum 0
+ ep.part.barHeight = Math.max(bottom - top - 2, 1); // minimum 1
+
+ var start = ep.part.top;
+ var end = ep.part.top + ep.part.height;
+
+ if (start > scroll.maxStart) {
+ scroll.maxStart = start;
+ }
+ if (end < scroll.minEnd) {
+ scroll.minEnd = end;
+ }
+
+ if (start < this.minStart) {
+ this.minStart = start;
+ }
+ if (end > this.maxEnd) {
+ this.maxEnd = end;
+ }
+ col.events.push(ep);
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ ep.cache = this._cache.events[j];
+ }
+
+ if (ep.part.start.getTime() === startTicks && ep.part.end.getTime() === endTicks) {
+ loadCache[j] = true;
+ }
+ }
+ }
+
+ }
+
+
+ // sort events inside rows
+ for (var i = 0; i < this._columns.length; i++) {
+ var col = this._columns[i];
+ col.events.sort(this._eventComparer);
+
+ // put into lines
+ for (var j = 0; j < col.events.length; j++) {
+ var e = col.events[j];
+ col.putIntoBlock(e);
+ }
+
+ for (var j = 0; j < col.blocks.length; j++) {
+ var block = col.blocks[j];
+ block.events.sort(this._eventComparer);
+ for (var k = 0; k < block.events.length; k++ ) {
+ var e = block.events[k];
+ block.putIntoLine(e);
+ }
+ }
+ }
+ };
+
+ this._eventComparer = function(a, b) {
+ if (!a || !b || !a.start || !b.start) {
+ return 0; // no sorting, invalid arguments
+ }
+
+ var byStart = a.start().getTime() - b.start().getTime();
+ if (byStart !== 0) {
+ return byStart;
+ }
+
+ var byEnd = b.end().getTime() - a.end().getTime(); // desc
+ return byEnd;
+ };
+
+ this.debug = function(msg, append) {
+ if (!this.debuggingEnabled) {
+ return;
+ }
+
+ if (!calendar.debugMessages) {
+ calendar.debugMessages = [];
+ }
+ calendar.debugMessages.push(msg);
+
+ if (typeof console !== 'undefined') {
+ console.log(msg);
+ }
+ };
+
+ this.getPixels = function(date, start) {
+ if (!start) start = this.startDate;
+
+ var startTicks = start.getTime();
+ var ticks = date.getTime();
+
+ var cache = this.cache.pixels[ticks + "_" + startTicks];
+ if (cache) {
+ return cache;
+ }
+
+ startTicks = start.getTime();
+
+ var boxTicks = 30 * 60 * 1000;
+ var topTicks = ticks - startTicks;
+ var boxOffsetTicks = topTicks % boxTicks;
+
+ var boxStartTicks = topTicks - boxOffsetTicks;
+ var boxEndTicks = boxStartTicks + boxTicks;
+ if (boxOffsetTicks === 0) {
+ boxEndTicks = boxStartTicks;
+ }
+
+ // it's linear scale so far
+ var result = {};
+ result.cut = false;
+ result.top = this._ticksToPixels(topTicks);
+ result.boxTop = this._ticksToPixels(boxStartTicks);
+ result.boxBottom = this._ticksToPixels(boxEndTicks);
+
+ this.cache.pixels[ticks + "_" + startTicks] = result;
+
+ return result;
+ };
+
+ this._ticksToPixels = function(ticks) {
+ return Math.floor( (this.cellHeight * ticks) / (1000 * 60 * 30) );
+ };
+
+ this._prepareVariables = function() {
+ this.startDate = new DayPilot.Date(this.startDate).getDatePart();
+ };
+
+ this._updateHeaderHeight = function() {
+ if (this.nav.corner) {
+ this.nav.corner.style.height = this.headerHeight + "px";
+ }
+ };
+
+ this._updateHeight = function() {
+ var sh = this._getScrollableHeight();
+ if (this.nav.scroll && sh > 0) {
+ this.nav.scroll.style.height = sh + "px";
+ }
+ };
+
+ this._angular = {};
+ this._angular.scope = null;
+ this._angular.notify = function() {
+ if (calendar._angular.scope) {
+ calendar._angular.scope["$apply"]();
+ }
+ };
+ this._angular.apply = function(f) {
+ // autoapply has been disabled
+ f();
+
+ /*
+ if (calendar.angularAutoApply && calendar._angular.scope) {
+ calendar._angular.scope["$apply"](f);
+ }
+ else {
+ f();
+ }
+ */
+ };
+
+ this._saveScrollHour = function() {
+ if (!calendar.nav.scroll) {
+ return;
+ }
+ var top = calendar.nav.scroll.scrollTop;
+ var pos = top / (2*calendar.cellHeight);
+ calendar._config.scrollHour = pos;
+ };
+
+ this._restoreScrollHour = function() {
+ var scrollpos = 0;
+ if (typeof calendar._config.scrollHour === "number") {
+ scrollpos = 2 * calendar.cellHeight * calendar._config.scrollHour;
+ //calendar._config.scrollHour = null;
+ }
+ else {
+ if (calendar.initScrollPos === 'Auto') {
+ if (this.heightSpec === "BusinessHours") {
+ scrollpos = 2 * this.cellHeight * this.businessBeginsHour;
+ }
+ else {
+ scrollpos = 0;
+ }
+ }
+ }
+
+ calendar.nav.scroll.scrollTop = scrollpos;
+ };
+
+ this._loadFromServer = function() {
+ // make sure it has a place to ask
+ if (this.backendUrl || typeof WebForm_DoCallback === 'function') {
+ return (typeof calendar.events.list === 'undefined') || (!calendar.events.list);
+ }
+ else {
+ return false;
+ }
+ };
+
+ this._show = function() {
+ if (this.nav.top.style.visibility === 'hidden') {
+ this.nav.top.style.visibility = 'visible';
+ }
+ };
+
+ this.show = function() {
+ calendar.visible = true;
+ calendar.nav.top.style.display = '';
+ this._fixScrollHeader();
+ };
+
+ this.hide = function() {
+ calendar.visible = false;
+ calendar.nav.top.style.display = 'none';
+ };
+ this._initShort = function() {
+ this._prepareVariables();
+ this._prepareColumns();
+ this._drawTop();
+ this._drawHeader();
+ this._drawMain();
+ this._fixScrollHeader();
+ this._enableScrolling();
+ this._registerGlobalHandlers();
+ DayPilotCalendar.register(this);
+
+ this._waitForVisibility();
+ this._callBack2('Init');
+ };
+
+ this._config = {};
+
+ this._saveConfig = function() {
+ this._config.themes = [];
+ this._config.themes.push(this.theme || this.cssClassPrefix);
+ };
+
+ this._clearThemes = function() {
+ var themes = this._config.themes;
+ for (var i = 0; i < themes.length; i++) {
+ var theme = themes[i];
+ DayPilot.Util.removeClass(this.nav.top, theme + "_main");
+ }
+ this._config.themes = [];
+ };
+
+ this._doAfterRender = function() {
+ this.afterRender(null, false);
+ if (typeof this.onAfterRender === "function") {
+ var args = {};
+ args.isCallBack = false;
+ this.onAfterRender(args);
+ }
+ };
+
+ this._doInit = function() {
+ if (typeof this.onInit === "function" && !this._onInitCalled) {
+ this._onInitCalled = true;
+ var args = {};
+ this.onInit(args);
+ }
+ };
+
+ this._visible = function() {
+ var el = calendar.nav.top;
+ if (!el) {
+ return false;
+ }
+ return el.offsetWidth > 0 && el.offsetHeight > 0;
+ };
+
+ this._waitForVisibility = function() {
+ var visible = calendar._visible;
+
+ if (!visible()) {
+ calendar._visibilityInterval = setInterval(function() {
+ if (visible()) {
+ calendar._enableScrolling();
+ calendar._fixScrollHeader();
+ clearInterval(calendar._visibilityInterval);
+ }
+ }, 100);
+ }
+ };
+
+ this._xssTextHtml = function(text, html) {
+
+ if (calendar._resolved._xssProtectionEnabled()) {
+ return DayPilot.Util.escapeTextHtml(text, html);
+ }
+
+ if (!DayPilot.Util.isNullOrUndefined(html)) {
+ return html;
+ }
+ if (DayPilot.Util.isNullOrUndefined(text)) {
+ return "";
+ }
+ return text;
+ };
+
+
+ this.internal = {};
+ this.internal.loadOptions = calendar._loadOptions;
+ this.internal.xssTextHtml = calendar._xssTextHtml;
+
+ this.init = function() {
+ this._loadTop();
+
+ var loadFromServer = this._loadFromServer();
+
+ this._saveConfig();
+
+ if (loadFromServer) {
+ this._initShort();
+ return;
+ }
+
+ this._prepareVariables();
+ this._prepareColumns();
+
+ this._loadEvents();
+
+ this._drawTop();
+ this._drawHeader();
+ this._drawMain();
+
+ this._show();
+
+ this._fixScrollHeader();
+ this._enableScrolling();
+ this._registerGlobalHandlers();
+ DayPilotCalendar.register(this);
+
+ if (this.events) { // are events available?
+ this._updateHeaderHeight();
+ this._drawEvents();
+ }
+
+ this._doAfterRender();
+ this._doInit();
+ this._waitForVisibility();
+ this._initialized = true;
+
+ return this;
+ };
+
+ this.Init = this.init;
+
+ this._loadOptions(options);
+ };
+
+ DayPilot.CalendarColumn = function(col, calendar) {
+ var column = this;
+
+ column.id = col.id;
+ column.name = col.name;
+ column.data = col.data;
+ column.start = new DayPilot.Date(col.start);
+ column.calendar = calendar;
+
+ column.toJSON = function() {
+ var json = {};
+ json.id = this.id;
+ if (this.start) {
+ json.start = this.start.toString();
+ }
+ json.name = this.name;
+ return json;
+ };
+ };
+
+
+ // publish the API
+ DayPilot.Calendar = DayPilotCalendar.Calendar;
+
+ // jQuery plugin
+ if (typeof jQuery !== 'undefined') {
+ (function( $ ){
+ $.fn.daypilotCalendar = function(options) {
+ var first = null;
+ var j = this.each(function() {
+ if (this.daypilot) { // already initialized
+ return;
+ };
+
+ var daypilot = new DayPilot.Calendar(this.id);
+ this.daypilot = daypilot;
+ for (name in options) {
+ daypilot[name] = options[name];
+ }
+ daypilot.init();
+ if (!first) {
+ first = daypilot;
+ }
+ });
+ if (this.length === 1) {
+ return first;
+ }
+ else {
+ return j;
+ }
+ };
+ })( jQuery );
+ }
+
+ (function registerAngularModule() {
+
+ var app = DayPilot.am();
+
+ if (!app) {
+ return;
+ }
+
+ app.directive("daypilotCalendar", ['$parse', function($parse) {
+// app.directive("daypilotCalendar", function() {
+ return {
+ "restrict": "E",
+ "template": "",
+ "replace": true,
+ "link": function (scope, element, attrs) {
+
+ var calendar = new DayPilot.Calendar(element[0]);
+ calendar._angular.scope = scope;
+ calendar.init();
+
+ var oattr = attrs["id"];
+ if (oattr) {
+ scope[oattr] = calendar;
+ }
+
+ // save DayPilot.Calendar object in the specified variable
+ var pas = attrs["publishAs"];
+ if (pas) {
+ var getter = $parse(pas);
+ var setter = getter.assign;
+ setter(scope, calendar);
+ }
+
+ // bind event handlers from attributes starting with "on"
+ for (var name in attrs) {
+ if (name.indexOf("on") === 0) { // event handler
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ scope["$apply"](function() {
+ f(scope, {"args": args});
+ });
+ };
+ })(name);
+ }
+ }
+
+ var watch = scope["$watch"];
+ var config = attrs["config"] || attrs["daypilotConfig"];
+ var events = attrs["events"] || attrs["daypilotEvents"];
+
+ watch.call(scope, config, function (value) {
+ //var bbOld = calendar.businessBeginsHour;
+
+ for (var name in value) {
+ calendar[name] = value[name];
+ }
+ calendar.update();
+ calendar._doInit();
+ }, true);
+
+ watch.call(scope, events, function(value) {
+ //var calendar = element.data("calendar");
+ calendar.events.list = value;
+ calendar.update();
+ }, true);
+
+ }
+ };
+ }]);
+ })();
+
+})();
diff --git a/static/src/js/src/daypilot-common.src.js b/static/src/js/src/daypilot-common.src.js
new file mode 100644
index 0000000..2fee00a
--- /dev/null
+++ b/static/src/js/src/daypilot-common.src.js
@@ -0,0 +1,3986 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+(function() {
+
+ if (typeof DayPilot.$ !== 'undefined') {
+ return;
+ }
+
+ if (typeof DayPilot.Global === "undefined") {
+ DayPilot.Global = {};
+ }
+
+ DayPilot.$ = function(id) {
+ return document.getElementById(id);
+ };
+
+ Object.defineProperty(DayPilot, 'isKhtml', {
+ get: function() { return typeof navigator !== "undefined" && navigator.userAgent && navigator.userAgent.indexOf("KHTML") !== -1; }
+ });
+
+ DayPilot.mo2 = function (target, ev){
+ ev = ev || window.event;
+
+ // IE
+ if (typeof(ev.offsetX) !== 'undefined') {
+
+ var coords = {x: ev.offsetX + 1, y:ev.offsetY + 1};
+ //document.title = "ie/doc:" + document.documentElement.scrollTop;
+
+ if (!target) {
+ return coords;
+ }
+
+ var current = ev.srcElement;
+ while (current && current !== target) {
+ if (current.tagName !== 'SPAN') { // hack for DayPilotMonth/IE, hour info on the right side of an event
+ coords.x += current.offsetLeft;
+ if (current.offsetTop > 0) { // hack for http://forums.daypilot.org/Topic.aspx/879/move_event_bug
+ coords.y += current.offsetTop - current.scrollTop;
+ }
+ }
+
+ current = current.offsetParent;
+ }
+
+ if (current) {
+ return coords;
+ }
+ return null;
+ }
+
+ // FF
+ if (typeof(ev.layerX) !== 'undefined') {
+
+ var coords = {x:ev.layerX, y:ev.layerY, src: ev.target};
+
+ if (!target) {
+ return coords;
+ }
+ var current = ev.target;
+
+ // find the positioned offsetParent, the layerX reference
+ while (current && current.style.position !== 'absolute' && current.style.position !== 'relative') {
+ current = current.parentNode;
+ if (DayPilot.isKhtml) { // hack for KHTML (Safari and Google Chrome), used in DPC/event moving
+ coords.y += current.scrollTop;
+ }
+ }
+
+ while (current && current !== target) {
+ coords.x += current.offsetLeft;
+ coords.y += current.offsetTop - current.scrollTop;
+ current = current.offsetParent;
+ }
+ if (current) {
+ return coords;
+ }
+
+ return null;
+ }
+
+ return null;
+ };
+
+ // mouse offset relative to the specified target
+ DayPilot.mo3 = function(target, ev) {
+ var coords;
+ var page = DayPilot.page(ev);
+ if (page) {
+ if (target) {
+ var abs = DayPilot.abs(target);
+ if (!abs) {
+ throw new Error("no abs");
+ return null;
+ }
+ coords = {x: page.x - abs.x, y: page.y - abs.y};
+ }
+ else {
+ coords = {x: page.x, y: page.y};
+ }
+ }
+ else {
+ coords = DayPilot.mo2(target, ev);
+ if (!coords) {
+ // throw new Error("no coords");
+ return null;
+ }
+ }
+
+ coords.shift = ev.shiftKey;
+ coords.meta = ev.metaKey;
+ coords.ctrl = ev.ctrlKey;
+ coords.alt = ev.altKey;
+
+ return coords;
+ };
+
+ DayPilot.browser = {};
+
+ Object.defineProperty(DayPilot.browser, 'hover', {
+ get: function() { return !window.matchMedia("(any-hover: none)").matches; }
+ });
+
+ DayPilot.touch = {};
+
+ // returns pageX, pageY (calculated from clientX if pageX is not available)
+ DayPilot.page = function(ev) {
+ // required for chrome/android
+ var target = ev.changedTouches && ev.changedTouches[0] ? ev.changedTouches[0] : ev;
+
+ if (typeof target.pageX !== 'undefined') {
+ return {x: target.pageX, y: target.pageY};
+ }
+ if (typeof ev.clientX !== 'undefined' && document.body && document.documentElement) {
+ return {
+ x: ev.clientX + document.body.scrollLeft + document.documentElement.scrollLeft,
+ y: ev.clientY + document.body.scrollTop + document.documentElement.scrollTop
+ };
+ }
+ // shouldn't happen
+ return null;
+ };
+
+ // absolute element position on page
+ DayPilot.abs = function(element, visible) {
+ if (!element) {
+ return null;
+ }
+
+ if (element.getBoundingClientRect) {
+ var r = DayPilot.absBoundingClientBased(element);
+
+ if (visible) {
+ // use diff, absOffsetBased is not as accurate
+ var full = DayPilot.absOffsetBased(element, false);
+ var visible = DayPilot.absOffsetBased(element, true);
+
+ r.x += visible.x - full.x;
+ r.y += visible.y - full.y;
+ r.w = visible.w;
+ r.h = visible.h;
+ }
+
+ return r;
+ }
+ else {
+ return DayPilot.absOffsetBased(element, visible);
+ }
+
+ };
+
+ DayPilot.absBoundingClientBased = function(element) {
+ var elemRect = element.getBoundingClientRect();
+
+ return {
+ // x: elemRect.left + window.scrollX, // IE11 doesn't support this
+ x: elemRect.left + window.pageXOffset,
+ // y: elemRect.top + window.scrollY, // IE11 doesn't support this
+ y: elemRect.top + window.pageYOffset,
+ w: element.clientWidth,
+ h: element.clientHeight,
+ toString: function() {
+ return "x:" + this.x + " y:" + this.y + " w:" + this.w + " h:" + this.h;
+ }
+ };
+ };
+
+
+ // old implementation of absolute position
+ // problems with adjacent float and margin-left in IE7
+ // still the best way to calculate the visible part of the element
+ DayPilot.absOffsetBased = function(element, visible) {
+ var r = {
+ x: element.offsetLeft,
+ y: element.offsetTop,
+ w: element.clientWidth,
+ h: element.clientHeight,
+ toString: function() {
+ return "x:" + this.x + " y:" + this.y + " w:" + this.w + " h:" + this.h;
+ }
+ };
+
+ while (element.offsetParent) {
+ element = element.offsetParent;
+
+ r.x -= element.scrollLeft;
+ r.y -= element.scrollTop;
+
+ if (visible) { // calculates the visible part
+ if (r.x < 0) {
+ r.w += r.x; // decrease width
+ r.x = 0;
+ }
+
+ if (r.y < 0) {
+ r.h += r.y; // decrease height
+ r.y = 0;
+ }
+
+ if (element.scrollLeft > 0 && r.x + r.w > element.clientWidth) {
+ r.w -= r.x + r.w - element.clientWidth;
+ }
+
+ if (element.scrollTop && r.y + r.h > element.clientHeight) {
+ r.h -= r.y + r.h - element.clientHeight;
+ }
+ }
+
+ r.x += element.offsetLeft;
+ r.y += element.offsetTop;
+
+ }
+
+ var pageOffset = DayPilot.pageOffset();
+ r.x += pageOffset.x;
+ r.y += pageOffset.y;
+
+ return r;
+ };
+
+ DayPilot.isArray = function(o) {
+ return Object.prototype.toString.call(o) === '[object Array]';
+ };
+
+ // distance of two points, works with x and y
+ DayPilot.distance = function(point1, point2) {
+ return Math.sqrt(Math.pow(point1.x - point2.x, 2) + Math.pow(point1.y - point2.y, 2));
+ };
+
+ DayPilot.sheet = function() {
+
+ if (typeof window === "undefined") {
+ // next.js server-side rendering
+ var sheet = {};
+ sheet.add = function() {};
+ sheet.commit = function() {};
+ return sheet;
+ }
+
+ var style = document.createElement("style");
+ style.setAttribute("type", "text/css");
+ if (!style.styleSheet) { // ie
+ style.appendChild(document.createTextNode(""));
+ }
+
+ var h = document.head || document.getElementsByTagName('head')[0];
+ h.appendChild(style);
+
+ var oldStyle = !! style.styleSheet; // old ie
+
+ var sheet = {};
+ sheet.rules = [];
+ sheet.commit = function() {
+ if (oldStyle) {
+ style.styleSheet.cssText = this.rules.join("\n");
+ }
+ };
+
+ sheet.add = function(selector, rules, index) {
+ if (oldStyle) {
+ this.rules.push(selector + "{" + rules + "\u007d");
+ return;
+ }
+ if(style.sheet.insertRule) {
+ if (typeof index === "undefined") {
+ index = style.sheet.cssRules.length;
+ }
+ style.sheet.insertRule(selector + "{" + rules + "\u007d", index);
+ }
+ else if (style.sheet.addRule) {
+ style.sheet.addRule(selector, rules, index);
+ }
+ };
+ return sheet;
+ };
+
+ DayPilot.gs = function(el, styleProp) {
+ return window.getComputedStyle(el, null).getPropertyValue(styleProp) || "";
+ };
+
+ DayPilot.StyleReader = function(element) {
+
+ this.get = function(property) {
+ if (!element) {
+ return null;
+ }
+ return DayPilot.gs(element, property);
+ };
+
+ this.getPx = function(property) {
+ var val = this.get(property);
+ if (val.indexOf("px") === -1) {
+ return undefined;
+ }
+ else {
+ return parseInt(val, 10);
+ }
+ };
+
+ };
+
+ // register the default themes
+ (function() {
+ if (DayPilot.Global.defaultCss) {
+ return;
+ }
+
+ var sheet = DayPilot.sheet();
+
+ sheet.add(".calendar_default_main", "border: 1px solid #c0c0c0; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;");
+ sheet.add(".calendar_default_main *, .calendar_default_main *:before, .calendar_default_main *:after", "box-sizing: content-box;"); // bootstrap
+ sheet.add(".calendar_default_rowheader_inner,.calendar_default_cornerright_inner,.calendar_default_corner_inner,.calendar_default_colheader_inner,.calendar_default_alldayheader_inner", "color: #333;background: #f3f3f3;");
+ sheet.add(".calendar_default_cornerright_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px; border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_direction_rtl .calendar_default_cornerright_inner", "border-right: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_rowheader_inner", "font-size: 16pt;text-align: right; position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0; padding: 3px;");
+ sheet.add(".calendar_default_direction_rtl .calendar_default_rowheader_inner", "border-right: none;");
+ sheet.add(".calendar_default_corner_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_direction_rtl .calendar_default_corner_inner", "border-right: none;");
+ sheet.add(".calendar_default_rowheader_minutes", "font-size:10px;vertical-align: super;padding-left: 2px;padding-right: 2px;");
+ sheet.add(".calendar_default_colheader_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0; display: flex; align-items: center; justify-content: center; font-size: 13px;");
+ sheet.add(".calendar_default_cell_inner", "position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #ddd;border-bottom: 1px solid #ddd; background: #f9f9f9;");
+ sheet.add(".calendar_default_cell_business .calendar_default_cell_inner", "background: #fff");
+ sheet.add(".calendar_default_alldayheader_inner", "text-align: center;position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".calendar_default_message", "opacity: 0.9; padding: 10px; color: #ffffff;background: #ffa216;");
+ sheet.add(".calendar_default_alldayevent_inner,.calendar_default_event_inner", 'color: #333; border: 1px solid #999;'); // border-top: 4px solid #1066a8;
+ sheet.add(".calendar_default_event_bar", "top: 0px;bottom: 0px;left: 0px;width: 6px;background-color: #9dc8e8;");
+ sheet.add(".calendar_default_event_bar_inner", "position: absolute;width: 6px;background-color: #1066a8;");
+ sheet.add(".calendar_default_alldayevent_inner,.calendar_default_event_inner", 'background: #fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".calendar_default_selected .calendar_default_event_inner", "background: #ddd;");
+ sheet.add(".calendar_default_alldayevent_inner", "position: absolute;top: 2px;bottom: 2px;left: 2px;right: 2px;overflow:hidden;padding: 2px;margin-right: 1px; font-size: 13px;");
+ sheet.add(".calendar_default_event_withheader .calendar_default_event_inner", "padding-top: 15px;");
+ sheet.add(".calendar_default_event", "cursor: default;");
+ sheet.add(".calendar_default_event_inner", "position: absolute;overflow: hidden;top: 0px;bottom: 0px;left: 0px;right: 0px;padding: 2px 2px 2px 8px; font-size: 13px;");
+ sheet.add(".calendar_default_event_delete", "background: url() center center no-repeat; opacity: 0.6; cursor: pointer;");
+ sheet.add(".calendar_default_event_delete:hover", "opacity: 1;-ms-filter: none;");
+ sheet.add(".calendar_default_scroll_up", "background: url();");
+ sheet.add(".calendar_default_scroll_down", "background: url();");
+ sheet.add(".calendar_default_now", "background-color: red;");
+ sheet.add(".calendar_default_now:before", "content: ''; top: -5px; border-width: 5px; border-color: transparent transparent transparent red; border-style: solid; width: 0px; height:0px; position: absolute; -moz-transform: scale(.9999);");
+ sheet.add(".calendar_default_shadow_top", 'box-sizing: border-box; padding:2px;border:1px solid #ccc;background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".calendar_default_shadow_bottom", 'box-sizing: border-box; padding:2px;border:1px solid #ccc;background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".calendar_default_crosshair_vertical, .calendar_default_crosshair_horizontal, .calendar_default_crosshair_left, .calendar_default_crosshair_top", "background-color: gray; opacity: 0.2;");
+ sheet.add(".calendar_default_loading", "background-color: orange; color: white; padding: 2px;");
+ sheet.add(".calendar_default_scroll", "background-color: #f3f3f3;");
+ sheet.add(".calendar_default_event_moving_source", "opacity: 0.5;");
+ sheet.add(".calendar_default_shadow_inner", "box-sizing: border-box; background-color: #bbbbbb;border: 1px solid #888888;opacity: 0.5;height: 100%;");
+ sheet.add(".calendar_default_shadow", "box-shadow: 0 2px 5px rgba(0,0,0,.2);");
+
+
+ // menu
+ sheet.add(".menu_default_main", "user-select:none; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif;font-size: 13px;border: 1px solid #dddddd;background-color: white;padding: 0px;cursor: default;background-image: url();background-repeat: repeat-y;xborder-radius: 5px;-moz-box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);-webkit-box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);box-shadow:0px 2px 3px rgba(000,000,000,0.3),inset 0px 0px 2px rgba(255,255,255,0.8);");
+ sheet.add(".menu_default_main, .menu_default_main *, .menu_default_main *:before, .menu_default_main *:after", "box-sizing: content-box;");
+ sheet.add(".menu_default_title", "background-color: #f2f2f2;border-bottom: 1px solid gray;padding: 4px 4px 4px 37px;");
+ sheet.add(".menu_default_main a", "padding: 2px 2px 2px 35px;color: black;text-decoration: none;cursor: default;");
+ sheet.add(".menu_default_main.menu_default_withchildren a", "padding: 2px 35px 2px 35px;");
+ sheet.add(".menu_default_main a img", "margin-left: 6px;margin-top: 2px;");
+ sheet.add(".menu_default_item_text", "display: block;height: 20px;line-height: 20px; overflow:hidden;padding-left: 2px;padding-right: 20px; white-space: nowrap;");
+ sheet.add(".menu_default_main a:hover", "background-color: #f3f3f3;");
+ sheet.add(".menu_default_main div div", "border-top: 1px solid #dddddd;margin-top: 2px;margin-bottom: 2px;margin-left: 28px;");
+ sheet.add(".menu_default_main a.menu_default_item_disabled", "color: #ccc");
+ sheet.add(".menu_default_item_haschildren.menu_default_item_haschildren_active", 'background-color: #f3f3f3;');
+ sheet.add(".menu_default_item_haschildren a:before", "content: ''; border-width: 5px; border-color: transparent transparent transparent #666; border-style: solid; width: 0px; height:0px; position: absolute; right: 5px; margin-top: 5px;");
+ sheet.add(".menu_default_item_icon", "position: absolute; top:0px; left: 0px; padding: 2px 2px 2px 8px;");
+ sheet.add(".menu_default_item a i", "height: 20px;line-height: 20px;");
+ sheet.add(".menu_default_item .menu_default_item_symbol", "width: 18px; height: 18px; color: #999; margin-left: 6px;margin-top: 2px;");
+
+
+ // menubar
+ sheet.add(".menubar_default_main", "border-bottom: 1px solid #ccc; font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px; user-select:none;");
+ sheet.add(".menubar_default_item", "display: inline-block; padding: 6px 10px; cursor: default;");
+ sheet.add(".menubar_default_item:hover", "background-color: #f2f2f2;");
+ sheet.add(".menubar_default_item_active", "background-color: #f2f2f2;");
+
+
+ sheet.add(".scheduler_default_selected .scheduler_default_event_inner", "background: #ddd;");
+ sheet.add(".scheduler_default_main", "border: 1px solid #c0c0c0;font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;");
+ sheet.add(".scheduler_default_timeheader", "cursor: default;color: #333;");
+ sheet.add(".scheduler_default_message", "opacity: 0.9;filter: alpha(opacity=90);padding: 10px; color: #ffffff;background: #ffa216;");
+ sheet.add(".scheduler_default_timeheadergroup,.scheduler_default_timeheadercol", "color: #333;background: #f3f3f3;");
+ sheet.add(".scheduler_default_rowheader,.scheduler_default_corner", "color: #333;background: #f3f3f3;");
+ sheet.add(".scheduler_default_rowheader_inner", "position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #eee;padding: 2px;");
+ sheet.add(".scheduler_default_timeheadergroup, .scheduler_default_timeheadercol", "text-align: center;");
+ sheet.add(".scheduler_default_timeheadergroup_inner", "position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".scheduler_default_timeheadercol_inner", "position: absolute;left: 0px;right: 0px;top: 0px;bottom: 0px;border-right: 1px solid #c0c0c0;");
+ sheet.add(".scheduler_default_divider", "background-color: #c0c0c0;");
+ sheet.add(".scheduler_default_divider_horizontal", "background-color: #c0c0c0;");
+ sheet.add(".scheduler_default_matrix_vertical_line", "background-color: #eee;");
+ sheet.add(".scheduler_default_matrix_vertical_break", "background-color: #000;");
+ sheet.add(".scheduler_default_matrix_horizontal_line", "background-color: #eee;");
+ sheet.add(".scheduler_default_resourcedivider", "background-color: #c0c0c0;");
+ sheet.add(".scheduler_default_shadow_inner", "background-color: #666666;opacity: 0.5;filter: alpha(opacity=50);height: 100%;xborder-radius: 5px;");
+ sheet.add(".scheduler_default_event", "color:#333; font-size: 13px;");
+ sheet.add(".scheduler_default_event_inner", "position:absolute;top:0px;left:0px;right:0px;bottom:0px;padding:5px 2px 2px 2px;overflow:hidden;border:1px solid #ccc;");
+ sheet.add(".scheduler_default_event_bar", "top:0px;left:0px;right:0px;height:4px;background-color:#9dc8e8;");
+ sheet.add(".scheduler_default_event_bar_inner", "position:absolute;height:4px;background-color:#1066a8;");
+ sheet.add(".scheduler_default_event_inner", 'background:#fff;background: -webkit-gradient(linear, left top, left bottom, from(#ffffff), to(#eeeeee));background: -webkit-linear-gradient(top, #ffffff 0%, #eeeeee);background: -moz-linear-gradient(top, #ffffff 0%, #eeeeee);background: -ms-linear-gradient(top, #ffffff 0%, #eeeeee);background: -o-linear-gradient(top, #ffffff 0%, #eeeeee);background: linear-gradient(top, #ffffff 0%, #eeeeee);filter: progid:DXImageTransform.Microsoft.Gradient(startColorStr="#ffffff", endColorStr="#eeeeee");');
+ sheet.add(".scheduler_default_event_float_inner", "padding:6px 2px 2px 8px;"); // space for arrow
+ sheet.add(".scheduler_default_event_float_inner:after", 'content:"";border-color: transparent #666 transparent transparent;border-style:solid;border-width:5px;width:0;height:0;position:absolute;top:8px;left:-4px;');
+ sheet.add(".scheduler_default_columnheader_inner", "font-weight: bold;");
+ sheet.add(".scheduler_default_columnheader_splitter", "background-color: #666;opacity: 0.5;filter: alpha(opacity=50);");
+ sheet.add(".scheduler_default_columnheader_cell_inner", "padding: 2px;");
+ sheet.add(".scheduler_default_cell", "background-color: #f9f9f9;");
+ sheet.add(".scheduler_default_cell.scheduler_default_cell_business", "background-color: #fff;");
+
+ sheet.add(".navigator_default_main", "border-left: 1px solid #c0c0c0;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;background-color: white;color: #000000; box-sizing: content-box;");
+ sheet.add(".navigator_default_main *, .navigator_default_main *:before, .navigator_default_main *:after", "box-sizing: content-box;");
+ sheet.add(".navigator_default_month", "font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 12px;");
+ sheet.add(".navigator_default_day", "color: black;");
+ sheet.add(".navigator_default_weekend", "background-color: #f0f0f0;");
+ sheet.add(".navigator_default_dayheader", "color: black;");
+ sheet.add(".navigator_default_line", "border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".navigator_default_dayother", "color: gray;");
+ sheet.add(".navigator_default_todaybox", "border: 1px solid red;");
+ sheet.add(".navigator_default_title, .navigator_default_titleleft, .navigator_default_titleright", "border-top: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;color: #333;background: #f3f3f3;");
+ sheet.add(".navigator_default_busy", "font-weight: bold;");
+ sheet.add(".navigator_default_cell", "text-align: center;");
+ sheet.add(".navigator_default_select .navigator_default_cell_box", "background-color: #FFE794; opacity: 0.5;");
+ sheet.add(".navigator_default_title", "text-align: center;");
+ sheet.add(".navigator_default_titleleft, .navigator_default_titleright", "text-align: center;");
+ sheet.add(".navigator_default_dayheader", "text-align: center;");
+ sheet.add(".navigator_default_weeknumber", "text-align: center; color: #999;");
+ sheet.add(".navigator_default_cell_text", "cursor: pointer;");
+
+ sheet.add(".month_default_main", "border: 1px solid #c0c0c0;font-family: -apple-system,system-ui,BlinkMacSystemFont,'Segoe UI',Roboto,'Helvetica Neue',Arial,sans-serif; font-size: 13px;color: #333;");
+ sheet.add(".month_default_main *, .month_default_main *:before, .month_default_main *:after", "box-sizing: content-box;");
+ sheet.add(".month_default_cell_inner", "border-right: 1px solid #ddd;border-bottom: 1px solid #ddd;position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;background-color: #f9f9f9;");
+ sheet.add(".month_default_cell_business .month_default_cell_inner", "background-color: #fff;");
+ sheet.add(".month_default_cell_header", "text-align: right; padding: 4px; box-sizing: border-box;");
+ sheet.add(".month_default_header_inner", 'position: absolute;top: 0px;left: 0px;bottom: 0px;right: 0px;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;cursor: default;color: #333;background: #f3f3f3; overflow:hidden; display: flex; align-items: center; justify-content: center;');
+ sheet.add(".month_default_message", 'padding: 10px;opacity: 0.9; color: #ffffff;background: #ffa216;');
+ sheet.add(".month_default_event_inner", 'position: absolute;top: 0px;bottom: 0px;left: 1px;right: 1px;overflow:hidden;padding: 2px;padding-left: 10px;color: #333;background: #fff;background: linear-gradient(to bottom, #ffffff 0%, #eeeeee);border: 1px solid #999;border-radius: 0px;display: flex; align-items: center; font-size: 13px;');
+ sheet.add(".month_default_event_continueright .month_default_event_inner", "border-top-right-radius: 0px;border-bottom-right-radius: 0px;border-right-style: dotted;");
+ sheet.add(".month_default_event_continueleft .month_default_event_inner", "border-top-left-radius: 0px;border-bottom-left-radius: 0px;border-left-style: dotted;");
+
+ sheet.add(".month_default_event_bar", "top: 0px;bottom: 0px;left: 0px;width: 6px;");
+ sheet.add(".month_default_event_bar_inner", "position: absolute;width: 6px;background-color: #1066a8;");
+ sheet.add(".month_default_event_continueleft .month_default_event_bar", "display: none;");
+
+ sheet.add(".month_default_selected .month_default_event_inner", "background: #ddd;");
+ // sheet.add(".month_default_shadow_inner", "background-color: #666666;opacity: 0.5;height: 100%;");
+ sheet.add(".month_default_event_delete", "background: url() center center no-repeat; opacity: 0.6; cursor: pointer;");
+ sheet.add(".month_default_event_delete:hover", "opacity: 1;-ms-filter: none;");
+ sheet.add(".month_default_event_timeleft", "color: #ccc; font-size: 8pt");
+ sheet.add(".month_default_event_timeright", "color: #ccc; font-size: 8pt; text-align: right;");
+ sheet.add(".month_default_loading", "background-color: orange; color: white; padding: 2px;");
+ sheet.add(".month_default_shadow_inner", "box-sizing: border-box; background-color: #bbbbbb;border: 1px solid #888888;opacity: 0.5;height: 100%;");
+ sheet.add(".month_default_shadow", "box-shadow: 0 2px 5px rgba(0,0,0,.2);");
+
+ sheet.commit();
+
+ DayPilot.Global.defaultCss = true;
+ })();
+
+ // document element
+ DayPilot.doc = function() {
+ var de = document.documentElement;
+ return (de && de.clientHeight) ? de : document.body;
+ };
+
+ DayPilot.guid = function() {
+ var S4 = function() {
+ return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
+ };
+ return ("" + S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4());
+ };
+
+ DayPilot.pageOffset = function() {
+ if (typeof pageXOffset !== 'undefined') {
+ return { x: pageXOffset, y: pageYOffset };
+ }
+ var d = DayPilot.doc();
+ return { x: d.scrollLeft, y: d.scrollTop };
+ };
+
+ DayPilot.indexOf = function(array, object) {
+ if (!array || !array.length) {
+ return -1;
+ }
+ for (var i = 0; i < array.length; i++) {
+ if (array[i] === object) {
+ return i;
+ }
+ }
+ return -1;
+ };
+
+ // all children
+ DayPilot.ac = function(e, children) {
+ if (!children) {
+ var children = [];
+ }
+ for (var i = 0; e.children && i < e.children.length; i++) {
+ children.push(e.children[i]);
+ DayPilot.ac(e.children[i], children);
+ }
+
+ return children;
+ };
+
+ // remove from array
+ DayPilot.rfa = function(array, object) {
+ var i = DayPilot.indexOf(array, object);
+ if (i === -1) {
+ return;
+ }
+ array.splice(i, 1);
+ };
+
+ // mouse coords
+ DayPilot.mc = function(ev){
+ if(ev.pageX || ev.pageY){
+ return {x:ev.pageX, y:ev.pageY};
+ }
+ return {
+ x:ev.clientX + document.documentElement.scrollLeft,
+ y:ev.clientY + document.documentElement.scrollTop
+ };
+ };
+
+ DayPilot.Stats = {};
+ DayPilot.Stats.eventObjects = 0;
+ DayPilot.Stats.dateObjects = 0;
+ DayPilot.Stats.cacheHitsCtor = 0;
+ DayPilot.Stats.cacheHitsParsing = 0;
+ DayPilot.Stats.cacheHitsTicks = 0;
+
+ DayPilot.re = function(el, ev, func) {
+ if (!func) {
+ return;
+ }
+ if (!ev) {
+ return;
+ }
+ if (!el) {
+ return;
+ }
+ el.addEventListener(ev, func, false);
+ };
+
+ DayPilot.rePassive = function(el, ev, func) {
+ if (!func) {
+ return;
+ }
+ if (!ev) {
+ return;
+ }
+ if (!el) {
+ return;
+ }
+ el.addEventListener(ev, func, {"passive": true});
+ };
+
+ DayPilot.reNonPassive = function(el, ev, func) {
+ if (!func) {
+ return;
+ }
+ if (!ev) {
+ return;
+ }
+ if (!el) {
+ return;
+ }
+ el.addEventListener(ev, func, {"passive": false});
+ };
+
+ // purge
+ // thanks to http://javascript.crockford.com/memory/leak.html
+ DayPilot.pu = function(d) {
+ //var removed = [];
+ //var start = new Date();
+ var a = d.attributes, i, l, n;
+ if (a) {
+ l = a.length;
+ for (i = 0; i < l; i += 1) {
+ if (!a[i]) {
+ continue;
+ }
+ n = a[i].name;
+ if (typeof d[n] === 'function') {
+ d[n] = null;
+ }
+ }
+ }
+ a = d.childNodes;
+ if (a) {
+ l = a.length;
+ for (i = 0; i < l; i += 1) {
+ var children = DayPilot.pu(d.childNodes[i]);
+ }
+ }
+ };
+
+ // delete element
+ DayPilot.de = function(e) {
+ if (!e) {
+ return;
+ }
+ if (DayPilot.isArray(e)) {
+ for (var i = 0; i < e.length; i++) {
+ DayPilot.de(e[i]);
+ }
+ return;
+ }
+ e.parentNode && e.parentNode.removeChild(e);
+ };
+
+ // vertical scrollbar width
+ DayPilot.sw = function(element) {
+ if (!element) {
+ return 0;
+ }
+ return element.offsetWidth - element.clientWidth;
+ };
+
+ // angular module
+ DayPilot.am = function() {
+ if (typeof angular === "undefined") {
+ return null;
+ }
+ if (!DayPilot.am.cached) {
+ DayPilot.am.cached = angular.module("daypilot", []);
+ }
+ return DayPilot.am.cached;
+ };
+
+ DayPilot.Selection = function (start, end, resource, root) {
+ this.type = 'selection';
+ this.start = start.isDayPilotDate ? start : new DayPilot.Date(start);
+ this.end = end.isDayPilotDate ? end : new DayPilot.Date(end);
+ this.resource = resource;
+ this.root = root;
+
+ this.toJSON = function(key) {
+ var json = {};
+ json.start = this.start;
+ json.end = this.end;
+ json.resource = this.resource;
+
+ return json;
+ };
+ };
+
+ /* XMLHttpRequest */
+
+ DayPilot.request = function(url, callback, postData, errorCallback) {
+ var req = DayPilot.createXmlHttp();
+ if (!req) {
+ return;
+ }
+
+ req.open("POST", url, true);
+ req.setRequestHeader('Content-type', 'text/plain');
+ req.onreadystatechange = function() {
+ if (req.readyState !== 4)
+ return;
+ if (req.status !== 200 && req.status !== 304) {
+ if (errorCallback) {
+ errorCallback(req);
+ }
+ else {
+ if (window.console) { console.log('HTTP error ' + req.status); }
+ }
+ return;
+ }
+ callback(req);
+ };
+ if (req.readyState === 4) {
+ return;
+ }
+ if (typeof postData === 'object') {
+ postData = JSON.stringify(postData);
+ }
+ req.send(postData);
+ };
+
+ DayPilot.ajax = function(params) {
+ if (!params) {
+ throw new DayPilot.Exception("Parameter object required.");
+ }
+
+ if (typeof params.url !== "string") {
+ throw new DayPilot.Exception("The parameter object must have 'url' property.")
+ }
+
+ var req = DayPilot.createXmlHttp();
+ if (!req) {
+ throw new DayPilot.Exception("Unable to create XMLHttpRequest object");
+ }
+
+ var dataIsObject = typeof params.data === "object";
+
+ var data = params.data;
+ var method = params.method || (params.data ? "POST" : "GET");
+ var success = params.success || function() {};
+ var error = params.error || function() {};
+ var url = params.url;
+ var contentType = params.contentType || (dataIsObject ? "application/json" : "text/plain");
+ var headers = params.headers || {};
+
+ req.open(method, url, true);
+ req.setRequestHeader('Content-type', contentType);
+
+ // overriding the content-type is allowed
+ DayPilot.Util.ownPropsAsArray(headers).forEach(function(item) {
+ req.setRequestHeader(item.key, item.val);
+ });
+
+ req.onreadystatechange = function() {
+ if (req.readyState !== 4) {
+ return;
+ }
+ if (req.status !== 200 && req.status !== 201 && req.status !== 204 && req.status !== 304) {
+ if (error) {
+ var args = {};
+ args.request = req;
+ error(args);
+ }
+ else {
+ if (window.console) { console.log('HTTP error ' + req.status); }
+ }
+ return;
+ }
+ var args = {};
+ args.request = req;
+ if (req.responseText) {
+ args.data = JSON.parse(req.responseText);
+ }
+ success(args);
+ };
+ if (req.readyState === 4) {
+ return;
+ }
+ if (dataIsObject) {
+ data = JSON.stringify(data);
+ }
+ req.send(data);
+ };
+
+ DayPilot.createXmlHttp = function() {
+ return new XMLHttpRequest();
+ };
+
+ DayPilot.Http = {};
+
+ DayPilot.Http.ajax = function(params) {
+ DayPilot.ajax(params);
+ };
+
+ /*
+allowed params:
+- headers
+- contentType
+ */
+ DayPilot.Http.get = function(url, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "GET";
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Http.post = function(url, data, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "POST";
+ aparams.data = data;
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Http.put = function(url, data, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "PUT";
+ aparams.data = data;
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Http.delete = function(url, params) {
+ params = params || {};
+ return new Promise(function(resolve, reject) {
+ var aparams = {};
+ aparams.url = url;
+ aparams.method = "DELETE";
+ aparams.success = function(args) {
+ resolve(args);
+ };
+ aparams.error = function(args) {
+ reject(args);
+ };
+ aparams.contentType = params.contentType;
+ aparams.headers = params.headers;
+
+ DayPilot.ajax(aparams);
+ });
+ };
+
+ DayPilot.Util = {};
+ DayPilot.Util.addClass = function(object, name) {
+ if (!object) {
+ return;
+ }
+ if (!object.className) {
+ object.className = name;
+ return;
+ }
+ var already = new RegExp("(^|\\s)" + name + "($|\\s)");
+ if (!already.test(object.className)) {
+ object.className = object.className + ' ' + name;
+ }
+ };
+
+ DayPilot.Util.removeClass = function(object, name) {
+ if (!object) {
+ return;
+ }
+ var already = new RegExp("(^|\\s)" + name + "($|\\s)");
+ object.className = object.className.replace(already, ' ').replace(/^\s\s*/, '').replace(/\s\s*$/, ''); // trim spaces
+ };
+
+ DayPilot.Util.copyProps = function(source, target, props) {
+ if (!target) {
+ target = {};
+ }
+ if (!source) {
+ return target;
+ }
+ if (typeof props === 'undefined') {
+ for (var name in source) {
+ if (source.hasOwnProperty(name) && typeof source[name] !== 'undefined') {
+ target[name] = source[name];
+ }
+ }
+ }
+ else {
+ for (var i = 0; i < props.length; i++) {
+ var name = props[i];
+ if (typeof source[name] !== 'undefined') {
+ target[name] = source[name];
+ }
+ }
+ }
+ return target;
+ };
+
+ DayPilot.Util.ownPropsAsArray = function(object) {
+ var result = [];
+
+ if (!object) {
+ return result;
+ }
+
+ for (var name in object) {
+ if (object.hasOwnProperty(name)) {
+ var item = {};
+ item.key = name;
+ item.val = object[name];
+ result.push(item);
+ }
+ }
+
+ return result;
+ };
+
+ DayPilot.Util.replaceCharAt = function(str, index, character) {
+ return str.substr(0, index) + character + str.substr(index + character.length);
+ };
+
+ DayPilot.Util.isNullOrUndefined = function(val) {
+ return val === null || typeof val === "undefined";
+ };
+
+ DayPilot.Util.escapeHtml = function(html) {
+ var div = document.createElement("div");
+ div.innerText = html;
+ return div.innerHTML;
+ };
+
+ DayPilot.Util.escapeTextHtml = function(text, html) {
+ if (!DayPilot.Util.isNullOrUndefined(html)) {
+ return html;
+ }
+ if (DayPilot.Util.isNullOrUndefined(text)) {
+ return "";
+ }
+ return DayPilot.Util.escapeHtml(text);
+ };
+
+ /* areas */
+
+ DayPilot.Areas = {};
+
+ /**
+ * Attach active areas to a target div.
+ * Works with "Visible", "TouchVisible" and "Hover" visibility
+ */
+ DayPilot.Areas.attach = function (div, e, options) {
+ var options = options || {};
+ var areas = options.areas;
+ var allowed = options.allowed || function() { return true; };
+ var offsetX = options.offsetX || 0;
+
+ // permanently visible active areas
+ areas = areasExtract(e, areas);
+
+ if (!areas) {
+ return;
+ }
+
+ if (!DayPilot.isArray(areas)) {
+ return;
+ }
+
+ if (areas.length === 0) {
+ return;
+ }
+
+ DayPilot.re(div, "mousemove", function(ev) {
+ if (!div.active && !div.areasDisabled && allowed()) {
+ DayPilot.Areas.showAreas(div, e, ev, areas, { "offsetX": offsetX, "eventDiv": options.eventDiv});
+ }
+ });
+ DayPilot.re(div, "mouseleave", function(ev) {
+ DayPilot.Areas.hideAreas(div, ev);
+ });
+
+ areas.forEach(function(area) {
+ if (!DayPilot.Areas.isVisible(area)) {
+ return;
+ }
+ var a = DayPilot.Areas.createArea(div, e, area, { "offsetX": offsetX, "eventDiv": options.eventDiv});
+ div.appendChild(a);
+ });
+ };
+
+ DayPilot.Areas.disable = function(div) {
+ div.areasDisabled = true;
+ Array.from(div.childNodes).filter(function(item) { return item.isActiveArea && !item.area.start; }).forEach(function(item) {
+ item._originalDisplay = item.style.display;
+ item.style.display = "none";
+ });
+ };
+
+ DayPilot.Areas.enable = function(div) {
+ div.areasDisabled = false;
+ Array.from(div.childNodes).filter(function(item) { return item.isActiveArea && !item.area.start; }).forEach(function(item) {
+ if (item._originalDisplay) {
+ item.style.display = item._originalDisplay;
+ }
+ else {
+ item.style.display = "";
+ }
+ });
+ };
+
+ DayPilot.Areas.remove = function(div) {
+ var divs = Array.from(div.childNodes).filter(function(item) { return item.isActiveArea; });
+
+ DayPilot.de(divs);
+ };
+
+ DayPilot.Areas.isVisible = function(area) {
+ var v = area.visibility || area.v || "Visible";
+ if (v === "Visible") {
+ return true;
+ }
+ if (v === "TouchVisible") {
+ if (!DayPilot.browser.hover) {
+ return true;
+ }
+ }
+ return false;
+ };
+
+ DayPilot.Areas.copy = function(areas) {
+ if (!DayPilot.isArray(areas)) {
+ return [];
+ }
+
+ return areas.map(function(area) {
+ return DayPilot.Util.copyProps(area, {});
+ });
+ };
+
+ /**
+ * Extracts areas array from the source object, giving priority to a standalone areas object.
+ * @param e
+ * @param areas
+ */
+ var areasExtract = function(e, areas) {
+ if (!DayPilot.isArray(areas)) {
+ areas = e.areas;
+ if (!areas) {
+ if (e.cache) {
+ areas = e.cache.areas;
+ }
+ else if (e.data) {
+ areas = e.data.areas;
+ }
+ }
+ }
+ return areas;
+ };
+
+
+ DayPilot.Areas.showAreas = function(div, e, ev, areas, options) {
+ if (DayPilot.Global.resizing) {
+ return;
+ }
+
+ if (DayPilot.Global.moving) {
+ return;
+ }
+
+ if (DayPilot.Global.selecting) {
+ return;
+ }
+
+ if (div.active) {
+ return;
+ }
+
+ if (!DayPilot.browser.hover) {
+ return;
+ }
+
+ if (DayPilot.Areas.all && DayPilot.Areas.all.length > 0) {
+ for (var i = 0; i < DayPilot.Areas.all.length; i++) {
+ var d = DayPilot.Areas.all[i];
+ if (d !== div) {
+ DayPilot.Areas.hideAreas(d, ev);
+ }
+ }
+ }
+
+ div.active = {};
+
+ //var areas;
+ if (!DayPilot.isArray(areas)) {
+ areas = e.areas;
+ if (!areas) {
+ if (e.cache) {
+ areas = e.cache.areas;
+ }
+ else if (e.data) {
+ areas = e.data.areas;
+ }
+ }
+ }
+
+ if (!areas || areas.length === 0) {
+ return;
+ }
+
+ if (div.areas && div.areas.length > 0) {
+ return;
+ }
+ div.areas = [];
+
+ for (var i = 0; i < areas.length; i++) {
+ var area = areas[i];
+
+ if (DayPilot.Areas.isVisible(area)) {
+ continue;
+ }
+
+ var a = DayPilot.Areas.createArea(div, e, area, options);
+
+ div.areas.push(a);
+ div.appendChild(a);
+
+ DayPilot.Areas.all.push(div);
+ }
+ div.active.children = DayPilot.ac(div);
+ };
+
+ /**
+ *
+ * @param div Target div element
+ * @param e Source data object
+ * @param area Area definition
+ * @returns {Element}
+ */
+ DayPilot.Areas.createArea = function(div, e, area, options) {
+
+ /*
+ actions:
+ - Default (equals not specified)
+ - None (cancels bubbling to parent element)
+ - ContextMenu
+ - HoverMenu
+ - ResizeStart
+ - ResizeEnd
+ - Move
+ - excluded: JavaScript, Bubble
+ */
+
+ var options = options || {};
+ var offsetX = options.offsetX || 0;
+ var ediv = options.eventDiv || div;
+
+ var a = document.createElement("div");
+ a.isActiveArea = true;
+ a.area = area;
+ a.setAttribute("unselectable", "on");
+ var w = area.w || area.width;
+ var h = area.h || area.height;
+ var css = area.cssClass || area.css || area.className;
+ if (typeof area.style !== "undefined") {
+ a.setAttribute("style", area.style);
+ }
+ a.style.position = "absolute";
+
+ a.style.width = resolvePosVal(w);
+ a.style.height = resolvePosVal(h);
+ a.style.right = resolvePosVal(area.right);
+ a.style.top = resolvePosVal(area.top);
+ a.style.left = resolvePosVal(area.left);
+ a.style.bottom = resolvePosVal(area.bottom);
+
+ if (typeof area.html !== 'undefined' || typeof area.text !== "undefined") {
+ // a.innerHTML = area.html || area.text;
+ a.innerHTML = DayPilot.Util.escapeTextHtml(area.text, area.html);
+ }
+ else if (area.icon) {
+ var iel = document.createElement("i");
+ iel.className = area.icon;
+ a.appendChild(iel);
+ }
+ else if (area.image) {
+ var img = document.createElement("img");
+ img.src = area.image;
+ a.appendChild(img);
+ }
+ else if (area.symbol) {
+ var ns = "http://www.w3.org/2000/svg";
+ var svg = document.createElementNS(ns,"svg");
+ svg.setAttribute("width", "100%");
+ svg.setAttribute("height", "100%");
+ var use = document.createElementNS(ns,"use");
+ use.setAttribute("href", area.symbol);
+ svg.appendChild(use);
+ a.appendChild(svg);
+ }
+
+ if (css) {
+ a.className = css;
+ }
+ if (area.toolTip) {
+ a.setAttribute("title", area.toolTip);
+ }
+ if (area.backColor) {
+ a.style.background = area.backColor;
+ }
+ if (area.background) { // alias
+ a.style.background = area.background;
+ }
+ if (area.fontColor) {
+ a.style.color = area.fontColor;
+ }
+ if (area.padding) {
+ a.style.padding = area.padding + "px";
+ a.style.boxSizing = "border-box";
+ }
+ if (area.verticalAlignment) {
+ a.style.display = "flex";
+ switch (area.verticalAlignment) {
+ case "center":
+ a.style.alignItems = "center";
+ break;
+ case "top":
+ a.style.alignItems = "flex-start";
+ break;
+ case "bottom":
+ a.style.alignItems = "flex-end";
+ break;
+ }
+ }
+ if (area.horizontalAlignment) {
+ a.style.display = "flex";
+ switch (area.horizontalAlignment) {
+ case "right":
+ a.style.justifyContent = "flex-end";
+ break;
+ case "left":
+ a.style.justifyContent = "flex-start";
+ break;
+ case "center":
+ a.style.justifyContent = "center";
+ break;
+ }
+ }
+
+ if (area.action === "ResizeEnd" || area.action === "ResizeStart" || area.action === "Move") {
+ if (e.calendar.isCalendar) {
+ switch (area.action) {
+ case "ResizeEnd":
+ area.cursor = "s-resize";
+ area.dpBorder = "bottom";
+ break;
+ case "ResizeStart":
+ area.cursor = "n-resize";
+ area.dpBorder = "top";
+ break;
+ case "Move":
+ area.cursor = "move";
+ break;
+ }
+ }
+ if (e.calendar.isScheduler || e.calendar.isMonth) {
+ switch (area.action) {
+ case "ResizeEnd":
+ area.cursor = "e-resize";
+ area.dpBorder = "right";
+ break;
+ case "ResizeStart":
+ area.cursor = "w-resize";
+ area.dpBorder = "left";
+ break;
+ case "Move":
+ area.cursor = "move";
+ break;
+ }
+ }
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+
+ if (e.calendar.internal && e.calendar.internal.dragInProgress && e.calendar.internal.dragInProgress()) { // resizing in progress
+ return;
+ }
+ ev.cancelBubble = true;
+
+ div.style.cursor = area.cursor;
+ if (area.dpBorder) {
+ div.dpBorder = area.dpBorder;
+ }
+ };
+ })(ediv, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ div.style.cursor = '';
+ };
+ })(ediv, e, area);
+ }
+ if ((area.action === "ResizeEnd" || area.action === "ResizeStart") && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ var touch = e.calendar.internal.touch;
+ var t = ev.touches ? ev.touches[0] : ev;
+ var coords = {x: t.pageX, y: t.pageY };
+ // immediately
+ e.calendar.coords = touch.relativeCoords(ev);
+ touch.preventEventTap = true;
+ if (e.calendar.isScheduler) {
+ touch.startResizing(div, area.action === "ResizeEnd" ? "right" : "left");
+ }
+ else if (e.calendar.isCalendar) {
+ touch.startResizing(div, area.action === "ResizeEnd" ? "bottom" : "top", coords);
+ }
+
+ };
+ })(ediv, e, area);
+ DayPilot.rePassive(a, DayPilot.touch.start, touchstart);
+ // a.addEventListener(DayPilot.touch.start, touchstart, {"passive": true});
+ }
+ }
+ if (area.action === "ContextMenu" && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+
+ showContextMenu(div, e, area, ev);
+ var touch = e.calendar.internal.touch;
+ touch.preventEventTap = true;
+ };
+ })(ediv, e, area);
+ var touchend = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+ };
+ })(ediv, e, area);
+ DayPilot.reNonPassive(a, DayPilot.touch.start, touchstart);
+ DayPilot.reNonPassive(a, DayPilot.touch.end, touchend);
+ }
+ }
+ if (area.action === "Bubble" && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+
+ var args = doOnClick(area, e, ev);
+ if (args.preventDefault.value) {
+ return;
+ }
+ showBubble(e, area, ev);
+ var touch = e.calendar.internal.touch;
+ touch.preventEventTap = true;
+
+ if (typeof area.onClicked === "function") {
+ area.onClicked(args);
+ }
+ };
+ })(ediv, e, area);
+ var touchend = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ ev.preventDefault();
+ };
+ })(ediv, e, area);
+ DayPilot.reNonPassive(a, DayPilot.touch.start, touchstart);
+ DayPilot.reNonPassive(a, DayPilot.touch.end, touchend);
+ }
+ }
+ if (area.action === "Move" && e.isEvent) {
+ if (e.calendar.internal.touch) {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ ev.cancelBubble = true;
+ var touch = e.calendar.internal.touch;
+ var t = ev.touches ? ev.touches[0] : ev;
+ var coords = {x: t.pageX, y: t.pageY };
+ // immediately
+ e.calendar.coords = touch.relativeCoords(ev);
+
+ // DayPilot.Global.movingAreaData = area.data;
+ if (DayPilot.Global && DayPilot.Global.touch) {
+ DayPilot.Global.touch.active = true;
+ }
+
+ touch.preventEventTap = true;
+ touch.startMoving(div, coords);
+ };
+ })(ediv, e, area);
+ DayPilot.rePassive(a, DayPilot.touch.start, touchstart);
+ // a.addEventListener(DayPilot.touch.start, touchstart, {"passive": true});
+ }
+ }
+ if (area.action === "Bubble" && e.isEvent) {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ if (area.bubble) {
+ area.bubble.showEvent(e, true);
+ }
+ else if (e.calendar.bubble) {
+ e.calendar.bubble.showEvent(e, true);
+ }
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ if (typeof DayPilot.Bubble !== "undefined") {
+ //DayPilot.Bubble.hideActive();
+ if (area.bubble) {
+ area.bubble.hideOnMouseOut();
+ }
+ else if (e.calendar.bubble) {
+ e.calendar.bubble.hideOnMouseOut();
+ }
+ }
+
+ };
+ })(div, e, area);
+ }
+ else if (area.action === "Bubble" && e.isRow) {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ if (area.bubble) {
+ area.bubble.showResource(e, true);
+ }
+ else if (e.calendar.resourceBubble) {
+ e.calendar.resourceBubble.showResource(e, true);
+ }
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ if (typeof DayPilot.Bubble !== "undefined") {
+ //DayPilot.Bubble.hideActive();
+ if (area.bubble) {
+ area.bubble.hideOnMouseOut();
+ }
+ else if (e.calendar.resourceBubble) {
+ e.calendar.resourceBubble.hideOnMouseOut();
+ }
+ }
+
+ };
+ })(div, e, area);
+ }
+ else if (area.action === "Bubble" && typeof DayPilot.Bubble !== "undefined" && area.bubble instanceof DayPilot.Bubble) {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ area.bubble.showHtml(null, null);
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ if (typeof DayPilot.Bubble !== "undefined") {
+ if (area.bubble) {
+ area.bubble.hideOnMouseOut();
+ }
+ }
+
+ };
+ })(div, e, area);
+ }
+ if (area.action === "HoverMenu") {
+ a.onmousemove = (function(div, e, area) {
+ return function(ev) {
+ var m = area.menu;
+ if (m && m.show) {
+ if (!m.visible) {
+ m.show(e);
+ }
+ else if (m.source && typeof m.source.id !== 'undefined' && m.source.id !== e.id) {
+ m.show(e);
+ }
+ m.cancelHideTimeout();
+ }
+ };
+ })(div, e, area);
+ a.onmouseout = (function(div, e, area) {
+ return function(ev) {
+ var m = area.menu;
+ if (!m) {
+ return;
+ }
+ if (m.hideOnMouseOver) {
+ m.delayedHide();
+ }
+ };
+ })(div, e, area);
+ }
+ if (area.action === "None") {
+ var touchstart = (function(div, e, area) {
+ return function(ev) {
+ var args = doOnClick(area, e, ev);
+
+ if (typeof area.onClicked === "function") {
+ area.onClicked(args);
+ }
+
+ ev.preventDefault();
+ ev.stopPropagation();
+ };
+ })(ediv, e, area);
+ DayPilot.reNonPassive(a, DayPilot.touch.start, touchstart);
+ }
+
+ // prevent event moving
+ a.onmousedown = (function(div, e, area) {
+ return function(ev) {
+ if (typeof area.onmousedown === 'function') { // obsolete, remove
+ area.onmousedown(ev);
+ }
+
+ if (typeof area.mousedown === 'function') { // internal
+ var args = {};
+ args.area = area;
+ args.div = div;
+ args.originalEvent = ev;
+ args.source = e;
+ area.mousedown(args);
+ }
+
+ if (area.action === "Move" && e.isRow) {
+ var row = e.$.row;
+ var startMoving = e.calendar.internal.rowStartMoving;
+
+ startMoving(row);
+ }
+
+ // cancel any bubble
+ if (typeof DayPilot.Bubble !== "undefined") {
+ DayPilot.Bubble.hideActive();
+ }
+
+ if (area.action === "Move") {
+ DayPilot.Global.movingAreaData = area.data;
+ }
+
+ if (area.action === "Move" && e.isEvent) {
+ if (e.calendar.internal && e.calendar.internal.startMoving) {
+ e.calendar.internal.startMoving(div, ev);
+ }
+ }
+
+ var cancel = true;
+
+ if (cancel) {
+ if (area.action === "Move" || area.action === "ResizeEnd" || area.action === "ResizeStart" || !area.action || area.action === "Default") {
+ return;
+ }
+ ev.preventDefault(); // prevents text selection on dragging
+ ev.cancelBubble = true;
+ }
+ };
+ })(div, e, area);
+ a.onclick = (function(div, e, area) {
+ return function(ev) {
+ var args = doOnClick(area, e, ev);
+
+ if (args.preventDefault.value) {
+ return;
+ }
+
+ switch (area.action) {
+ case "ContextMenu":
+ showContextMenu(div, e, area, ev);
+ ev.cancelBubble = true;
+ break;
+ case "None":
+ ev.cancelBubble = true;
+ break;
+ }
+
+ if (typeof area.onClicked === "function") {
+ area.onClicked(args);
+ }
+ };
+ })(div, e, area);
+
+ if (typeof area.onMouseEnter === "function") {
+ a.addEventListener("mouseenter", (function(div, e, area) {
+ return function(ev) {
+ var args = {};
+ args.area = area;
+ args.source = e;
+ args.originalEvent = ev;
+ area.onMouseEnter(args);
+ };
+ })(div, e, area));
+ }
+ if (typeof area.onMouseLeave === "function") {
+ a.addEventListener("mouseleave", (function(div, e, area) {
+ return function(ev) {
+ var args = {};
+ args.area = area;
+ args.source = e;
+ args.originalEvent = ev;
+ area.onMouseLeave(args);
+ };
+ })(div, e, area));
+ }
+
+ function doOnClick(area, source, originalEvent) {
+ var args = {};
+ args.area = area;
+ args.source = source;
+ args.originalEvent = originalEvent;
+ args.preventDefault = function() {
+ args.preventDefault.value = true;
+ };
+
+ if (typeof area.onClick === "function") {
+ area.onClick(args);
+ }
+ return args;
+ }
+
+ function showBubble(e, area, ev) {
+ if (DayPilot.Bubble) {
+ DayPilot.Bubble.touchPosition(ev);
+ }
+
+ if (e.calendar.bubble) {
+ e.calendar.bubble.showEvent(e, true);
+ }
+ }
+
+ function showContextMenu(div, e, area, ev) {
+ if (DayPilot.Menu) {
+ DayPilot.Menu.touchPosition(ev);
+ }
+
+ var m = area.contextMenu || area.menu;
+ if (!(m instanceof DayPilot.Menu)) {
+ if (e.isEvent && e.client.contextMenu()) {
+ m = e.client.contextMenu();
+ }
+ else if (e.isEvent && e.calendar.contextMenu) {
+ m = e.calendar.contextMenu;
+ }
+ }
+ if (m && m.show) {
+ var initiator = { "type": "area", "div": div, "e": e, "area": area, "a": a};
+ m.show(e, { "initiator": initiator});
+ }
+ }
+
+ function resolvePosVal(val) {
+ if (typeof val === "string" && isNaN(val)) {
+ return val;
+ }
+ else if (typeof val !== "undefined") {
+ return val + "px";
+ }
+ return undefined;
+ }
+
+ return a;
+ };
+
+ DayPilot.Areas.all = [];
+
+ DayPilot.Areas.hideAreas = function(div, ev) {
+ if (!div) {
+ return;
+ }
+
+ if (!div || !div.active) {
+ return;
+ }
+
+ var active = div.active;
+ var areas = div.areas;
+
+ if (active && active.children) {
+ if (ev) {
+ var target = ev.toElement || ev.relatedTarget;
+ if (~DayPilot.indexOf(active.children, target)) {
+ return;
+ }
+ }
+ }
+
+ if (!areas || areas.length === 0) {
+ div.active = null;
+ return;
+ }
+
+ DayPilot.de(areas);
+
+ div.active = null;
+ div.areas = [];
+
+ DayPilot.rfa(DayPilot.Areas.all, div);
+
+ active.children = null;
+ };
+
+ DayPilot.Areas.hideAll = function(ev) {
+ if (!DayPilot.Areas.all || DayPilot.Areas.all.length === 0) {
+ return;
+ }
+ for (var i = 0; i < DayPilot.Areas.all.length; i++) {
+ DayPilot.Areas.hideAreas(DayPilot.Areas.all[i], ev);
+ }
+
+ };
+
+ /* end of areas */
+
+ DayPilot.Exception = function(msg) {
+ return new Error(msg);
+ };
+
+ DayPilot.Locale = function(id, config) {
+ this.id = id;
+ this.dayNames = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
+ this.dayNamesShort = ["Su", "Mo", "Tu", "We", "Th", "Fr", "Sa"];
+ this.monthNames = ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"];
+ this.datePattern = "M/d/yyyy";
+ this.timePattern = "H:mm";
+ this.dateTimePattern = "M/d/yyyy H:mm";
+ this.timeFormat = "Clock12Hours";
+ this.weekStarts = 0;
+
+ if (config) {
+ for (var name in config) {
+ this[name] = config[name];
+ }
+ }
+ };
+
+ DayPilot.Locale.all = {};
+
+ DayPilot.Locale.find = function(id) {
+ if (!id) {
+ return null;
+ }
+ var normalized = id.toLowerCase();
+ if (normalized.length > 2) {
+ normalized = DayPilot.Util.replaceCharAt(normalized, 2, '-');
+ }
+ return DayPilot.Locale.all[normalized];
+ };
+
+ DayPilot.Locale.register = function(locale) {
+ DayPilot.Locale.all[locale.id] = locale;
+ };
+
+ DayPilot.Locale.register(new DayPilot.Locale('ca-es', {'dayNames':['diumenge','dilluns','dimarts','dimecres','dijous','divendres','dissabte'],'dayNamesShort':['dg','dl','dt','dc','dj','dv','ds'],'monthNames':['gener','febrer','març','abril','maig','juny','juliol','agost','setembre','octubre','novembre','desembre',''],'monthNamesShort':['gen.','febr.','març','abr.','maig','juny','jul.','ag.','set.','oct.','nov.','des.',''],'timePattern':'H:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('cs-cz', {'dayNames':['neděle','pondělí','úterý','středa','čtvrtek','pátek','sobota'],'dayNamesShort':['ne','po','út','st','čt','pá','so'],'monthNames':['leden','únor','březen','duben','květen','červen','červenec','srpen','září','říjen','listopad','prosinec',''],'monthNamesShort':['I','II','III','IV','V','VI','VII','VIII','IX','X','XI','XII',''],'timePattern':'H:mm','datePattern':'d. M. yyyy','dateTimePattern':'d. M. yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('da-dk', {'dayNames':['søndag','mandag','tirsdag','onsdag','torsdag','fredag','lørdag'],'dayNamesShort':['sø','ma','ti','on','to','fr','lø'],'monthNames':['januar','februar','marts','april','maj','juni','juli','august','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mar','apr','maj','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'HH:mm','datePattern':'dd-MM-yyyy','dateTimePattern':'dd-MM-yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-at', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Jänner','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jän','Feb','Mär','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-ch', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Januar','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jan','Feb','Mrz','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-de', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Januar','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jan','Feb','Mrz','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('de-lu', {'dayNames':['Sonntag','Montag','Dienstag','Mittwoch','Donnerstag','Freitag','Samstag'],'dayNamesShort':['So','Mo','Di','Mi','Do','Fr','Sa'],'monthNames':['Januar','Februar','März','April','Mai','Juni','Juli','August','September','Oktober','November','Dezember',''],'monthNamesShort':['Jan','Feb','Mrz','Apr','Mai','Jun','Jul','Aug','Sep','Okt','Nov','Dez',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-au', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'h:mm tt','datePattern':'d/MM/yyyy','dateTimePattern':'d/MM/yyyy h:mm tt','timeFormat':'Clock12Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-ca', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'h:mm tt','datePattern':'yyyy-MM-dd','dateTimePattern':'yyyy-MM-dd h:mm tt','timeFormat':'Clock12Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-gb', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('en-us', {'dayNames':['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],'dayNamesShort':['Su','Mo','Tu','We','Th','Fr','Sa'],'monthNames':['January','February','March','April','May','June','July','August','September','October','November','December',''],'monthNamesShort':['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec',''],'timePattern':'h:mm tt','datePattern':'M/d/yyyy','dateTimePattern':'M/d/yyyy h:mm tt','timeFormat':'Clock12Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('es-es', {'dayNames':['domingo','lunes','martes','miércoles','jueves','viernes','sábado'],'dayNamesShort':['D','L','M','X','J','V','S'],'monthNames':['enero','febrero','marzo','abril','mayo','junio','julio','agosto','septiembre','octubre','noviembre','diciembre',''],'monthNamesShort':['ene.','feb.','mar.','abr.','may.','jun.','jul.','ago.','sep.','oct.','nov.','dic.',''],'timePattern':'H:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('es-mx', {'dayNames':['domingo','lunes','martes','miércoles','jueves','viernes','sábado'],'dayNamesShort':['do.','lu.','ma.','mi.','ju.','vi.','sá.'],'monthNames':['enero','febrero','marzo','abril','mayo','junio','julio','agosto','septiembre','octubre','noviembre','diciembre',''],'monthNamesShort':['ene.','feb.','mar.','abr.','may.','jun.','jul.','ago.','sep.','oct.','nov.','dic.',''],'timePattern':'hh:mm tt','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy hh:mm tt','timeFormat':'Clock12Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('eu-es', {'dayNames':['igandea','astelehena','asteartea','asteazkena','osteguna','ostirala','larunbata'],'dayNamesShort':['ig','al','as','az','og','or','lr'],'monthNames':['urtarrila','otsaila','martxoa','apirila','maiatza','ekaina','uztaila','abuztua','iraila','urria','azaroa','abendua',''],'monthNamesShort':['urt.','ots.','mar.','api.','mai.','eka.','uzt.','abu.','ira.','urr.','aza.','abe.',''],'timePattern':'H:mm','datePattern':'yyyy/MM/dd','dateTimePattern':'yyyy/MM/dd H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fi-fi', {'dayNames':['sunnuntai','maanantai','tiistai','keskiviikko','torstai','perjantai','lauantai'],'dayNamesShort':['su','ma','ti','ke','to','pe','la'],'monthNames':['tammikuu','helmikuu','maaliskuu','huhtikuu','toukokuu','kesäkuu','heinäkuu','elokuu','syyskuu','lokakuu','marraskuu','joulukuu',''],'monthNamesShort':['tammi','helmi','maalis','huhti','touko','kesä','heinä','elo','syys','loka','marras','joulu',''],'timePattern':'H:mm','datePattern':'d.M.yyyy','dateTimePattern':'d.M.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-be', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd-MM-yy','dateTimePattern':'dd-MM-yy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-ch', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-fr', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('fr-lu', {'dayNames':['dimanche','lundi','mardi','mercredi','jeudi','vendredi','samedi'],'dayNamesShort':['di','lu','ma','me','je','ve','sa'],'monthNames':['janvier','février','mars','avril','mai','juin','juillet','août','septembre','octobre','novembre','décembre',''],'monthNamesShort':['janv.','févr.','mars','avr.','mai','juin','juil.','août','sept.','oct.','nov.','déc.',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('gl-es', {'dayNames':['domingo','luns','martes','mércores','xoves','venres','sábado'],'dayNamesShort':['do','lu','ma','mé','xo','ve','sá'],'monthNames':['xaneiro','febreiro','marzo','abril','maio','xuño','xullo','agosto','setembro','outubro','novembro','decembro',''],'monthNamesShort':['xan','feb','mar','abr','maio','xuño','xul','ago','set','out','nov','dec',''],'timePattern':'H:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('it-it', {'dayNames':['domenica','lunedì','martedì','mercoledì','giovedì','venerdì','sabato'],'dayNamesShort':['do','lu','ma','me','gi','ve','sa'],'monthNames':['gennaio','febbraio','marzo','aprile','maggio','giugno','luglio','agosto','settembre','ottobre','novembre','dicembre',''],'monthNamesShort':['gen','feb','mar','apr','mag','giu','lug','ago','set','ott','nov','dic',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('it-ch', {'dayNames':['domenica','lunedì','martedì','mercoledì','giovedì','venerdì','sabato'],'dayNamesShort':['do','lu','ma','me','gi','ve','sa'],'monthNames':['gennaio','febbraio','marzo','aprile','maggio','giugno','luglio','agosto','settembre','ottobre','novembre','dicembre',''],'monthNamesShort':['gen','feb','mar','apr','mag','giu','lug','ago','set','ott','nov','dic',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('ja-jp', {'dayNames':['日曜日','月曜日','火曜日','水曜日','木曜日','金曜日','土曜日'],'dayNamesShort':['日','月','火','水','木','金','土'],'monthNames':['1月','2月','3月','4月','5月','6月','7月','8月','9月','10月','11月','12月',''],'monthNamesShort':['1','2','3','4','5','6','7','8','9','10','11','12',''],'timePattern':'H:mm','datePattern':'yyyy/MM/dd','dateTimePattern':'yyyy/MM/dd H:mm','timeFormat':'Clock24Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('nb-no', {'dayNames':['søndag','mandag','tirsdag','onsdag','torsdag','fredag','lørdag'],'dayNamesShort':['sø','ma','ti','on','to','fr','lø'],'monthNames':['januar','februar','mars','april','mai','juni','juli','august','september','oktober','november','desember',''],'monthNamesShort':['jan','feb','mar','apr','mai','jun','jul','aug','sep','okt','nov','des',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('nl-nl', {'dayNames':['zondag','maandag','dinsdag','woensdag','donderdag','vrijdag','zaterdag'],'dayNamesShort':['zo','ma','di','wo','do','vr','za'],'monthNames':['januari','februari','maart','april','mei','juni','juli','augustus','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mrt','apr','mei','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'HH:mm','datePattern':'d-M-yyyy','dateTimePattern':'d-M-yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('nl-be', {'dayNames':['zondag','maandag','dinsdag','woensdag','donderdag','vrijdag','zaterdag'],'dayNamesShort':['zo','ma','di','wo','do','vr','za'],'monthNames':['januari','februari','maart','april','mei','juni','juli','augustus','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mrt','apr','mei','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'H:mm','datePattern':'d/MM/yyyy','dateTimePattern':'d/MM/yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('nn-no', {'dayNames':['søndag','måndag','tysdag','onsdag','torsdag','fredag','laurdag'],'dayNamesShort':['sø','må','ty','on','to','fr','la'],'monthNames':['januar','februar','mars','april','mai','juni','juli','august','september','oktober','november','desember',''],'monthNamesShort':['jan','feb','mar','apr','mai','jun','jul','aug','sep','okt','nov','des',''],'timePattern':'HH:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('pt-br', {'dayNames':['domingo','segunda-feira','terça-feira','quarta-feira','quinta-feira','sexta-feira','sábado'],'dayNamesShort':['D','S','T','Q','Q','S','S'],'monthNames':['janeiro','fevereiro','março','abril','maio','junho','julho','agosto','setembro','outubro','novembro','dezembro',''],'monthNamesShort':['jan','fev','mar','abr','mai','jun','jul','ago','set','out','nov','dez',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('pl-pl', {'dayNames':['niedziela','poniedziałek','wtorek','środa','czwartek','piątek','sobota'],'dayNamesShort':['N','Pn','Wt','Śr','Cz','Pt','So'],'monthNames':['styczeń','luty','marzec','kwiecień','maj','czerwiec','lipiec','sierpień','wrzesień','październik','listopad','grudzień',''],'monthNamesShort':['sty','lut','mar','kwi','maj','cze','lip','sie','wrz','paź','lis','gru',''],'timePattern':'HH:mm','datePattern':'yyyy-MM-dd','dateTimePattern':'yyyy-MM-dd HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('pt-pt', {'dayNames':['domingo','segunda-feira','terça-feira','quarta-feira','quinta-feira','sexta-feira','sábado'],'dayNamesShort':['D','S','T','Q','Q','S','S'],'monthNames':['janeiro','fevereiro','março','abril','maio','junho','julho','agosto','setembro','outubro','novembro','dezembro',''],'monthNamesShort':['jan','fev','mar','abr','mai','jun','jul','ago','set','out','nov','dez',''],'timePattern':'HH:mm','datePattern':'dd/MM/yyyy','dateTimePattern':'dd/MM/yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':0}));
+ DayPilot.Locale.register(new DayPilot.Locale('ro-ro', {'dayNames':['duminică','luni','marți','miercuri','joi','vineri','sâmbătă'],'dayNamesShort':['D','L','Ma','Mi','J','V','S'],'monthNames':['ianuarie','februarie','martie','aprilie','mai','iunie','iulie','august','septembrie','octombrie','noiembrie','decembrie',''],'monthNamesShort':['ian.','feb.','mar.','apr.','mai.','iun.','iul.','aug.','sep.','oct.','nov.','dec.',''],'timePattern':'H:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('ru-ru', {'dayNames':['воскресенье','понедельник','вторник','среда','четверг','пятница','суббота'],'dayNamesShort':['Вс','Пн','Вт','Ср','Чт','Пт','Сб'],'monthNames':['Январь','Февраль','Март','Апрель','Май','Июнь','Июль','Август','Сентябрь','Октябрь','Ноябрь','Декабрь',''],'monthNamesShort':['янв','фев','мар','апр','май','июн','июл','авг','сен','окт','ноя','дек',''],'timePattern':'H:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('sk-sk', {'dayNames':['nedeľa','pondelok','utorok','streda','štvrtok','piatok','sobota'],'dayNamesShort':['ne','po','ut','st','št','pi','so'],'monthNames':['január','február','marec','apríl','máj','jún','júl','august','september','október','november','december',''],'monthNamesShort':['1','2','3','4','5','6','7','8','9','10','11','12',''],'timePattern':'H:mm','datePattern':'d.M.yyyy','dateTimePattern':'d.M.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('sv-se', {'dayNames':['söndag','måndag','tisdag','onsdag','torsdag','fredag','lördag'],'dayNamesShort':['sö','må','ti','on','to','fr','lö'],'monthNames':['januari','februari','mars','april','maj','juni','juli','augusti','september','oktober','november','december',''],'monthNamesShort':['jan','feb','mar','apr','maj','jun','jul','aug','sep','okt','nov','dec',''],'timePattern':'HH:mm','datePattern':'yyyy-MM-dd','dateTimePattern':'yyyy-MM-dd HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('tr-tr', {'dayNames':['Pazar','Pazartesi','Salı','Çarşamba','Perşembe','Cuma','Cumartesi'],'dayNamesShort':['Pz','Pt','Sa','Ça','Pe','Cu','Ct'],'monthNames':['Ocak','Şubat','Mart','Nisan','Mayıs','Haziran','Temmuz','Ağustos','Eylül','Ekim','Kasım','Aralık',''],'monthNamesShort':['Oca','Şub','Mar','Nis','May','Haz','Tem','Ağu','Eyl','Eki','Kas','Ara',''],'timePattern':'HH:mm','datePattern':'d.M.yyyy','dateTimePattern':'d.M.yyyy HH:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('uk-ua', {'dayNames':['неділя','понеділок','вівторок','середа','четвер',"п'ятниця",'субота'],'dayNamesShort':['Нд','Пн','Вт','Ср','Чт','Пт','Сб'],'monthNames':['січень','лютий','березень','квітень','травень','червень','липень','серпень','вересень','жовтень','листопад','грудень',''],'monthNamesShort':['Січ','Лют','Бер','Кві','Тра','Чер','Лип','Сер','Вер','Жов','Лис','Гру',''],'timePattern':'H:mm','datePattern':'dd.MM.yyyy','dateTimePattern':'dd.MM.yyyy H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('zh-cn', {'dayNames':['星期日','星期一','星期二','星期三','星期四','星期五','星期六'],'dayNamesShort':['日','一','二','三','四','五','六'],'monthNames':['一月','二月','三月','四月','五月','六月','七月','八月','九月','十月','十一月','十二月',''],'monthNamesShort':['1月','2月','3月','4月','5月','6月','7月','8月','9月','10月','11月','12月',''],'timePattern':'H:mm','datePattern':'yyyy/M/d','dateTimePattern':'yyyy/M/d H:mm','timeFormat':'Clock24Hours','weekStarts':1}));
+ DayPilot.Locale.register(new DayPilot.Locale('zh-tw', {'dayNames':['星期日','星期一','星期二','星期三','星期四','星期五','星期六'],'dayNamesShort':['日','一','二','三','四','五','六'],'monthNames':['一月','二月','三月','四月','五月','六月','七月','八月','九月','十月','十一月','十二月',''],'monthNamesShort':['一月','二月','三月','四月','五月','六月','七月','八月','九月','十月','十一月','十二月',''],'timePattern':'tt hh:mm','datePattern':'yyyy/M/d','dateTimePattern':'yyyy/M/d tt hh:mm','timeFormat':'Clock12Hours','weekStarts':0}));
+
+ DayPilot.Locale.US = DayPilot.Locale.find("en-us");
+
+ /**
+ * Created on 2023-04-04.
+ */
+
+ DayPilot.Switcher = function(options) {
+
+ var This = this;
+
+ this._views = [];
+ this._triggers = [];
+ this._navigator = {};
+
+ this.selectedClass = null;
+
+ this._active = null;
+
+ this._day = DayPilot.Date.today();
+
+ this.onChange = null;
+ this.onChanged = null;
+ this.onSelect = null;
+
+ this._navigator.updateMode = function (mode) {
+ var control = This._navigator.control;
+ if (!control) {
+ return;
+ }
+ control.selectMode = mode;
+ control.select(This._day);
+ };
+
+ this.addView = function (spec, options) {
+ var element;
+ if (typeof spec === 'string') {
+ element = document.getElementById(spec);
+ if (!element) {
+ throw "Element not found: " + spec;
+ }
+ }
+ else { // DayPilot object, DOM element
+ element = spec;
+ }
+
+ var control = element;
+
+ var view = {};
+ view._isView = true;
+ view._id = control.id;
+ view.control = control;
+ view._options = options || {};
+ view._hide = function () {
+ if (control.hide) {
+ control.hide();
+ }
+ else if (control.nav && control.nav.top) {
+ control.nav.top.style.display = 'none';
+ }
+ else {
+ control.style.display = 'none';
+ }
+ };
+ view._sendNavigate = function(date) {
+ var serverBased = (function() {
+ if (control.backendUrl) { // ASP.NET MVC, Java
+ return true;
+ }
+ if (typeof WebForm_DoCallback === 'function' && control.uniqueID) { // ASP.NET WebForms
+ return true;
+ }
+ return false;
+ })();
+ if (serverBased) {
+ if (control.commandCallBack) {
+ control.commandCallBack("navigate", { "day": date });
+ }
+ }
+ else {
+ control.startDate = date;
+ control.update();
+ }
+ };
+ view._show = function () {
+ This._hideViews();
+ if (control.show) {
+ control.show();
+ }
+ else if (control.nav && control.nav.top) {
+ control.nav.top.style.display = '';
+ }
+ else {
+ control.style.display = '';
+ }
+ };
+ view._selectMode = function () { // for navigator
+ if (view._options.navigatorSelectMode) {
+ return view._options.navigatorSelectMode;
+ }
+
+ if (control.isCalendar) {
+ switch (control.viewType) {
+ case "Day":
+ return "day";
+ case "Week":
+ return "week";
+ case "WorkWeek":
+ return "week";
+ default:
+ return "day";
+ }
+ }
+ else if (control.isMonth) {
+ switch (control.viewType) {
+ case "Month":
+ return "month";
+ case "Weeks":
+ return "week";
+ default:
+ return "day";
+ }
+ }
+ return "day";
+ };
+
+ this._views.push(view);
+
+ return view;
+ };
+
+ this.addTrigger = function (id, control) {
+ var element;
+ if (typeof id === 'string') {
+ element = document.getElementById(id);
+ if (!element) {
+ throw "Element not found: " + id;
+ }
+ }
+ else {
+ element = id;
+ }
+
+ var view = this._findViewByControl(control);
+ if (!view) {
+ view = this.addView(control);
+ }
+
+ var trigger = {};
+ trigger._isTrigger = true;
+ trigger._element = element;
+ trigger._id = element.id;
+ trigger._view = view;
+ trigger._onClick = function (ev) {
+
+ This.show(trigger);
+ This._select(trigger);
+
+ if (ev) {
+ ev.preventDefault ? ev.preventDefault() : ev.returnValue = false;
+ }
+
+ };
+
+ DayPilot.re(element, 'click', trigger._onClick);
+
+ this._triggers.push(trigger);
+
+ return trigger;
+ };
+
+ // backwards compatibility
+ this.addButton = this.addTrigger;
+
+ this.select = function(id) {
+ var trigger = this._findTriggerById(id);
+ if (trigger) {
+ trigger._onClick();
+ }
+ else if (this._triggers.length > 0) {
+ this._triggers[0]._onClick();
+ }
+ };
+
+ this._findTriggerById = function(id) {
+ for (var i = 0; i < this._triggers.length; i++) {
+ var trigger = this._triggers[i];
+ if (trigger._id === id) {
+ return trigger;
+ }
+ }
+ return null;
+ };
+
+ this._select = function(trigger) {
+ if (!this.selectedClass) {
+ return;
+ }
+
+ for (var i = 0; i < this._triggers.length; i++) {
+ var s = this._triggers[i];
+ DayPilot.Util.removeClass(s._element, this.selectedClass);
+ }
+ DayPilot.Util.addClass(trigger._element, this.selectedClass);
+ };
+
+ this.addNavigator = function (control) {
+ //this.navigator = {};
+ This._navigator.control = control;
+
+ control.timeRangeSelectedHandling = "JavaScript";
+ control.onTimeRangeSelected = function() {
+ var start, end, day;
+ if (control.api === 1) {
+ start = arguments[0];
+ end = arguments[1];
+ day = arguments[2];
+ }
+ else {
+ var args = arguments[0];
+ start = args.start;
+ end = args.end;
+ day = args.day;
+ }
+ This._day = day;
+
+ navigate(start, end, day);
+
+ };
+ };
+
+ this.show = function (el) {
+ var view, trigger;
+ if (el._isTrigger) {
+ trigger = el;
+ view = trigger._view;
+ }
+ else {
+ view = el._isView ? el : this._findViewByControl(el);
+ if (this._active === view) {
+ return;
+ }
+ }
+
+ if (This.onSelect) {
+ var args = {};
+ args.source = trigger ? trigger._element : null;
+ args.target = view.control;
+
+ This.onSelect(args);
+ // TODO add preventDefault
+ }
+
+ this._active = view;
+ view._show();
+
+ var mode = view._selectMode();
+ This._navigator.updateMode(mode);
+
+ //This.navigator.select(This.day);
+
+ //This.active.sendNavigate(this.day);
+
+ // this ensures first onChange call if single day = today is displayed
+ var start = This._navigator.control.selectionStart;
+ var end = This._navigator.control.selectionEnd.addDays(1);
+ var day = This._navigator.control.selectionDay;
+ navigate(start, end, day);
+ };
+
+ this._findViewByControl = function (control) {
+ for (var i = 0; i < this._views.length; i++) {
+ if (this._views[i].control === control) {
+ return this._views[i];
+ }
+ }
+ return null;
+ };
+
+ this._hideViews = function () {
+ //var controls = [dp_day, dp_week, dp_month];
+ for (var i = 0; i < this._views.length; i++) {
+ this._views[i]._hide();
+ }
+ };
+
+ Object.defineProperty(this, "active", {
+ get: function() {
+ return This._active;
+ }
+ });
+
+ this.events = {};
+
+ this.events.load = function(url, success, error) {
+ if (This._active && This._active.control) {
+ This._active.control.events.load(url, success, error);
+ }
+ else {
+ throw "DayPilot.Switcher.events.load(): Active view not found";
+ }
+ };
+
+ this._previousArgs = null;
+
+ this._init = function() {
+ if (!options) {
+ return;
+ }
+
+ for (var name in options) {
+ if (name === "triggers") {
+ var triggers = options.triggers || [];
+ triggers.forEach(function(item) {
+ This.addTrigger(item.id, item.view);
+ });
+ }
+ else if (name === "navigator") {
+ This.addNavigator(options.navigator);
+ }
+ else {
+ This[name] = options[name];
+ }
+ }
+
+ };
+
+ this._init();
+
+ function navigate(start, end, day) {
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.day = day;
+ args.target = This._active.control;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ var previous = This._previousArgs;
+ if (previous) {
+ if (previous.start === args.start && previous.end === args.end && previous.day === args.day && previous.target === args.target) {
+ return; // duplicate, no change
+ }
+ }
+
+ This._previousArgs = args;
+
+ if (typeof This.onChange === "function") {
+ This.onChange(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // backwards compatibility
+ if (typeof This.onTimeRangeSelect === "function") {
+ This.onTimeRangeSelect(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ This._active._sendNavigate(This._day);
+
+ if (typeof This.onChanged === "function") {
+ This.onChanged(args);
+ }
+
+ if (typeof This.onTimeRangeSelected === "function") {
+ This.onTimeRangeSelected(args);
+ }
+
+ }
+
+ };
+
+ // DayPilot.Date START
+
+ DayPilot.Duration = function(ticks) {
+ var d = this;
+
+ var day = 1000*60*60*24.0;
+ var hour = 1000*60*60.0;
+ var minute = 1000*60.0;
+ var second = 1000.0;
+
+ if (arguments.length === 2) {
+ var start = arguments[0];
+ var end = arguments[1];
+
+ if (!(start instanceof DayPilot.Date) && (typeof start !== "string")) {
+ throw "DayPilot.Duration(): Invalid start argument, DayPilot.Date expected";
+ }
+ if (!(end instanceof DayPilot.Date) && (typeof end !== "string")) {
+ throw "DayPilot.Duration(): Invalid end argument, DayPilot.Date expected";
+ }
+ if (typeof start === "string") {
+ start = new DayPilot.Date(start);
+ }
+ if (typeof end === "string") {
+ end = new DayPilot.Date(end);
+ }
+ ticks = end.getTime() - start.getTime();
+ }
+
+ this.ticks = ticks;
+
+ // caching, allows direct comparison
+ if (DayPilot.Date.Cache.DurationCtor["" + ticks]) {
+ return DayPilot.Date.Cache.DurationCtor["" + ticks];
+ }
+ DayPilot.Date.Cache.DurationCtor["" + ticks] = this;
+
+ this.toString = function(pattern) {
+ if (!pattern) {
+ return d.days() + "." + d.hours() + ":" + d.minutes() + ":" + d.seconds() + "." + d.milliseconds();
+ }
+
+ var minutes = d.minutes();
+ minutes = (minutes < 10 ? "0" : "") + minutes;
+
+ // dumb replacement
+ var result = pattern;
+ result = result.replace("mm", minutes);
+ result = result.replace("m", d.minutes());
+ result = result.replace("H", d.hours());
+ result = result.replace("h", d.hours());
+ result = result.replace("d", d.days());
+ result = result.replace("s", d.seconds());
+ return result;
+ };
+
+ this.totalHours = function() {
+ return d.ticks / hour;
+ };
+
+ this.totalDays = function() {
+ return d.ticks / day;
+ };
+
+ this.totalMinutes = function() {
+ return d.ticks / minute;
+ };
+
+ this.totalSeconds = function() {
+ return d.ticks / second;
+ };
+
+ this.days = function() {
+ return Math.floor(d.totalDays());
+ };
+
+ this.hours = function() {
+ var hourPartTicks = d.ticks - d.days()*day;
+ return Math.floor(hourPartTicks/hour);
+ };
+
+ this.minutes = function() {
+ var minutePartTicks = d.ticks - Math.floor(d.totalHours()) * hour;
+ return Math.floor(minutePartTicks/minute);
+ };
+
+ this.seconds = function() {
+ var secondPartTicks = d.ticks - Math.floor(d.totalMinutes()) * minute;
+ return Math.floor(secondPartTicks/second);
+ };
+
+ this.milliseconds = function() {
+ return d.ticks % second;
+ };
+
+ };
+
+ DayPilot.Duration.weeks = function(i) {
+ return new DayPilot.Duration(i * 1000*60*60*24*7);
+ };
+
+ DayPilot.Duration.days = function(i) {
+ return new DayPilot.Duration(i * 1000*60*60*24);
+ };
+
+ DayPilot.Duration.hours = function(i) {
+ return new DayPilot.Duration(i * 1000*60*60);
+ };
+
+ DayPilot.Duration.minutes = function(i) {
+ return new DayPilot.Duration(i * 1000*60);
+ };
+
+ DayPilot.Duration.seconds = function(i) {
+ return new DayPilot.Duration(i * 1000);
+ };
+
+ // alias to DayPilot.Duration
+ // disabled, doesn't work with caching
+ DayPilot.TimeSpan = function() {
+
+ throw "Please use DayPilot.Duration class instead of DayPilot.TimeSpan.";
+ // DayPilot.Duration.apply(this, arguments);
+ };
+ try {
+ DayPilot.TimeSpan.prototype = Object.create(DayPilot.Duration.prototype); // make instanceof work
+ }
+ catch (e) {} // doesn't work in IE8
+
+ // DayPilot.TimeSpan.prototype.constructor = DayPilot.TimeSpan; // not necessary, it's an alias, not an inherited class
+
+ /* Date utils */
+
+ // DayPilot.Date class
+ /* Constructor signatures:
+
+ -- new DayPilot.Date(date, isLocal)
+ date - JavaScript Date object
+ isLocal - true if the local time should be taken from date, otherwise GMT base is used
+
+ -- new DayPilot.Date() - returns now, using local date
+
+ -- new DayPilot.Date(string)
+ string - date in ISO 8601 format, e.g. 2009-01-01T00:00:00
+
+ */
+ DayPilot.Date = function(date, readLocal) {
+
+ if (date instanceof DayPilot.Date) { // it's already a DayPilot.Date object, return it (no copy)
+ return date;
+ }
+
+ var ticks;
+
+ if (DayPilot.Util.isNullOrUndefined(date)) { // date not set, use NOW
+ ticks = DayPilot.DateUtil.fromLocal().getTime();
+ date = ticks;
+ }
+
+ var cache = DayPilot.Date.Cache.Ctor;
+ if (cache[date]) {
+ DayPilot.Stats.cacheHitsCtor += 1;
+ return cache[date];
+ }
+
+ var isString = false;
+
+ if (typeof date === "string") {
+ ticks = DayPilot.DateUtil.fromStringSortable(date, readLocal).getTime();
+ isString = true;
+ }
+ else if (typeof date === "number") {
+ if (isNaN(date)) {
+ throw "Cannot create DayPilot.Date from NaN";
+ }
+ ticks = date;
+ }
+ else if (date instanceof Date) {
+ if (readLocal) {
+ ticks = DayPilot.DateUtil.fromLocal(date).getTime();
+ }
+ else {
+ ticks = date.getTime();
+ }
+ }
+ else {
+ throw "Unrecognized parameter: use Date, number or string in ISO 8601 format";
+ }
+
+ var value = ticksToSortable(ticks); // normalized value
+
+ if (cache[value]) {
+ return cache[value];
+ }
+
+ cache[value] = this;
+ cache[ticks] = this;
+ if (isString && value !== date && DayPilot.DateUtil.hasTzSpec(date)) { // don't cache strings with TZ spec
+ cache[date] = this;
+ }
+
+ if (Object.defineProperty) {
+ Object.defineProperty(this, "ticks", {
+ get: function() { return ticks; }
+ });
+ Object.defineProperty(this, "value", {
+ "value": value,
+ "writable": false,
+ "enumerable": true
+ });
+ }
+ else {
+ this.ticks = ticks;
+ this.value = value;
+ }
+
+ if (DayPilot.Date.Config.legacyShowD) {
+ this.d = new Date(ticks);
+ }
+
+ DayPilot.Stats.dateObjects += 1;
+ };
+
+ DayPilot.Date.Config = {};
+ DayPilot.Date.Config.legacyShowD = false;
+
+ DayPilot.Date.Cache = {};
+ DayPilot.Date.Cache.Parsing = {};
+ DayPilot.Date.Cache.Ctor = {};
+ DayPilot.Date.Cache.Ticks = {};
+ DayPilot.Date.Cache.DurationCtor = {};
+
+ DayPilot.Date.Cache.clear = function() {
+ DayPilot.Date.Cache.Parsing = {};
+ DayPilot.Date.Cache.Ctor = {};
+ DayPilot.Date.Cache.Ticks = {};
+ DayPilot.Date.Cache.DurationCtor = {};
+ };
+
+
+ DayPilot.Date.prototype.addDays = function(days) {
+ if (!days) {
+ return this;
+ }
+ return new DayPilot.Date(this.ticks + days * 24 * 60 * 60 * 1000);
+ };
+
+ DayPilot.Date.prototype.addHours = function(hours) {
+ if (!hours) {
+ return this;
+ }
+ return this.addTime(hours * 60 * 60 * 1000);
+ };
+
+ DayPilot.Date.prototype.addMilliseconds = function(millis) {
+ if (!millis) {
+ return this;
+ }
+ return this.addTime(millis);
+ };
+
+ DayPilot.Date.prototype.addMinutes = function(minutes) {
+ if (!minutes) {
+ return this;
+ }
+ return this.addTime(minutes * 60 * 1000);
+ };
+
+ DayPilot.Date.prototype.addMonths = function(months) {
+ if (!months) {
+ return this;
+ }
+
+ var date = new Date(this.ticks);
+
+ var y = date.getUTCFullYear();
+ var m = date.getUTCMonth() + 1;
+
+ if (months > 0) {
+ while (months >= 12) {
+ months -= 12;
+ y++;
+ }
+ if (months > 12 - m) {
+ y++;
+ m = months - (12 - m);
+ }
+ else {
+ m += months;
+ }
+ }
+ else {
+ while (months <= -12) {
+ months += 12;
+ y--;
+ }
+ if (m + months <= 0) { //
+ y--;
+ m = 12 + m + months;
+ }
+ else {
+ m = m + months;
+ }
+ }
+
+ var d = new Date(date.getTime());
+ d.setUTCDate(1);
+ d.setUTCFullYear(y);
+ d.setUTCMonth(m - 1);
+
+ //var max = DayPilot.Date.daysInMonth(y, m);
+ var max = new DayPilot.Date(d).daysInMonth();
+ d.setUTCDate(Math.min(max, date.getUTCDate()));
+
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.addSeconds = function(seconds) {
+ if (!seconds) {
+ return this;
+ }
+ return this.addTime(seconds * 1000);
+ };
+
+ DayPilot.Date.prototype.addTime = function(ticks) {
+ if (!ticks) {
+ return this;
+ }
+ if (ticks instanceof DayPilot.Duration) {
+ ticks = ticks.ticks;
+ }
+ return new DayPilot.Date(this.ticks + ticks);
+ };
+
+ DayPilot.Date.prototype.addYears = function(years) {
+ var original = new Date(this.ticks);
+ var d = new Date(this.ticks);
+ var y = this.getYear() + years;
+ var m = this.getMonth();
+
+ d.setUTCDate(1);
+ d.setUTCFullYear(y);
+ d.setUTCMonth(m);
+
+ //var max = DayPilot.Date.daysInMonth(y, m + 1);
+ var max = new DayPilot.Date(d).daysInMonth();
+ d.setUTCDate(Math.min(max, original.getUTCDate()));
+
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.dayOfWeek = function() {
+ return new Date(this.ticks).getUTCDay();
+ };
+
+ DayPilot.Date.prototype.getDayOfWeek = function() {
+ return new Date(this.ticks).getUTCDay();
+ };
+
+ DayPilot.Date.prototype.getDayOfYear = function() {
+ var first = this.firstDayOfYear();
+ return DayPilot.DateUtil.daysDiff(first, this) + 1;
+ };
+
+ DayPilot.Date.prototype.daysInMonth = function() {
+ var date = new Date(this.ticks);
+ var month = date.getUTCMonth() + 1;
+ var year = date.getUTCFullYear();
+
+
+ var m = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];
+ if (month !== 2)
+ return m[month - 1];
+ if (year % 4 !== 0)
+ return m[1];
+ if (year % 100 === 0 && year % 400 !== 0)
+ return m[1];
+ return m[1] + 1;
+
+ };
+
+ DayPilot.Date.prototype.daysInYear = function() {
+ var year = this.getYear();
+ if (year % 4 !== 0) {
+ return 365;
+ }
+ if (year % 100 === 0 && year % 400 !== 0) {
+ return 365;
+ }
+ return 366;
+ };
+
+ DayPilot.Date.prototype.dayOfYear = function() {
+ return Math.ceil((this.getDatePart().getTime() - this.firstDayOfYear().getTime()) / 86400000) + 1;
+ };
+
+ // not required, direct comparison can be used
+ DayPilot.Date.prototype.equals = function(another) {
+ if (another === null) {
+ return false;
+ }
+ if (another instanceof DayPilot.Date) {
+ return this === another;
+ }
+ else {
+ throw "The parameter must be a DayPilot.Date object (DayPilot.Date.equals())";
+ }
+ };
+
+ DayPilot.Date.prototype.firstDayOfMonth = function() {
+ //var utc = DayPilot.Date.firstDayOfMonth(this.getYear(), this.getMonth() + 1);
+ //return new DayPilot.Date(utc);
+
+ var d = new Date();
+ d.setUTCFullYear(this.getYear(), this.getMonth(), 1);
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ return new DayPilot.Date(d);
+
+ };
+
+ DayPilot.Date.prototype.firstDayOfYear = function() {
+ var year = this.getYear();
+ var d = new Date();
+ d.setUTCFullYear(year, 0, 1);
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.firstDayOfWeek = function(weekStarts) {
+ var d = this;
+ if (weekStarts instanceof DayPilot.Locale) {
+ weekStarts = weekStarts.weekStarts;
+ }
+ else if (typeof weekStarts === "string" && DayPilot.Locale.find(weekStarts)) {
+ var locale = DayPilot.Locale.find(weekStarts);
+ weekStarts = locale.weekStarts;
+ }
+ else {
+ weekStarts = weekStarts || 0;
+ }
+
+ var day = d.dayOfWeek();
+ while (day !== weekStarts) {
+ d = d.addDays(-1);
+ day = d.dayOfWeek();
+ }
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.getDay = function() {
+ return new Date(this.ticks).getUTCDate();
+ };
+
+ DayPilot.Date.prototype.getDatePart = function() {
+ var d = new Date(this.ticks);
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.getYear = function() {
+ return new Date(this.ticks).getUTCFullYear();
+ };
+
+ DayPilot.Date.prototype.getHours = function() {
+ return new Date(this.ticks).getUTCHours();
+ };
+
+ DayPilot.Date.prototype.getMilliseconds = function() {
+ return new Date(this.ticks).getUTCMilliseconds();
+ };
+
+ DayPilot.Date.prototype.getMinutes = function() {
+ return new Date(this.ticks).getUTCMinutes();
+ };
+
+ DayPilot.Date.prototype.getMonth = function() {
+ return new Date(this.ticks).getUTCMonth();
+ };
+
+ DayPilot.Date.prototype.getSeconds = function() {
+ return new Date(this.ticks).getUTCSeconds();
+ };
+
+ DayPilot.Date.prototype.getTotalTicks = function() {
+ return this.getTime();
+ };
+
+ // undocumented
+ DayPilot.Date.prototype.getTime = function() {
+ return this.ticks;
+ };
+
+ DayPilot.Date.prototype.getTimePart = function() {
+ var datePart = this.getDatePart();
+ return DayPilot.DateUtil.diff(this, datePart);
+ };
+
+ DayPilot.Date.prototype.lastDayOfMonth = function() {
+ //var utc = DayPilot.Date.lastDayOfMonth(this.getYear(), this.getMonth() + 1);
+ //return new DayPilot.Date(utc);
+ var d = new Date(this.firstDayOfMonth().getTime());
+ var length = this.daysInMonth();
+ d.setUTCDate(length);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.Date.prototype.weekNumber = function() {
+ var first = this.firstDayOfYear();
+ var days = (this.getTime() - first.getTime()) / 86400000;
+ return Math.ceil((days + first.dayOfWeek() + 1) / 7);
+ };
+
+ // ISO 8601
+ DayPilot.Date.prototype.weekNumberISO = function() {
+ var thursdayFlag = false;
+ var dayOfYear = this.dayOfYear();
+
+ var startWeekDayOfYear = this.firstDayOfYear().dayOfWeek();
+ var endWeekDayOfYear = this.firstDayOfYear().addYears(1).addDays(-1).dayOfWeek();
+ //int startWeekDayOfYear = new DateTime(date.getYear(), 1, 1).getDayOfWeekOrdinal();
+ //int endWeekDayOfYear = new DateTime(date.getYear(), 12, 31).getDayOfWeekOrdinal();
+
+ if (startWeekDayOfYear === 0) {
+ startWeekDayOfYear = 7;
+ }
+ if (endWeekDayOfYear === 0) {
+ endWeekDayOfYear = 7;
+ }
+
+ var daysInFirstWeek = 8 - (startWeekDayOfYear);
+
+ if (startWeekDayOfYear === 4 || endWeekDayOfYear === 4) {
+ thursdayFlag = true;
+ }
+
+ var fullWeeks = Math.ceil((dayOfYear - (daysInFirstWeek)) / 7.0);
+
+ var weekNumber = fullWeeks;
+
+ if (daysInFirstWeek >= 4) {
+ weekNumber = weekNumber + 1;
+ }
+
+ if (weekNumber > 52 && !thursdayFlag) {
+ weekNumber = 1;
+ }
+
+ if (weekNumber === 0) {
+ weekNumber = this.firstDayOfYear().addDays(-1).weekNumberISO(); //weekNrISO8601(new DateTime(date.getYear() - 1, 12, 31));
+ }
+
+ return weekNumber;
+
+ };
+
+ DayPilot.Date.prototype.toDateLocal = function() {
+ var date = new Date(this.ticks);
+
+ var d = new Date();
+ d.setFullYear(date.getUTCFullYear(), date.getUTCMonth(), date.getUTCDate());
+ d.setHours(date.getUTCHours());
+ d.setMinutes(date.getUTCMinutes());
+ d.setSeconds(date.getUTCSeconds());
+ d.setMilliseconds(date.getUTCMilliseconds());
+ return d;
+
+ };
+
+ DayPilot.Date.prototype.toDate = function() {
+ return new Date(this.ticks);
+ };
+
+ DayPilot.Date.prototype.toJSON = function() {
+ return this.value;
+ };
+
+ // formatting and languages needed here
+ DayPilot.Date.prototype.toString = function(pattern, locale) {
+ if (!pattern) {
+ return this.toStringSortable();
+ }
+ return new Pattern(pattern, locale).print(this);
+ };
+
+ DayPilot.Date.prototype.toStringSortable = function() {
+ return ticksToSortable(this.ticks);
+ };
+
+ function ticksToSortable(ticks) {
+
+ var cache = DayPilot.Date.Cache.Ticks;
+ if (cache[ticks]) {
+ DayPilot.Stats.cacheHitsTicks += 1;
+ return cache[ticks];
+ }
+
+ var d = new Date(ticks);
+
+ var millisecond;
+ var ms = d.getUTCMilliseconds();
+
+ if (ms === 0) {
+ millisecond = "";
+ }
+ else if (ms < 10) {
+ millisecond = ".00" + ms;
+ }
+ else if (ms < 100) {
+ millisecond = ".0" + ms;
+ }
+ else {
+ millisecond = "." + ms
+ }
+
+ var second = d.getUTCSeconds();
+ if (second < 10)
+ second = "0" + second;
+ var minute = d.getUTCMinutes();
+ if (minute < 10)
+ minute = "0" + minute;
+ var hour = d.getUTCHours();
+ if (hour < 10)
+ hour = "0" + hour;
+ var day = d.getUTCDate();
+ if (day < 10)
+ day = "0" + day;
+ var month = d.getUTCMonth() + 1;
+ if (month < 10)
+ month = "0" + month;
+ var year = d.getUTCFullYear();
+
+ if (year <= 0) {
+ throw "The minimum year supported is 1.";
+ }
+ if (year < 10) {
+ year = "000" + year;
+ }
+ else if (year < 100) {
+ year = "00" + year;
+ }
+ else if (year < 1000) {
+ year = "0" + year;
+ }
+
+ var result = year + "-" + month + "-" + day + 'T' + hour + ":" + minute + ":" + second + millisecond;
+ cache[ticks] = result;
+ return result;
+ }
+
+ /* static functions, return DayPilot.Date object */
+
+ // returns null if parsing was not successful
+ DayPilot.Date.parse = function(str, pattern, locale) {
+ var p = new Pattern(pattern, locale);
+ return p.parse(str);
+ };
+
+ var todayCount = 0;
+
+ DayPilot.Date.today = function() {
+ //return new DayPilot.Date().getDatePart();
+ return new DayPilot.Date(DayPilot.DateUtil.localToday(), true);
+ };
+
+ DayPilot.Date.now = function() {
+ return new DayPilot.Date();
+ };
+
+ DayPilot.Date.fromYearMonthDay = function(year, month, day) {
+ month = month || 1;
+ day = day || 1;
+
+ var d = new Date(0);
+ d.setUTCFullYear(year);
+ d.setUTCMonth(month - 1);
+ d.setUTCDate(day);
+ return new DayPilot.Date(d);
+ };
+
+ DayPilot.DateUtil = {};
+
+ /* internal functions, all operate with GMT base of the date object
+ (except of DayPilot.DateUtil.fromLocal()) */
+
+ DayPilot.DateUtil.fromStringSortable = function(string, readLocal) {
+ /*
+ Supported formats:
+ 2015-01-01
+ 2015-01-01T00:00:00
+ 2015-01-01T00:00:00.000
+ 2015-01-01T00:00:00Z
+ 2015-01-01T00:00:00.000Z
+ 2015-01-01T00:00:00+01:00
+ 2015-01-01T00:00:00.000+01:00
+
+ */
+
+ if (!string) {
+ throw "Can't create DayPilot.Date from an empty string";
+ }
+
+ var len = string.length;
+ var date = len === 10;
+ var datetime = len === 19;
+ var long = len > 19;
+
+ if (!date && !datetime && !long) {
+ throw "Invalid string format (use '2010-01-01' or '2010-01-01T00:00:00'): " + string;
+ }
+
+ if (DayPilot.Date.Cache.Parsing[string] && !readLocal) {
+ DayPilot.Stats.cacheHitsParsing += 1;
+ return DayPilot.Date.Cache.Parsing[string];
+ }
+
+ var year = string.substring(0, 4);
+ var month = string.substring(5, 7);
+ var day = string.substring(8, 10);
+
+ var d = new Date(0);
+ d.setUTCFullYear(year, month - 1, day);
+
+ if (date) {
+ /*
+ d.setUTCHours(0);
+ d.setUTCMinutes(0);
+ d.setUTCSeconds(0);
+ d.setUTCMilliseconds(0);
+ */
+ //result = d;
+ DayPilot.Date.Cache.Parsing[string] = d;
+ return d;
+ }
+
+ var hours = string.substring(11, 13);
+ var minutes = string.substring(14, 16);
+ var seconds = string.substring(17, 19);
+
+ d.setUTCHours(hours);
+ d.setUTCMinutes(minutes);
+ d.setUTCSeconds(seconds);
+ //d.setUTCMilliseconds(0);
+ //result = d;
+
+ if (datetime) {
+ DayPilot.Date.Cache.Parsing[string] = d;
+ return d;
+ }
+
+ var tzdir = string[19];
+
+ var tzoffset = 0;
+
+ if (tzdir === ".") {
+ var ms = parseInt(string.substring(20, 23)); /// .000
+ d.setUTCMilliseconds(ms);
+ tzoffset = DayPilot.DateUtil.getTzOffsetMinutes(string.substring(23));
+ }
+ else {
+ tzoffset = DayPilot.DateUtil.getTzOffsetMinutes(string.substring(19));
+ }
+
+ var dd = new DayPilot.Date(d);
+ if (!readLocal) {
+ dd = dd.addMinutes(-tzoffset);
+ }
+
+ d = dd.toDate(); // get UTC base
+
+ DayPilot.Date.Cache.Parsing[string] = d;
+ return d;
+ };
+
+ DayPilot.DateUtil.getTzOffsetMinutes = function(string) {
+ if (DayPilot.Util.isNullOrUndefined(string) || string === "") {
+ return 0;
+ }
+ if (string === "Z") {
+ return 0;
+ }
+
+ var tzdir = string[0];
+
+ var tzhours = parseInt(string.substring(1, 3));
+ var tzminutes = parseInt(string.substring(4));
+ var tzoffset = tzhours * 60 + tzminutes;
+
+ if (tzdir === "-") {
+ return -tzoffset;
+ }
+ else if (tzdir === "+") {
+ return tzoffset;
+ }
+ else {
+ throw "Invalid timezone spec: " + string;
+ }
+ };
+
+ DayPilot.DateUtil.hasTzSpec = function(string) {
+ if (string.indexOf("+")) {
+ return true;
+ }
+ if (string.indexOf("-")) {
+ return true;
+ }
+ return false;
+ };
+
+
+ // rename candidate: diffDays
+ DayPilot.DateUtil.daysDiff = function(first, second) {
+ (first && second) || (function() { throw "two parameters required"; })();
+
+ first = new DayPilot.Date(first);
+ second = new DayPilot.Date(second);
+
+ if (first.getTime() > second.getTime()) {
+ return null;
+ }
+
+ var i = 0;
+ var fDay = first.getDatePart();
+ var sDay = second.getDatePart();
+
+ while (fDay < sDay) {
+ fDay = fDay.addDays(1);
+ i++;
+ }
+
+ return i;
+ };
+
+ DayPilot.DateUtil.daysSpan = function(first, second) {
+ (first && second) || (function() { throw "two parameters required"; })();
+
+ first = new DayPilot.Date(first);
+ second = new DayPilot.Date(second);
+
+ if (first.getTime() === second.getTime()) {
+ return 0;
+ }
+
+ var diff = DayPilot.DateUtil.daysDiff(first, second);
+
+ if (second.getTime() == second.getDatePart().getTime()) {
+ diff--;
+ }
+
+ return diff;
+ };
+
+ DayPilot.DateUtil.diff = function(first, second) { // = first - second
+ if (!(first && second && first.getTime && second.getTime)) {
+ throw "Both compared objects must be Date objects (DayPilot.Date.diff).";
+ }
+
+ return first.getTime() - second.getTime();
+ };
+
+ // returns Date object
+ DayPilot.DateUtil.fromLocal = function(localDate) {
+ if (!localDate) {
+ localDate = new Date();
+ }
+
+ var d = new Date();
+ d.setUTCFullYear(localDate.getFullYear(), localDate.getMonth(), localDate.getDate());
+ d.setUTCHours(localDate.getHours());
+ d.setUTCMinutes(localDate.getMinutes());
+ d.setUTCSeconds(localDate.getSeconds());
+ d.setUTCMilliseconds(localDate.getMilliseconds());
+ return d;
+ };
+
+ DayPilot.DateUtil.localToday = function() {
+ var d = new Date();
+ d.setHours(0);
+ d.setMinutes(0);
+ d.setSeconds(0);
+ d.setMilliseconds(0);
+ return d;
+ };
+
+ // rename candidate: toHourString
+ DayPilot.DateUtil.hours = function(date, use12) {
+
+ var minute = date.getUTCMinutes();
+ if (minute < 10)
+ minute = "0" + minute;
+
+
+ var hour = date.getUTCHours();
+ //if (hour < 10) hour = "0" + hour;
+
+ if (use12) {
+ var am = hour < 12;
+ var hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ var suffix = am ? "AM" : "PM";
+ return hour + ':' + minute + ' ' + suffix;
+ }
+ else {
+ return hour + ':' + minute;
+ }
+ };
+
+ DayPilot.DateUtil.max = function(first, second) {
+ if (first.getTime() > second.getTime()) {
+ return first;
+ }
+ else {
+ return second;
+ }
+ };
+
+ DayPilot.DateUtil.min = function(first, second) {
+ if (first.getTime() < second.getTime()) {
+ return first;
+ }
+ else {
+ return second;
+ }
+ };
+
+ var Pattern = function(pattern, locale) {
+ if (typeof locale === "string") {
+ locale = DayPilot.Locale.find(locale);
+ }
+ var locale = locale || DayPilot.Locale.US;
+ var all = [
+ {"seq": "yyyy", "expr": "[0-9]{4,4\u007d", "str": function(d) {
+ return d.getYear();
+ }},
+ {"seq": "yy", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ return d.getYear() % 100;
+ }},
+ {"seq": "mm", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getMinutes();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "m", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getMinutes();
+ return r;
+ }},
+ {"seq": "HH", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getHours();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "H", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getHours();
+ return r;
+ }},
+ {"seq": "hh", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var hour = d.getHours();
+ var hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ var r = hour;
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "h", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var hour = d.getHours();
+ var hour = hour % 12;
+ if (hour === 0) {
+ hour = 12;
+ }
+ return hour;
+ }},
+ {"seq": "ss", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getSeconds();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "s", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getSeconds();
+ return r;
+ }},
+ {"seq": "MMMM", "expr": "[^\\s0-9]*", "str": function(d) {
+ var r = locale.monthNames[d.getMonth()];
+ return r;
+ }, "transform" : function(input) {
+ var index = DayPilot.indexOf(locale.monthNames, input, equalsIgnoreCase);
+ if (index < 0) {
+ return null;
+ }
+ return index + 1;
+ }},
+ {"seq": "MMM", "expr": "[^\\s0-9]*", "str": function(d) { // \u0073 = 's'
+ var r = locale.monthNamesShort[d.getMonth()];
+ return r;
+ }, "transform" : function(input) {
+ var index = DayPilot.indexOf(locale.monthNamesShort, input, equalsIgnoreCase);
+ if (index < 0) {
+ return null;
+ }
+ return index + 1;
+ }},
+ {"seq": "MM", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getMonth() + 1;
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "M", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getMonth() + 1;
+ return r;
+ }},
+ {"seq": "dddd", "expr": "[^\\s0-9]*", "str": function(d) {
+ var r = locale.dayNames[d.getDayOfWeek()];
+ return r;
+ }},
+ {"seq": "ddd", "expr": "[^\\s0-9]*", "str": function(d) {
+ var r = locale.dayNamesShort[d.getDayOfWeek()];
+ return r;
+ }},
+ {"seq": "dd", "expr": "[0-9]{2,2\u007d", "str": function(d) {
+ var r = d.getDay();
+ return r < 10 ? "0" + r : r;
+ }},
+ {"seq": "%d", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getDay();
+ return r;
+ }},
+ {"seq": "d", "expr": "[0-9]{1,2\u007d", "str": function(d) {
+ var r = d.getDay();
+ return r;
+ }},
+ {"seq": "tt", "expr": "(AM|PM|am|pm)", "str": function(d) {
+ var hour = d.getHours();
+ var am = hour < 12;
+ return am ? "AM" : "PM";
+ }, "transform" : function(input) {
+ return input.toUpperCase();
+ }},
+ ];
+
+ var escapeRegex = function(text) {
+ return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&");
+ };
+
+ this.init = function() {
+ this.year = this.findSequence("yyyy");
+ this.month = this.findSequence("MMMM") || this.findSequence("MMM") || this.findSequence("MM") || this.findSequence("M");
+ this.day = this.findSequence("dd") || this.findSequence("d");
+
+ this.hours = this.findSequence("HH") || this.findSequence("H");
+ this.minutes = this.findSequence("mm") || this.findSequence("m");
+ this.seconds = this.findSequence("ss") || this.findSequence("s");
+
+ this.ampm = this.findSequence("tt");
+ this.hours12 = this.findSequence("hh") || this.findSequence("h");
+
+ /*if (this.hours && this.ampm) {
+ throw new DayPilot.Exception("'HH' and 'H' specifiers cannot be used in combination with 'tt'. Use 12-hour clock specifiers: 'hh' or 'h'.");
+ }*/
+ };
+
+ this.findSequence = function(seq) {
+
+ function defaultTransform(value) {
+ return parseInt(value);
+ }
+
+ var index = pattern.indexOf(seq);
+ if (index === -1) {
+ return null;
+ }
+ return {
+ "findValue": function(input) {
+ var prepared = escapeRegex(pattern);
+ var transform = null;
+ for (var i = 0; i < all.length; i++) {
+ var len = all[i].length;
+ var pick = (seq === all[i].seq);
+ //var expr = "";
+ var expr = all[i].expr;
+ if (pick) {
+ expr = "(" + expr + ")";
+ transform = all[i].transform;
+ }
+ prepared = prepared.replace(all[i].seq, expr);
+ }
+
+ prepared = "^" + prepared + "$";
+
+ try {
+ var r = new RegExp(prepared);
+ var array = r.exec(input);
+ if (!array) {
+ return null;
+ }
+ transform = transform || defaultTransform; // parseInt is the default transform/parse function
+ return transform(array[1]);
+ }
+ catch (e) {
+ throw "unable to create regex from: " + prepared;
+ }
+ }
+ };
+ };
+
+ this.print = function(date) {
+ // always recompiles the pattern
+
+ var find = function(t) {
+ for (var i = 0; i < all.length; i++) {
+ if (all[i] && all[i].seq === t) {
+ return all[i];
+ }
+ }
+ return null;
+ };
+
+ var eos = pattern.length <= 0;
+ var pos = 0;
+ var components = [];
+
+ while (!eos) {
+ var rem = pattern.substring(pos);
+ var matches = /%?(.)\1*/.exec(rem); // matches a sequence of identical characters, with an optional '%' preceding char
+ if (matches && matches.length > 0) {
+ var match = matches[0];
+ var q = find(match);
+ if (q) {
+ components.push(q);
+ }
+ else {
+ components.push(match);
+ }
+ pos += match.length;
+ eos = pattern.length <= pos;
+ }
+ else {
+ eos = true;
+ }
+ }
+
+ // resolve placeholders
+ for (var i = 0; i < components.length; i++) {
+ var c = components[i];
+ if (typeof c !== 'string') {
+ components[i] = c.str(date);
+ }
+ }
+
+ return components.join("");
+ };
+
+
+
+ this.parse = function(input) {
+
+ var year = this.year.findValue(input);
+ if (!year) {
+ return null; // unparseable
+ }
+
+ var month = this.month.findValue(input);
+ if (DayPilot.Util.isNullOrUndefined(month)) {
+ return null;
+ }
+ if (month > 12 || month < 1) {
+ return null;
+ }
+ var day = this.day.findValue(input);
+
+ var daysInMonth = DayPilot.Date.fromYearMonthDay(year, month).daysInMonth();
+ if (day < 1 || day > daysInMonth) {
+ return null;
+ }
+
+ var hours = this.hours ? this.hours.findValue(input) : 0;
+ var minutes = this.minutes ? this.minutes.findValue(input) : 0;
+ var seconds = this.seconds ? this.seconds.findValue(input) : 0;
+
+ var ampm = this.ampm ? this.ampm.findValue(input): null;
+
+ if (this.ampm && this.hours12) {
+
+ var hours12 = this.hours12.findValue(input);
+
+ if (hours12 < 1 || hours12 > 12) {
+ return null;
+ }
+
+ if (ampm === "PM") {
+ if (hours12 === 12) {
+ hours = 12;
+ }
+ else {
+ hours = hours12 + 12;
+ }
+ }
+ else {
+ if (hours12 === 12) {
+ hours = 0;
+ }
+ else {
+ hours = hours12;
+ }
+ }
+
+ }
+
+ if (hours < 0 || hours > 23) {
+ return null;
+ }
+
+ if (minutes < 0 || minutes > 59) {
+ return null;
+ }
+
+ if (seconds < 0 || seconds > 59) {
+ return null;
+ }
+
+ var d = new Date();
+ d.setUTCFullYear(year, month - 1, day);
+ d.setUTCHours(hours);
+ d.setUTCMinutes(minutes);
+ d.setUTCSeconds(seconds);
+ d.setUTCMilliseconds(0);
+
+ return new DayPilot.Date(d);
+ };
+
+ this.init();
+
+ };
+
+ function equalsIgnoreCase(str1, str2) {
+ if (DayPilot.Util.isNullOrUndefined(str1)) {
+ return false;
+ }
+ if (DayPilot.Util.isNullOrUndefined(str2)) {
+ return false;
+ }
+ return str1.toLocaleLowerCase() === str2.toLocaleLowerCase();
+ }
+
+ // DayPilot.Date END
+
+
+ DayPilot.ColorUtil = {};
+
+ function toHex(dec) {
+ dec = Math.min(dec, 255);
+ dec = Math.max(dec, 0);
+ var str = dec.toString(16);
+ return (dec < 16) ? "0" + str : str;
+ }
+
+ DayPilot.ColorUtil.hexToRgb = function(hex) {
+ if (!/^#[0-9a-f]{6}$/i.test(hex)) {
+ throw new DayPilot.Exception("Invalid color, only full hex color string accepted, eg. '#ffaaff'.");
+ }
+ hex = hex.replace("#", "");
+ return {
+ r: parseInt(hex.substring(0, 2), 16),
+ g: parseInt(hex.substring(2, 4), 16),
+ b: parseInt(hex.substring(4, 6), 16),
+ };
+ };
+
+ DayPilot.ColorUtil.rgbToHex = function(rgb) {
+ return "#" + toHex(rgb.r) + toHex(rgb.g) + toHex(rgb.b);
+ };
+
+ // pt in 255 base
+ DayPilot.ColorUtil.adjustLuminance = function(rgb, pt) {
+ return {
+ r: rgb.r + pt,
+ g: rgb.g + pt,
+ b: rgb.b + pt
+ };
+ };
+
+ DayPilot.ColorUtil.darker = function(hexColor, steps) {
+ var src = DayPilot.ColorUtil.hexToRgb(hexColor);
+ steps = steps || 1;
+ var step = 17; // (0xbb - 0xaa = 17)
+ var pt = steps * step;
+ var target = DayPilot.ColorUtil.adjustLuminance(src, -pt);
+ return DayPilot.ColorUtil.rgbToHex(target);
+ };
+
+ DayPilot.ColorUtil.lighter = function(hexColor, steps) {
+ if (typeof steps !== "number") {
+ steps = 1;
+ }
+ return DayPilot.ColorUtil.darker(hexColor, -steps);
+ };
+
+
+ DayPilot.ColorUtil.pl = function(hexColor) {
+ var rgb = DayPilot.ColorUtil.hexToRgb(hexColor);
+ var r = rgb.r / 255;
+ var g = rgb.g / 255;
+ var b = rgb.b / 255;
+ var pl = Math.sqrt(0.299*r*r + 0.587*g*g + 0.114*b*b);
+ return pl;
+ };
+
+ DayPilot.ColorUtil.contrasting = function(hexColor, light, dark) {
+ var pl = DayPilot.ColorUtil.pl(hexColor);
+ light = light || "#ffffff";
+ dark = dark || "#000000";
+ return pl > 0.5 ? dark : light;
+ };
+
+
+ DayPilot.Event = function(data, calendar, part) {
+ var e = this;
+ this.calendar = calendar;
+ this.data = data ? data : {};
+ this.part = part ? part : {};
+
+ // backwards compatibility, still accepts id in "value"
+ if (typeof this.data.id === 'undefined') {
+ this.data.id = this.data.value;
+ }
+
+ var copy = {};
+ var synced = ["id", "text", "start", "end", "resource"];
+
+ this.isEvent = true;
+
+ // internal
+ this.temp = function() {
+ if (copy.dirty) {
+ return copy;
+ }
+ for (var i = 0; i < synced.length; i++) {
+ copy[synced[i]] = e.data[synced[i]];
+ }
+ copy.dirty = true;
+ return copy;
+
+ };
+
+ // internal
+ this.copy = function() {
+ var result = {};
+ for (var i = 0; i < synced.length; i++) {
+ result[synced[i]] = e.data[synced[i]];
+ }
+ return result;
+ };
+
+ this.commit = function() {
+ if (!copy.dirty) {
+ return;
+ }
+
+ for (var i = 0; i < synced.length; i++) {
+ e.data[synced[i]] = copy[synced[i]];
+ }
+
+ copy.dirty = false;
+ };
+
+ this.dirty = function() {
+ return copy.dirty;
+ };
+
+ this.id = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.id;
+ }
+ else {
+ this.temp().id = val;
+ }
+ };
+ // obsolete, use id() instead
+ this.value = function(val) {
+ if (typeof val === 'undefined') {
+ return e.id();
+ }
+ else {
+ e.id(val);
+ }
+ };
+ this.text = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.text;
+ }
+ else {
+ this.temp().text = val;
+ this.client.innerHTML(val); // update the HTML automatically
+ }
+ };
+ this.start = function(val) {
+ if (typeof val === 'undefined') {
+ return new DayPilot.Date(e.data.start);
+ }
+ else {
+ this.temp().start = new DayPilot.Date(val);
+ }
+ };
+ this.end = function(val) {
+ if (typeof val === 'undefined') {
+ return new DayPilot.Date(e.data.end);
+ }
+ else {
+ this.temp().end = new DayPilot.Date(val);
+ }
+ };
+ this.resource = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.resource;
+ }
+ else {
+ this.temp().resource = val;
+ }
+ };
+ this.partStart = function() {
+ return new DayPilot.Date(this.part.start);
+ };
+ this.partEnd = function() {
+ return new DayPilot.Date(this.part.end);
+ };
+
+ this.tag = function(field) {
+ var values = e.data.tag;
+ if (!values) {
+ return null;
+ }
+ if (typeof field === 'undefined') {
+ return e.data.tag;
+ }
+ var fields = e.calendar.tagFields;
+ var index = -1;
+ for (var i = 0; i < fields.length; i++) {
+ if (field === fields[i])
+ index = i;
+ }
+ if (index === -1) {
+ throw "Field name not found.";
+ }
+ //var tags = t.split('&');
+ return values[index];
+ };
+
+ this.client = {};
+ this.client.innerHTML = function(val) {
+ if (typeof val === 'undefined') {
+ var data = e.cache || e.data;
+ var xssTextHtml = e.calendar && e.calendar.internal && e.calendar.internal.xssTextHtml;
+ if (xssTextHtml) {
+ return xssTextHtml(data.text, data.html);
+ }
+ return DayPilot.Util.escapeTextHtml(data.text, data.html);
+
+ /*
+ if (e.cache && typeof e.cache.html !== "undefined") {
+ return e.cache.html;
+ }
+ if (typeof e.data.html !== "undefined") {
+ return e.data.html;
+ }
+ return e.data.text;
+ */
+ }
+ else {
+ e.data.html = val;
+ }
+ };
+
+ this.client.html = this.client.innerHTML;
+
+ this.client.header = function(val) {
+ if (typeof val === 'undefined') {
+ return e.data.header;
+ }
+ else {
+ e.data.header = val;
+ }
+ };
+
+ this.client.cssClass = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.cssClass !== "undefined") {
+ return e.cache.cssClass;
+ }
+ return e.data.cssClass;
+ }
+ else {
+ e.data.cssClass = val;
+ }
+ };
+ this.client.toolTip = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.toolTip !== "undefined") {
+ return e.cache.toolTip;
+ }
+ return typeof e.data.toolTip !== 'undefined' ? e.data.toolTip : e.data.text;
+ }
+ else {
+ e.data.toolTip = val;
+ }
+ };
+
+ this.client.barVisible = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.barHidden !== "undefined") {
+ return !e.cache.barHidden;
+ }
+ return e.calendar.durationBarVisible && !e.data.barHidden;
+ }
+ else {
+ e.data.barHidden = !val;
+ }
+ };
+
+ this.client.backColor = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.backColor !== "undefined") {
+ return e.cache.backColor;
+ }
+ return typeof e.data.backColor !== "undefined" ? e.data.backColor : e.calendar.eventBackColor;
+ }
+ else {
+ e.data.backColor = val;
+ }
+ };
+
+ this.client.borderColor = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.borderColor !== "undefined") {
+ return e.cache.borderColor;
+ }
+ return typeof e.data.borderColor !== "undefined" ? e.data.borderColor : e.calendar.eventBorderColor;
+ }
+ else {
+ e.data.borderColor = val;
+ }
+ };
+
+ this.client.contextMenu = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.oContextMenu) {
+ return e.oContextMenu;
+ }
+ var cm = e.cache ? e.cache.contextMenu : e.data.contextMenu;
+ }
+ else {
+ e.oContextMenu = val;
+ }
+ };
+
+ this.client.moveEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.moveDisabled !== "undefined") {
+ return !e.cache.moveDisabled;
+ }
+
+ return e.calendar.eventMoveHandling !== 'Disabled' && !e.data.moveDisabled;
+ }
+ else {
+ e.data.moveDisabled = !val;
+ }
+ };
+
+ this.client.resizeEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.resizeDisabled !== "undefined") {
+ return !e.cache.resizeDisabled;
+ }
+ return e.calendar.eventResizeHandling !== 'Disabled' && !e.data.resizeDisabled;
+ }
+ else {
+ e.data.resizeDisabled = !val;
+ }
+ };
+
+ this.client.clickEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.clickDisabled !== "undefined") {
+ return !e.cache.clickDisabled;
+ }
+ return e.calendar.eventClickHandling !== 'Disabled' && !e.data.clickDisabled;
+ }
+ else {
+ e.data.clickDisabled = !val;
+ }
+ };
+
+ this.client.rightClickEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.rightClickDisabled !== "undefined") {
+ return !e.cache.rightClickDisabled;
+ }
+ return e.calendar.eventRightClickHandling !== 'Disabled' && !e.data.rightClickDisabled;
+ }
+ else {
+ e.data.rightClickDisabled = !val;
+ }
+ };
+
+ this.client.deleteEnabled = function(val) {
+ if (typeof val === 'undefined') {
+ if (e.cache && typeof e.cache.deleteDisabled !== "undefined") {
+ return !e.cache.deleteDisabled;
+ }
+ return e.calendar.eventDeleteHandling !== 'Disabled' && !e.data.deleteDisabled;
+ }
+ else {
+ e.data.deleteDisabled = !val;
+ }
+ };
+
+ this.toJSON = function(key) {
+ var json = {};
+ json.value = this.id(); // still sending it with the old name
+ json.id = this.id();
+ json.text = this.text();
+ json.start = this.start();
+ json.end = this.end();
+ json.tag = {};
+
+ if (e.calendar && e.calendar.tagFields) {
+ var fields = e.calendar.tagFields;
+ for (var i = 0; i < fields.length; i++) {
+ json.tag[fields[i]] = this.tag(fields[i]);
+ }
+ }
+
+ return json;
+ };
+ };
+
+})();
+
+/* JSON */
+// thanks to http://www.json.org/js.html
+
+
+// declares DayPilot.JSON.stringify()
+DayPilot.JSON = {};
+
+(function () {
+ function f(n) {
+ return n < 10 ? '0' + n : n;
+ }
+
+ if (typeof Date.prototype.toJSON2 !== 'function') {
+
+ Date.prototype.toJSON2 = function (key) {
+ return this.getUTCFullYear() + '-' +
+ f(this.getUTCMonth() + 1) + '-' +
+ f(this.getUTCDate()) + 'T' +
+ f(this.getUTCHours()) + ':' +
+ f(this.getUTCMinutes()) + ':' +
+ f(this.getUTCSeconds()) + '';
+ };
+
+ String.prototype.toJSON =
+ Number.prototype.toJSON =
+ Boolean.prototype.toJSON = function (key) {
+ return this.valueOf();
+ };
+ }
+
+ var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
+ escapeable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
+ gap,
+ indent,
+ meta = {
+ '\b': '\\b',
+ '\t': '\\t',
+ '\n': '\\n',
+ '\f': '\\f',
+ '\r': '\\r',
+ '"' : '\\"',
+ '\\': '\\\\'
+ },
+ rep;
+
+ function quote(string) {
+ escapeable.lastIndex = 0;
+ return escapeable.test(string) ?
+ '"' + string.replace(escapeable, function (a) {
+ var c = meta[a];
+ if (typeof c === 'string') {
+ return c;
+ }
+ return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
+ }) + '"' :
+ '"' + string + '"';
+ }
+
+ function str(key, holder) {
+ var i,
+ k,
+ v,
+ length,
+ mind = gap,
+ partial,
+ value = holder[key];
+ if (value && typeof value === 'object' && typeof value.toJSON2 === 'function') {
+ value = value.toJSON2(key);
+ }
+ else if (value && typeof value === 'object' && typeof value.toJSON === 'function' && !value.ignoreToJSON) {
+ value = value.toJSON(key);
+ }
+ if (typeof rep === 'function') {
+ value = rep.call(holder, key, value);
+ }
+ switch (typeof value) {
+ case 'string':
+ return quote(value);
+ case 'number':
+ return isFinite(value) ? String(value) : 'null';
+ case 'boolean':
+ case 'null':
+ return String(value);
+ case 'object':
+ if (!value) {
+ return 'null';
+ }
+ gap += indent;
+ partial = [];
+ if (typeof value.length === 'number' &&
+ !value.propertyIsEnumerable('length')) {
+ length = value.length;
+ for (i = 0; i < length; i += 1) {
+ partial[i] = str(i, value) || 'null';
+ }
+ v = partial.length === 0 ? '[]' :
+ gap ? '[\n' + gap +
+ partial.join(',\n' + gap) + '\n' +
+ mind + ']' :
+ '[' + partial.join(',') + ']';
+ gap = mind;
+ return v;
+ }
+ if (rep && typeof rep === 'object') {
+ length = rep.length;
+ for (i = 0; i < length; i += 1) {
+ k = rep[i];
+ if (typeof k === 'string') {
+ v = str(k, value);
+ if (v) {
+ partial.push(quote(k) + (gap ? ': ' : ':') + v);
+ }
+ }
+ }
+ } else {
+ for (k in value) {
+ if (Object.hasOwnProperty.call(value, k)) {
+ v = str(k, value);
+ if (v) {
+ partial.push(quote(k) + (gap ? ': ' : ':') + v);
+ }
+ }
+ }
+ }
+ v = (partial.length === 0) ? '{\u007D' :
+ gap ? '{\n' + gap + partial.join(',\n' + gap) + '\n' +
+ mind + '\u007D' : '{' + partial.join(',') + '\u007D';
+ gap = mind;
+ return v;
+ }
+ }
+
+ if (typeof DayPilot.JSON.stringify !== 'function') {
+ DayPilot.JSON.stringify = function (value, replacer, space) {
+ var i;
+ gap = '';
+ indent = '';
+ if (typeof space === 'number') {
+ for (i = 0; i < space; i += 1) {
+ indent += ' ';
+ }
+ } else if (typeof space === 'string') {
+ indent = space;
+ }
+ rep = replacer;
+ if (replacer && typeof replacer !== 'function' && (typeof replacer !== 'object' || typeof replacer.length !== 'number')) {
+ throw new Error('JSON.stringify');
+ }
+ return str('', {'': value});
+ };
+ }
+
+})();
diff --git a/static/src/js/src/daypilot-datepicker.src.js b/static/src/js/src/daypilot-datepicker.src.js
new file mode 100644
index 0000000..941009a
--- /dev/null
+++ b/static/src/js/src/daypilot-datepicker.src.js
@@ -0,0 +1,284 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+(function() {
+
+ if (typeof DayPilot.DatePicker !== 'undefined' && DayPilot.DatePicker.close) {
+ return;
+ }
+
+ DayPilot.DatePicker = function(properties) {
+ this.v = '2024.3.539-lite';
+ var navigatorId = "navigator_" + new Date().getTime();
+ var This = this;
+
+ this.onShow = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+ // this.dateFormat = null;
+
+ this.prepare = function() {
+ this.locale = "en-us";
+ this.target = null;
+ this.targetAlignment = "left";
+ this.resetTarget = true;
+ this.pattern = this._resolved.locale().datePattern; // "M/d/yyyy"
+ this.theme = "navigator_default";
+ this.patterns = [];
+ this.zIndex = null;
+
+ // load settings
+ if (properties) {
+ for (var name in properties) {
+ this[name] = properties[name];
+ }
+ }
+
+ };
+
+ this.init = function() {
+
+ this.date = new DayPilot.Date(this.date);
+
+ var value = this._readFromTarget();
+
+ if (this.resetTarget && !value) {
+ this._writeToTarget(this.date);
+ }
+ else if (!this.resetTarget) {
+ This.date = value;
+ }
+
+ var target = this._element();
+ if (target) {
+ target.addEventListener("input", function() {
+ This.date = This._readFromTarget();
+ if (This.date) {
+ This.navigator.select(This.date, {dontNotify: true});
+ }
+ });
+ }
+
+ document.addEventListener("mousedown", function() {
+ This.close();
+ });
+
+ return this;
+
+ };
+
+ this.close = function() {
+ if (!this._visible) {
+ return;
+ }
+
+ this._visible = false;
+
+ if (this.navigator) {
+ this.navigator.dispose();
+ }
+ this.div.innerHTML = '';
+ if (this.div && this.div.parentNode === document.body) {
+ document.body.removeChild(this.div);
+ }
+ };
+
+ this.setDate = function(date) {
+ this.date = new DayPilot.Date(date);
+ this._writeToTarget(this.date);
+ };
+
+ this._readFromTarget = function() {
+ // recognized targets: input (value), other DOM elements (innerHTML)
+ var element = this._element();
+
+ if (!element) {
+ return this.date;
+ }
+
+ var value = null;
+ if (element.tagName === "INPUT") {
+ value = element.value;
+ }
+ else {
+ value = element.innerText;
+ }
+
+ if (!value) {
+ return null;
+ }
+
+ var date = DayPilot.Date.parse(value, This.pattern);
+ for (var i = 0; i < This.patterns.length; i++) {
+ if (date) {
+ return date;
+ }
+ date = DayPilot.Date.parse(value, This.patterns[i]);
+ }
+
+ return date;
+ };
+
+ this._writeToTarget = function(date) {
+ var element = this._element();
+
+ if (!element) {
+ return;
+ }
+
+ var value = date.toString(This.pattern, This.locale);
+ if (element.tagName === "INPUT") {
+ element.value = value;
+ }
+ else {
+ element.innerHTML = value;
+ }
+
+ };
+
+ this._resolved = {};
+ this._resolved.locale = function() {
+ return DayPilot.Locale.find(This.locale);
+ };
+
+ /* this._resolved.dateFormat = function() {
+ if (typeof This.dateFormat === "string") {
+ return This.dateFormat;
+ }
+ return This._resolved.locale().datePattern;
+ }*/
+
+ this._element = function() {
+ var id = this.target;
+ // accept DOM element or id (string)
+ var element = (id && id.nodeType && id.nodeType === 1 ) ? id : document.getElementById(id);
+ return element;
+ };
+
+ Object.defineProperty(this, "visible", {
+ get: function() { return This._visible; }
+ });
+
+ this.show = function() {
+
+ if (this._visible) {
+ return;
+ }
+
+ var element = this._element();
+ var navigator = this.navigator;
+
+ var navigator = new DayPilot.Navigator(navigatorId);
+ navigator.api = 2;
+ navigator.cssOnly = true;
+ navigator.theme = This.theme;
+ navigator.weekStarts = "Auto";
+ navigator.locale = This.locale;
+ navigator.onTimeRangeSelected = function(args) {
+ This.date = args.start;
+
+ var start = args.start.addTime(navigator._pickerTimePart);
+ var value = start.toString(This.pattern, This.locale);
+
+ var args = {};
+ args.start = start;
+ args.date = start;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof This.onTimeRangeSelect === 'function') {
+ This.onTimeRangeSelect(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ This._writeToTarget(value);
+ This.close();
+
+ if (typeof This.onTimeRangeSelected === 'function') {
+ This.onTimeRangeSelected(args);
+ }
+ };
+
+ this.navigator = navigator;
+
+ var position = DayPilot.abs(element);
+ var height = element.offsetHeight;
+
+ var align = This.targetAlignment;
+
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+
+ if (align === "left") {
+ div.style.left = position.x + "px";
+ }
+
+
+ div.style.top = (position.y + height) + "px";
+ if (This.zIndex) {
+ div.style.zIndex = This.zIndex;
+ }
+
+ var nav = document.createElement("div");
+ nav.id = navigatorId;
+ div.appendChild(nav);
+
+ div.addEventListener("mousedown", function(ev) {
+ var ev = ev || window.event;
+ ev.cancelBubble = true;
+ ev.stopPropagation && ev.stopPropagation();
+ });
+
+ document.body.appendChild(div);
+
+ this.div = div;
+
+ var selected = This._readFromTarget() || new DayPilot.Date().getDatePart();
+
+ navigator.startDate = selected;
+ navigator._pickerTimePart = selected.getTimePart();
+ // navigator.selectionStart = selected.getDatePart();
+ navigator.selectionDay = selected.getDatePart();
+ navigator.init();
+
+ if (align === "right") {
+ var left = (position.x + element.offsetWidth - navigator.nav.top.offsetWidth);
+ div.style.left = left + "px";
+ }
+
+ this._visible = true;
+ if (this.onShow) {
+ this.onShow();
+ }
+ };
+
+ this.prepare(); // prepare only called once, in the constructor
+ this.init();
+ };
+
+})();
diff --git a/static/src/js/src/daypilot-menu.src.js b/static/src/js/src/daypilot-menu.src.js
new file mode 100644
index 0000000..e46d93c
--- /dev/null
+++ b/static/src/js/src/daypilot-menu.src.js
@@ -0,0 +1,974 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function(DayPilot) {
+ 'use strict';
+
+ if (typeof DayPilot.Menu !== 'undefined' && DayPilot.Menu.def) {
+ return;
+ }
+
+ var doNothing = function() {};
+
+ var DayPilotMenu = {};
+
+ DayPilotMenu.mouse = null;
+ DayPilotMenu.menu = null;
+ DayPilotMenu.handlersRegistered = false;
+ DayPilotMenu.hideTimeout = null;
+ DayPilotMenu.waitingSubmenu = null;
+
+ DayPilot.Menu = function(items) {
+ var menu = this;
+ var initiatorAreaDiv = null;
+
+ this.v = '2024.3.539-lite';
+ this.zIndex = 120; // more than 10,001 used by ModalPopupExtender
+ //this.useShadow = true;
+ this.cssClassPrefix = "menu_default";
+ this.cssOnly = true;
+ this.menuTitle = null;
+ this.showMenuTitle = false;
+ this.hideOnMouseOut = false;
+ this.hideAfter = 200;
+ this.theme = null;
+
+ this.onShow = null;
+
+ // hiding internal properties for angular
+ this._state = function() {};
+ //this._state.ref = null; // ref object, used for position
+
+ if (items && DayPilot.isArray(items)) {
+ this.items = items;
+ }
+
+ // angular change detection
+ this.toJSON = function() {
+ return null;
+ };
+
+ this.show = function(e, options) {
+ options = options || {};
+
+ var value = null;
+ if (!e) {
+ value = null;
+ }
+ else if (typeof e.id === 'string' || typeof e.id === 'number') {
+ value = e.id;
+ }
+ else if (typeof e.id === 'function') {
+ value = e.id();
+ }
+ else if (typeof e.value === 'function') {
+ value = e.value();
+ }
+
+ if (typeof(DayPilot.Bubble) !== 'undefined') { // hide any bubble if active
+ DayPilot.Bubble.hideActive();
+ }
+
+ if (!options.submenu) {
+ DayPilotMenu.menuClean();
+ }
+
+ // clear old data
+ this._state.submenu = null;
+
+ if (DayPilotMenu.mouse === null) { // not possible to execute before mouse move (TODO)
+ return;
+ }
+
+ if (!menu.cssOnly) {
+ menu.cssOnly = true;
+ }
+
+ var source = null;
+ if (e && e.isRow && e.$.row.task) {
+ source = new DayPilot.Task(e.$.row.task, e.calendar);
+ source.menuType = "resource";
+ }
+ else if (e && e.isEvent && e.data.task) {
+ source = new DayPilot.Task(e, e.calendar);
+ }
+ else {
+ source = e;
+ }
+
+ if (typeof menu.onShow === "function") {
+ var args = {};
+ args.source = source;
+ args.menu = menu;
+ args.preventDefault = function () {
+ args.preventDefault.value = true;
+ };
+ menu.onShow(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+ div.style.top = "0px";
+ div.style.left = "0px";
+ div.style.display = 'none';
+ div.style.overflow = 'hidden';
+ div.style.zIndex = this.zIndex + 1;
+ div.className = this._applyCssClass('main');
+ div.onclick = function(ev) {
+ ev.cancelBubble = true;
+ this.parentNode.removeChild(this);
+ };
+
+ if (this.hideOnMouseOut) {
+ div.onmousemove = function(ev) {
+ clearTimeout(DayPilotMenu.hideTimeout);
+ };
+ div.onmouseleave = function(ev) {
+ menu.delayedHide({"hideParent": true});
+ };
+ }
+
+ if (!this.items || this.items.length === 0) {
+ throw "No menu items defined.";
+ }
+
+ if (this.showMenuTitle) {
+ var title = document.createElement("div");
+ title.innerHTML = this.menuTitle;
+ title.className = this._applyCssClass("title");
+ div.appendChild(title);
+ }
+
+ for (var i = 0; i < this.items.length; i++) {
+ var mi = this.items[i];
+ var item = document.createElement("div");
+ DayPilot.Util.addClass(item, this._applyCssClass("item"));
+ if (mi.items) {
+ DayPilot.Util.addClass(item, this._applyCssClass("item_haschildren"));
+ DayPilot.Util.addClass(div, this._applyCssClass(("withchildren")));
+ }
+
+ if (typeof mi === 'undefined') {
+ continue;
+ }
+
+ if (mi.hidden) {
+ continue;
+ }
+
+
+ if (mi.text === '-') {
+ var separator = document.createElement("div");
+ separator.addEventListener("click", function(ev) {
+ ev.stopPropagation();
+ });
+
+ item.appendChild(separator);
+ }
+ else {
+ var link = document.createElement("a");
+ link.style.position = 'relative';
+ link.style.display = "block";
+
+ if (mi.cssClass) {
+ DayPilot.Util.addClass(link, mi.cssClass);
+ }
+
+ if (mi.disabled) {
+ DayPilot.Util.addClass(link, menu._applyCssClass("item_disabled"));
+ }
+ else {
+ if (mi.onclick || mi.onClick) {
+ link.item = mi;
+ link.onclick = (function(mi, link) {
+ return function (e) {
+ if (typeof mi.onClick === "function") {
+ var args = {};
+ args.item = mi;
+ args.source = link.source;
+ args.originalEvent = e;
+ args.preventDefault = function () {
+ args.preventDefault.value = true;
+ };
+ mi.onClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+ if (mi.onclick) {
+ mi.onclick.call(link, e);
+ }
+ };
+ })(mi, link);
+
+ var assignTouchEnd = function(mi, link) {
+ return function(e) {
+ e.stopPropagation();
+ e.preventDefault();
+
+ var cleanup = function() {
+ window.setTimeout(function() {
+ link.source.calendar.internal.touch.active = false;
+ }, 500);
+ };
+
+ if (typeof mi.onClick === "function") {
+ var args = {};
+ args.item = mi;
+ args.source = link.source;
+ args.originalEvent = e;
+ args.preventDefault = function() {
+ args.preventDefault.value = true;
+ };
+ mi.onClick(args);
+ if (args.preventDefault.value) {
+ cleanup();
+ return;
+ }
+ }
+
+ if (mi.onclick) {
+ mi.onclick.call(link, e);
+ }
+
+ DayPilotMenu.menuClean();
+ cleanup();
+ };
+ };
+
+ DayPilot.reNonPassive(link, "touchstart", function(ev) {
+ ev.stopPropagation();
+ ev.preventDefault();
+
+ link.source.calendar.internal.touch.active = true;
+ });
+ DayPilot.reNonPassive(link, "touchend", assignTouchEnd(mi, link));
+
+ // link.ontouchend = assignTouchEnd(mi, link);
+ }
+
+
+ if (mi.items && !mi.disabled) {
+ var assign = function(mi, link) {
+ return function(ev) {
+ ev.preventDefault();
+ ev.stopPropagation();
+ menu._showSubmenu(mi, link);
+ };
+ };
+ link.ontouchend = assign(mi, link);
+ }
+
+ if (mi.onclick) {
+ doNothing();
+ }
+ else if (mi.href) {
+ link.href = mi.href.replace(/\x7B0\x7D/gim, value); // for NavigateUrl actions, only for backwards compatibility
+ if (mi.target) {
+ link.setAttribute("target", mi.target);
+ }
+ }
+ else if (mi.command) {
+ var assign = function(mi, link) {
+ return function(e) {
+ var source = link.source;
+ var item = mi;
+ item.action = item.action ? item.action : 'CallBack';
+ var cal = source.calendar || source.root;
+
+ if (source instanceof DayPilot.Link) {
+ cal.internal.linkMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Selection) {
+ cal.internal.timeRangeMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Event) {
+ cal.internal.eventMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Selection) {
+ cal.internal.timeRangeMenuClick(item.command, source, item.action);
+ return;
+ }
+ else if (source instanceof DayPilot.Task) {
+ if (source.menuType === "resource") {
+ cal.internal.resourceHeaderMenuClick(item.command, link.menuSource, item.action);
+ }
+ else {
+ cal.internal.eventMenuClick(item.command, link.menuSource, item.action);
+ }
+ return;
+ }
+ else {
+ switch (source.menuType) { // TODO legacy, remove
+ case 'resource':
+ cal.internal.resourceHeaderMenuClick(item.command, source, item.action);
+ return;
+ case 'selection': // fully replaced
+ cal.internal.timeRangeMenuClick(item.command, source, item.action);
+ return;
+ default: // fully replaced
+ cal.internal.eventMenuClick(item.command, source, item.action);
+ return;
+ }
+ }
+
+ e.preventDefault();
+ };
+ };
+ link.onclick = assign(mi, link);
+ link.ontouchend = assign(mi, link);
+ }
+
+ }
+
+
+ if (mi.items) {
+ link.addEventListener("click", function(ev) {
+ ev.stopPropagation();
+ });
+ }
+
+ link.source = source;
+ link.menuSource = e;
+
+ var span = document.createElement("span");
+ span.className = menu._applyCssClass("item_text");
+ span.innerHTML = DayPilot.Util.escapeTextHtml(mi.text, mi.html);
+ link.appendChild(span);
+
+ if (mi.image) {
+ var image = document.createElement("img");
+ image.src = mi.image;
+ image.style.position = 'absolute';
+ image.style.top = '0px';
+ image.style.left = '0px';
+
+ link.appendChild(image);
+ }
+
+ if (mi.icon) {
+ var icon = document.createElement("span");
+ icon.className = menu._applyCssClass("item_icon");
+
+ var iel = document.createElement("i");
+ iel.className = mi.icon;
+ icon.appendChild(iel);
+
+ link.appendChild(icon);
+ }
+
+ if (mi.symbol) {
+ var ns = "http://www.w3.org/2000/svg";
+ var svg = document.createElementNS(ns,"svg");
+ svg.setAttribute("width", "100%");
+ svg.setAttribute("height", "100%");
+ var use = document.createElementNS(ns,"use");
+ use.setAttribute("href", mi.symbol);
+ svg.appendChild(use);
+
+ var svgWrap = document.createElement("span");
+ svgWrap.className = menu._applyCssClass("item_symbol");
+ svgWrap.style.position = "absolute";
+ svgWrap.style.top = "0px";
+ svgWrap.style.left = "0px";
+ svgWrap.appendChild(svg);
+
+ link.appendChild(svgWrap);
+ }
+
+ var assignOnMouseOver = function(mi, link) {
+ return function() {
+ var source = link.source;
+ var item = mi;
+
+ var ws = DayPilotMenu.waitingSubmenu;
+ if (ws) {
+ if (ws.parent === item) {
+ return;
+ }
+ else {
+ clearTimeout(ws.timeout);
+ DayPilotMenu.waitingSubmenu = null;
+ }
+ }
+
+ if (mi.disabled) {
+ return;
+ }
+
+ DayPilotMenu.waitingSubmenu = {};
+ DayPilotMenu.waitingSubmenu.parent = item;
+ DayPilotMenu.waitingSubmenu.timeout = setTimeout(function() {
+
+ DayPilotMenu.waitingSubmenu = null;
+
+ menu._showSubmenu(item, link);
+
+ }, 300);
+ };
+ };
+
+ link.onmouseover = assignOnMouseOver(mi, link);
+
+ item.appendChild(link);
+ }
+
+ div.appendChild(item);
+
+ }
+
+ var delayedDismiss = function(e) {
+ window.setTimeout(function() {
+ DayPilotMenu.menuClean();
+ DayPilot.MenuBar.deactivate();
+ }, 100);
+ };
+
+ div.onclick = delayedDismiss;
+ div.ontouchend = delayedDismiss;
+
+ div.onmousedown = function(e) {
+ e = e || window.event;
+ e.cancelBubble = true;
+ if (e.stopPropagation)
+ e.stopPropagation();
+ };
+ div.oncontextmenu = function() {
+ return false;
+ };
+
+ document.body.appendChild(div);
+ menu._state.visible = true;
+ menu._state.source = e;
+
+ div.style.display = '';
+ var height = div.offsetHeight;
+ var width = div.offsetWidth;
+ div.style.display = 'none';
+
+ // don't show the menu outside of the visible window
+ var windowHeight = document.documentElement.clientHeight;
+ // required for ipad with zoom instead of document.documentElement.clientWidth
+ var windowWidth = window.innerWidth;
+
+ var windowMargin = (typeof options.windowMargin == "number") ? options.windowMargin : 5;
+
+ (function showInitiator() {
+ var initiator = options.initiator;
+ // initiator = options.initiator;
+ if (!initiator) {
+ return;
+ }
+ var div = initiator.div;
+ var e = initiator.e;
+ var area = initiator.area;
+
+ var v = area.visibility || area.v || "Visible";
+ var a = initiator.a;
+ if (v !== "Visible") {
+ // make sure the source area is visible
+ a = DayPilot.Areas.createArea(div, e, area);
+ div.appendChild(a);
+ // will be used to remove it on hide
+ initiatorAreaDiv = a;
+ }
+
+ if (a) {
+ var abs = DayPilot.abs(a);
+ options.x = abs.x;
+ options.y = abs.y + abs.h + 2;
+ }
+
+ })();
+
+
+ (function adjustPosition() {
+
+ // don't show it exactly under the cursor
+ var x = (typeof options.x === "number") ? options.x : DayPilotMenu.mouse.x + 1;
+ var y = (typeof options.y === "number") ? options.y : DayPilotMenu.mouse.y + 1;
+
+ var topOffset = document.body.scrollTop || document.documentElement.scrollTop;
+ var leftOffset = document.body.scrollLeft || document.documentElement.scrollLeft;
+
+ if (y - topOffset > windowHeight - height && windowHeight !== 0) {
+ var offsetY = y - topOffset - (windowHeight - height) + windowMargin;
+ div.style.top = (y - offsetY) + 'px';
+ }
+ else {
+ div.style.top = y + 'px';
+ }
+
+ if (options.align === "right") {
+ x -= width;
+ }
+
+ if (x - leftOffset > windowWidth - width && windowWidth !== 0) {
+ var offsetX = x - leftOffset - (windowWidth - width) + windowMargin;
+ div.style.left = (x - offsetX) + 'px';
+ }
+ else {
+ div.style.left = x + 'px';
+ }
+/*
+ if (DayPilotMenu.mouse.clientY > windowHeight - height && windowHeight !== 0) {
+ var offsetY = DayPilotMenu.mouse.clientY - (windowHeight - height) + 5;
+ div.style.top = (y - offsetY) + 'px';
+ }
+ else {
+ div.style.top = y + 'px';
+ }
+
+ if (DayPilotMenu.mouse.clientX > windowWidth - width && windowWidth !== 0) {
+ var offsetX = DayPilotMenu.mouse.clientX - (windowWidth - width) + 5;
+ div.style.left = (x - offsetX) + 'px';
+ }
+ else {
+ div.style.left = x + 'px';
+ }
+*/
+ })();
+
+ if (options.parentLink) {
+
+ var parent = options.parentLink;
+
+ var verticalOffset = parseInt(new DayPilot.StyleReader(div).get("border-top-width"));
+
+ var pos = DayPilot.abs(options.parentLink.parentNode);
+ var x = pos.x + parent.offsetWidth;
+ var y = pos.y - verticalOffset;
+
+ if (x + width > windowWidth) {
+ x = Math.max(0, pos.x - width);
+ }
+
+ var docScrollTop = document.body.scrollTop + document.documentElement.scrollTop;
+ if (y + height - docScrollTop > windowHeight) {
+ y = Math.max(0, windowHeight - height + docScrollTop);
+ }
+
+ div.style.left = x + "px";
+ div.style.top = y + "px";
+
+ }
+ div.style.display = '';
+
+ this.addShadow(div);
+ this._state.div = div;
+
+ if (!options.submenu) {
+ DayPilot.Menu.active = this;
+ }
+
+ //this._initiator = null;
+
+ };
+
+ this._showSubmenu = function(item, link) {
+ var mi = item;
+ var source = link.source;
+
+ if (menu._state.submenu && menu._state.submenu.item === item) { // already visible
+ return;
+ }
+
+ if (menu._state.submenu && menu._state.submenu.item !== item) { // hide submenus of other items
+ DayPilot.Util.removeClass(menu._state.submenu.link.parentNode, menu._applyCssClass("item_haschildren_active"));
+ menu._state.submenu.menu.hide();
+ menu._state.submenu = null;
+ }
+
+ if (!item.items) { // no submenu for this item
+ return;
+ }
+
+ var options = menu.cloneOptions();
+ options.items = item.items;
+
+ menu._state.submenu = {};
+ menu._state.submenu.menu = new DayPilot.Menu(options);
+ menu._state.submenu.menu._parentMenu = menu;
+ menu._state.submenu.menu.show(source, {"submenu": true, "parentLink": link, "parentItem": mi});
+ menu._state.submenu.item = item;
+ menu._state.submenu.link = link;
+ DayPilot.Util.addClass(link.parentNode, menu._applyCssClass("item_haschildren_active"));
+
+ };
+
+ this._applyCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ var sep = (this.cssOnly ? "_" : "");
+ if (prefix) {
+ return prefix + sep + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this.cloneOptions = function() {
+ return DayPilot.Util.copyProps(options, {}, ["cssClassPrefix", "theme", "hideAfter", "hideOnMouseOut", "zIndex"]);
+ };
+
+ this.hide = function(props) {
+ props = props || {};
+
+ if (this._state.submenu) {
+ this._state.submenu.menu.hide();
+ }
+
+ var ws = DayPilotMenu.waitingSubmenu;
+ if (ws) {
+ DayPilotMenu.waitingSubmenu = null;
+ clearTimeout(ws.timeout);
+ }
+
+ this.removeShadow();
+ if (this._state.div && this._state.div.parentNode === document.body) {
+ document.body.removeChild(this._state.div);
+ }
+
+/*
+ if (this._initiator) {
+ DayPilot.de(this._initiator);
+ this._initiator = null;
+ }
+*/
+
+ if (initiatorAreaDiv) {
+ DayPilot.de(initiatorAreaDiv);
+ initiatorAreaDiv = null;
+ }
+
+ menu._state.visible = false;
+ menu._state.source = null;
+
+ if (menu._parentMenu && props.hideParent) {
+ menu._parentMenu.hide(props);
+ }
+
+ if (DayPilot.Menu.active === menu) {
+ DayPilot.Menu.active = null;
+ }
+
+ if (typeof this.onHide === "function") {
+ var args = {};
+ this.onHide(args);
+ }
+
+ };
+
+ this.delayedHide = function(props) {
+ DayPilotMenu.hideTimeout = setTimeout(function() {
+ menu.hide(props);
+ }, menu.hideAfter);
+ };
+
+ this.cancelHideTimeout = function() {
+ clearTimeout(DayPilotMenu.hideTimeout);
+ };
+
+ // detects the mouse position, use when creating menu right before opening (.show)
+ this.init = function(ev) {
+ DayPilotMenu.mouseMove(ev);
+ return this;
+ };
+
+ // disabled
+ this.addShadow = function(object) {};
+
+ // disabled
+ this.removeShadow = function() {
+ /* if (!this._state.shadows) {
+ return;
+ }
+
+ for (var i = 0; i < this._state.shadows.length; i++) {
+ document.body.removeChild(this._state.shadows[i]);
+ }
+ this._state.shadows = [];*/
+ };
+
+
+ var options = DayPilot.isArray(items) ? null : items;
+ if (options) {
+ for (var name in options) {
+ this[name] = options[name];
+ }
+ }
+
+ };
+
+ DayPilot.MenuBar = function(id, options) {
+ var menubar = this;
+
+ options = options || {};
+
+ this.items = [];
+ this.theme = "menubar_default";
+ this.windowMargin = 0;
+
+ this.nav = {};
+ this.elements = {};
+ this.elements.items = DayPilot.list();
+
+ this._active = null;
+ this._initialized = false;
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+
+ this._cssClass = function(cl) {
+ return this.theme + "_" + cl;
+ };
+
+ this._show = function() {
+ this.nav.top = document.getElementById(id);
+
+ var top = this.nav.top;
+ top.className = this._cssClass("main");
+
+ DayPilot.list(menubar.items).forEach(function(item) {
+ var div = document.createElement("span");
+ div.innerHTML = DayPilot.Util.escapeTextHtml(item.text, item.html);
+ div.className = menubar._cssClass("item");
+ if (item.cssClass) {
+ div.classList.add(item.cssClass);
+ }
+ div.data = item;
+ div.onclick = function(e) {
+ if (menubar.active && menubar.active.item === item) {
+ menubar._hideActive();
+ }
+ else if (item.children) {
+ menubar._activate(div);
+ return;
+ }
+
+ if (typeof item.onClick === "function") {
+ var args = {};
+ args.item = item;
+ args.originalEvent = e;
+ item.onClick(args);
+ }
+ };
+ div.onmousedown = function(ev) {
+ ev.stopPropagation();
+ };
+ div.onmouseover = function() {
+ if (menubar.active && menubar.active.item !== item) {
+ menubar._activate(div);
+ }
+ };
+
+ top.appendChild(div);
+ menubar.elements.items.push(div);
+ });
+ };
+
+ this._hideActive = function() {
+ var activeCss = menubar._cssClass("item_active");
+ menubar.elements.items.forEach(function(div) {
+ DayPilot.Util.removeClass(div, activeCss);
+ });
+
+ if (menubar.active && menubar.active.menu) {
+ menubar.active.menu.hide();
+ }
+ menubar.active = null;
+ };
+
+ this._isActive = function(div) {
+ if (!menubar.active) {
+ return false;
+ }
+ return menubar.active.item === div.data;
+ };
+
+ this._activate = function(div) {
+ if (menubar._isActive(div)) {
+ return;
+ }
+
+ menubar._hideActive();
+
+ var item = div.data;
+ var a = menubar.active = {};
+ a.item = item;
+ a.div = div;
+
+ var activeCss = menubar._cssClass("item_active");
+ DayPilot.Util.addClass(div, activeCss);
+
+ var abs = DayPilot.abs(div);
+
+ if (item.children) {
+ a.menu = new DayPilot.Menu({"items": item.children});
+ // a.menu.show(null, { "x": abs.x + abs.w, "y": abs.y + abs.h, "align": item.align, "windowMargin": menubar.windowMargin});
+ var x = abs.x;
+ if (item.align === "right") {
+ x += abs.w;
+ }
+ a.menu.show(null, { "x": x, "y": abs.y + abs.h, "align": item.align, "windowMargin": menubar.windowMargin});
+ }
+
+ DayPilot.MenuBar.active = menubar;
+ };
+
+ this.init = function() {
+ this._show();
+ this._initialized = true;
+ return this;
+ };
+
+ this.dispose = function() {
+ if (!this._initialized) {
+ return;
+ }
+ this.nav.top.innerHTML = "";
+ this.elements.items = [];
+ };
+
+ };
+
+ DayPilot.MenuBar.deactivate = function() {
+ if (DayPilot.MenuBar.active) {
+ DayPilot.MenuBar.active._hideActive();
+ DayPilot.MenuBar.active = null;
+ }
+ };
+
+ DayPilotMenu.menuClean = function() {
+ if (typeof(DayPilot.Menu.active) === 'undefined')
+ return;
+
+ if (DayPilot.Menu.active) {
+ DayPilot.Menu.active.hide();
+ DayPilot.Menu.active = null;
+ }
+
+ };
+
+ DayPilotMenu.mouseDown = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ DayPilotMenu.menuClean();
+
+ DayPilot.MenuBar.deactivate();
+ };
+
+ DayPilotMenu.mouseMove = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ DayPilotMenu.mouse = DayPilotMenu.mousePosition(ev);
+ };
+
+ DayPilotMenu.touchMove = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ DayPilotMenu.mouse = DayPilotMenu.touchPosition(ev);
+ };
+
+ DayPilotMenu.touchStart = function(ev) {
+ if (typeof(DayPilotMenu) === 'undefined') {
+ return;
+ }
+ //DayPilotMenu.menuClean();
+ DayPilotMenu.mouse = DayPilotMenu.touchPosition(ev);
+ };
+
+ DayPilotMenu.touchEnd = function(ev) {
+ // do not call menuClean() here, it doesn't work with eventTapAndHoldHandling="ContextMenu"
+ // DayPilotMenu.menuClean();
+ };
+
+ DayPilotMenu.touchPosition = function(ev) {
+ if (!ev || !ev.touches) {
+ return null;
+ }
+ var touch = ev.touches[0];
+ var mouse = {};
+ mouse.x = touch.pageX;
+ mouse.y = touch.pageY;
+ return mouse;
+ };
+
+ DayPilotMenu.mousePosition = function(e) {
+ return DayPilot.mo3(null, e);
+ };
+
+ DayPilot.Menu.touchPosition = function(ev) {
+ if (ev.touches) {
+ DayPilotMenu.mouse = DayPilotMenu.touchPosition(ev);
+ }
+ };
+ // publish the API
+
+ DayPilot.Menu.hide = function(options) {
+ options = options || {};
+
+ if (options.calendar) {
+ var active = DayPilot.Menu.active;
+ if (active) {
+ var source = active._state.source;
+ if (source && source.calendar === options.calendar) {
+ DayPilotMenu.menuClean();
+ }
+ }
+ }
+ else {
+ DayPilotMenu.menuClean();
+ }
+ };
+
+ // current
+ //DayPilot.Menu = DayPilotMenu.Menu;
+ if (!DayPilotMenu.handlersRegistered && typeof document !== 'undefined') {
+ DayPilot.re(document, 'mousemove', DayPilotMenu.mouseMove);
+ DayPilot.re(document, 'mousedown', DayPilotMenu.mouseDown);
+ DayPilot.re(document, 'touchmove', DayPilotMenu.touchMove);
+ DayPilot.re(document, 'touchstart', DayPilotMenu.touchStart);
+ DayPilot.re(document, 'touchend', DayPilotMenu.touchEnd);
+ DayPilotMenu.handlersRegistered = true;
+ }
+
+ DayPilot.Menu.def = {};
+
+})(DayPilot);
diff --git a/static/src/js/src/daypilot-modal.src.js b/static/src/js/src/daypilot-modal.src.js
new file mode 100644
index 0000000..72acfdc
--- /dev/null
+++ b/static/src/js/src/daypilot-modal.src.js
@@ -0,0 +1,3650 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+
+v3.15.0
+https://modal.daypilot.org/
+
+*/
+
+if (typeof(DayPilot) === 'undefined') {
+ DayPilot = {};
+}
+
+(function(DayPilot) {
+ 'use strict';
+
+ // modal.js -> 1482
+
+ if (DayPilot.ModalStatic) {
+ return;
+ }
+
+ DayPilot.ModalStatic = {};
+
+ DayPilot.ModalStatic.list = [];
+
+ // hide the last one
+ DayPilot.ModalStatic.hide = function() {
+ if (this.list.length > 0) {
+ var last = this.list.pop();
+ if (last) {
+ last.hide();
+ }
+ }
+ };
+
+ DayPilot.ModalStatic.remove = function(modal) {
+ var list = DayPilot.ModalStatic.list;
+ for (var i = 0; i < list.length; i++) {
+ if (list[i] === modal) {
+ list.splice(i, 1);
+ return;
+ }
+ }
+ };
+
+ DayPilot.ModalStatic.close = function(result) {
+ DayPilot.ModalStatic.result(result);
+ DayPilot.ModalStatic.hide();
+ };
+
+ DayPilot.ModalStatic.result = function(r) {
+ var list = DayPilot.ModalStatic.list;
+ if (list.length > 0) {
+ list[list.length - 1].result = r;
+ }
+ };
+
+ DayPilot.ModalStatic.displayed = function(modal) {
+ var list = DayPilot.ModalStatic.list;
+ for (var i = 0; i < list.length; i++) {
+ if (list[i] === modal) {
+ return true;
+ }
+ }
+ return false;
+ };
+
+ DayPilot.ModalStatic.stretch = function() {
+ if (this.list.length > 0) {
+ var last = this.list[this.list.length - 1];
+ if (last) {
+ last.stretch();
+ }
+ }
+ };
+
+ DayPilot.ModalStatic.last = function() {
+ var list = DayPilot.ModalStatic.list;
+ if (list.length > 0) {
+ return list[list.length - 1];
+ }
+ return null;
+ };
+
+ var Sheet = function() {
+
+ if (typeof window === "undefined") {
+ // next.js server-side rendering
+ var sheet = {};
+ sheet.add = function() {};
+ sheet.commit = function() {};
+ return sheet;
+ }
+
+ var style = document.createElement("style");
+ style.setAttribute("type", "text/css");
+ if (!style.styleSheet) { // ie
+ style.appendChild(document.createTextNode(""));
+ }
+
+ var h = document.head || document.getElementsByTagName('head')[0];
+ h.appendChild(style);
+
+ var oldStyle = !! style.styleSheet; // old ie
+
+ var sheet = {};
+ sheet.rules = [];
+ sheet.commit = function() {
+ try {
+ if (oldStyle) {
+ style.styleSheet.cssText = this.rules.join("\n");
+ }
+ }
+ catch (e) {
+ //alert("Error registering the built-in stylesheet (IE stylesheet limit reached). Stylesheet count: " + document.styleSheets.length);
+ }
+ };
+
+ sheet.add = function(selector, rules, index) {
+ if (oldStyle) {
+ this.rules.push(selector + "{" + rules + "\u007d");
+ return;
+ }
+ if(style.sheet.insertRule) { // normal browsers, ie9+
+ if (typeof index === "undefined") {
+ index = style.sheet.cssRules.length;
+ }
+ style.sheet.insertRule(selector + "{" + rules + "\u007d", index);
+ }
+ else if (style.sheet.addRule) {
+ style.sheet.addRule(selector, rules, index);
+ }
+ else {
+ throw "No CSS registration method found";
+ }
+ };
+ return sheet;
+ };
+
+ var iconCalendar = "";
+ var iconExpand = "";
+
+ var sheet = new Sheet();
+ sheet.add(".modal_default_main", "border: 10px solid #ccc; max-width: 90%;");
+ sheet.add(".modal_default_main:focus", "outline: none;");
+ sheet.add(".modal_default_content", "padding: 10px 0px;");
+ sheet.add(".modal_default_inner", "padding: 20px;");
+ sheet.add(".modal_default_input", "padding: 10px 0px;");
+ sheet.add(".modal_default_buttons", "margin-top: 10px;");
+ sheet.add(".modal_default_buttons", "padding: 10px 0px;");
+ sheet.add(".modal_default_form_item", "padding: 10px 0px; position: relative;");
+ sheet.add(".modal_default_form_item_level1", "border-left: 2px solid #ccc; margin-left: 10px; padding-left: 20px;");
+ sheet.add(".modal_default_form_item.modal_default_form_title", "font-size: 1.5rem; font-weight: bold;");
+ sheet.add(".modal_default_form_item input[type=text]", "width: 100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item textarea", "width: 100%; height: 200px; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item input[type=select]", "width: 100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item label", "display: block;");
+ sheet.add(".modal_default_form_item select", "width: 100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item_label", "margin: 2px 0px;");
+ sheet.add(".modal_default_form_item_image img", "max-width: 100%; height: auto;");
+
+ sheet.add(".modal_default_form_item_invalid", "");
+ sheet.add(".modal_default_form_item_invalid_message", "position: absolute; right: 0px; top: 9px; background-color: red; color: #ffffff; padding: 2px; border-radius: 2px;");
+
+ sheet.add(".modal_default_background", "opacity: 0.5; background-color: #000;");
+ sheet.add(".modal_default_ok", "padding: 3px; width: 80px;");
+ sheet.add(".modal_default_cancel", "padding: 3px; width: 80px;");
+
+ sheet.add(".modal_default_form_item_date", "position: relative;");
+ sheet.add(".modal_default_form_item_date:after", "content: ''; position: absolute; right: 7px; top: 50%; margin-top: 3px; width: 10px; height: 15px; background-image:url(" + iconCalendar + ")");
+
+ if (typeof navigator !== "undefined" && navigator.userAgent.indexOf("Edge") !== -1) {
+ sheet.add(".modal_default_form_item_date input::-ms-clear", "display: none;");
+ }
+
+ sheet.add(".modal_default_form_item_scrollable_scroll", "width: 100%; height: 200px; box-sizing: border-box; border: 1px solid #ccc; overflow-y: auto;");
+ sheet.add(".modal_default_form_item_scrollable_scroll_content", "padding: 5px;");
+
+ sheet.add(".modal_default_form_item_searchable", "position: relative;");
+ sheet.add(".modal_default_form_item_searchable_icon", "");
+ sheet.add(".modal_default_form_item_searchable_icon:after", "content:''; position: absolute; right: 5px; top: 50%; margin-top: -8px; width: 10px; height: 15px; background-image:url(" + iconExpand + ");");
+ sheet.add(".modal_default_form_item_searchable_list", "box-sizing: border-box; border: 1px solid #999; max-height: 150px; overflow-y: auto;");
+ sheet.add(".modal_default_form_item_searchable_list_item", "background: white; padding: 2px; cursor: default;");
+ sheet.add(".modal_default_form_item_searchable_list_item_highlight", "background: #ccc;");
+
+ sheet.add(".modal_default_form_item_time", "position: relative;");
+ sheet.add(".modal_default_form_item_time_icon", "");
+ sheet.add(".modal_default_form_item_time_icon:after", "content:''; position: absolute; right: 5px; top: 50%; margin-top: -8px; width: 10px; height: 15px; background-image:url(" + iconExpand + ");");
+ sheet.add(".modal_default_form_item_time_list", "box-sizing: border-box; border: 1px solid #999; max-height: 150px; overflow-y: auto;");
+ sheet.add(".modal_default_form_item_time_list_item", "background: white; padding: 2px; cursor: default;");
+ sheet.add(".modal_default_form_item_time_list_item_highlight", "background: #ccc;");
+
+ sheet.add(".modal_default_form_item_datetime_parent", "display: flex;");
+ sheet.add(".modal_default_form_item_datetime .modal_default_form_item_time_main", "margin-left: 5px;");
+ sheet.add(".modal_default_form_item_datetime input[type='text'].modal_default_input_date ", "");
+
+ sheet.add(".modal_default_form_item_tabular_main", "margin-top: 10px;");
+ sheet.add(".modal_default_form_item_tabular_table", "display: table; width: 100%; xbackground-color: #fff; border-collapse: collapse;");
+ sheet.add(".modal_default_form_item_tabular_tbody", "display: table-row-group;");
+ sheet.add(".modal_default_form_item_tabular_row", "display: table-row;");
+ sheet.add(".modal_default_form_item_tabular_row.modal_default_form_item_tabular_header", ""); // used to be bold
+ sheet.add(".modal_default_form_item_tabular_cell.modal_default_form_item_tabular_rowaction", "padding: 0px; width: 23px;"); // _delete: width + marginLeft
+ sheet.add(".modal_default_form_item_tabular_cell", "display: table-cell; border: 0px; padding: 2px 2px 2px 0px; cursor: default; vertical-align: bottom;");
+ sheet.add(".modal_default_form_item_tabular_header .modal_default_form_item_tabular_cell", "padding-left: 0px; padding-bottom: 0px;");
+ sheet.add(".modal_default_form_item_tabular_table input[type=text], .modal_default_form_item_tabular_table input[type=number]", "width:100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item_tabular_table select", "width:100%; height:100%; box-sizing: border-box;");
+ sheet.add(".modal_default_form_item_tabular_plus", "display: inline-block; background-color: #ccc; color: white; width: 20px; height: 20px; border-radius: 10px; box-sizing: border-box; position: relative; margin-left: 3px; margin-top: 3px; cursor: pointer;");
+ sheet.add(".modal_default_form_item_tabular_plus:after", "content: ''; position: absolute; left: 5px; top: 5px; width: 10px; height: 10px; background-image: url(\"\")");
+ sheet.add(".modal_default_form_item_tabular_delete", "display: inline-block; background-color: #ccc; color: white; width: 20px; height: 20px; border-radius: 10px; box-sizing: border-box; position: relative; margin-left: 3px; margin-top: 3px; cursor: pointer;");
+ sheet.add(".modal_default_form_item_tabular_delete:after", "content: ''; position: absolute; left: 5px; top: 5px; width: 10px; height: 10px; background-image: url(\"\")");
+ sheet.add(".modal_default_form_item_tabular_disabled .modal_default_form_item_tabular_plus", "display: none;");
+ sheet.add(".modal_default_form_item_tabular_plus_max.modal_default_form_item_tabular_plus", "display: none;");
+ sheet.add(".modal_default_form_item_tabular_disabled .modal_default_form_item_tabular_delete", "visibility: hidden;");
+ sheet.add(".modal_default_form_item_tabular_empty", "height: 1px; margin: 5px 23px 5px 0px; background-color: #ccc;");
+ sheet.add(".modal_default_form_item_tabular_spacer .modal_default_form_item_tabular_cell", "padding: 0px;");
+
+ sheet.add(".modal_min_main", "border: 1px solid #ccc; max-width: 90%;");
+ sheet.add(".modal_min_background", "opacity: 0.5; background-color: #000;");
+ sheet.add(".modal_min_ok", "padding: 3px 10px;");
+ sheet.add(".modal_min_cancel", "padding: 3px 10px;");
+
+
+ sheet.add(".navigator_modal_main", "border-left: 1px solid #c0c0c0;border-right: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;background-color: white;color: #000000; box-sizing: content-box;");
+ sheet.add(".navigator_modal_main *, .navigator_modal_main *:before, .navigator_modal_main *:after", "box-sizing: content-box;");
+ sheet.add(".navigator_modal_month", "font-size: 11px;");
+ sheet.add(".navigator_modal_day", "color: black;");
+ sheet.add(".navigator_modal_weekend", "background-color: #f0f0f0;");
+ sheet.add(".navigator_modal_dayheader", "color: black;");
+ sheet.add(".navigator_modal_line", "border-bottom: 1px solid #c0c0c0;");
+ sheet.add(".navigator_modal_dayother", "color: gray;");
+ sheet.add(".navigator_modal_todaybox", "border: 1px solid red;");
+ sheet.add(".navigator_modal_title, .navigator_modal_titleleft, .navigator_modal_titleright", 'border-top: 1px solid #c0c0c0;border-bottom: 1px solid #c0c0c0;color: #333;background: #f3f3f3;');
+ sheet.add(".navigator_modal_busy", "font-weight: bold;");
+ sheet.add(".navigator_modal_cell", "text-align: center;");
+ sheet.add(".navigator_modal_select .navigator_modal_cell_box", "background-color: #FFE794; opacity: 0.5;");
+ sheet.add(".navigator_modal_title", "text-align: center;");
+ sheet.add(".navigator_modal_titleleft, .navigator_modal_titleright", "text-align: center;");
+ sheet.add(".navigator_modal_dayheader", "text-align: center;");
+ sheet.add(".navigator_modal_weeknumber", "text-align: center;");
+ sheet.add(".navigator_modal_cell_text", "cursor: pointer;");
+
+ sheet.commit();
+
+ DayPilot.Modal = function(options) {
+
+ // default values
+ this.autoFocus = true;
+ this.focus = null; // field to be focused - field id as string or {id, value}
+ this.autoStretch = true; // height will be increased automatically to avoid scrollbar, until this.maxHeight is reached
+ this.autoStretchFirstLoadOnly = false;
+ this.className = null;
+ this.theme = "modal_default";
+ this.disposeOnClose = true;
+ this.dragDrop = true;
+ this.loadingHtml = null;
+ this.maxHeight = null; // if not set, it will stretch until the bottom space is equal to this.top
+ this.scrollWithPage = true; // modal window will scroll with the page
+ this.useIframe = true; // only for showHtml()
+ this.zIndex = 99999;
+
+ this.left = null; // will be centered if null
+ this.width = 600;
+ this.top = 20;
+ this.height = 200; // see also autoStretch
+ this.locale = null; // used for DayPilot.Modal.form() items with "date" type
+
+ // event handler
+ this.closed = null;
+ this.onClose = null;
+ this.onClosed = null;
+ this.onShow = null;
+
+ // internal
+ var This = this;
+ this.id = '_' + new Date().getTime() + 'n' + (Math.random() * 10);
+ this._registered = false;
+
+ // drag&drop
+ this._start = null;
+ this._coords = null;
+
+ this.showHtml = function(html) {
+
+ if (DayPilot.ModalStatic.displayed(this)) {
+ throw "This modal dialog is already displayed.";
+ }
+
+ if (!this.div) {
+ this._create();
+ }
+ this._update();
+
+ if (this.useIframe) {
+ var delayed = function(p, innerHTML) {
+ return function() {
+ p.setInnerHTML(p.id + "iframe", innerHTML);
+ };
+ };
+
+ window.setTimeout(delayed(this, html), 0);
+ }
+ else {
+ if (html.nodeType) {
+ this.div.appendChild(html);
+ }
+ else {
+ this.div.innerHTML = html;
+ }
+ }
+
+ this._update();
+ this._register();
+ this._doShow();
+
+ };
+
+ this.showUrl = function(url) {
+
+ if (DayPilot.ModalStatic.displayed(this)) {
+ throw "This modal dialog is already displayed.";
+ }
+
+ //this.useIframe = true; // forced
+
+ if (this.useIframe) {
+ if (!this.div) {
+ this._create();
+ }
+
+ var loadingHtml = this.loadingHtml;
+ if (loadingHtml) {
+ this.iframe.src = "about:blank";
+ this.setInnerHTML(this.id + "iframe", loadingHtml);
+ }
+
+ this.re(this.iframe, "load", this._onIframeLoad);
+
+ this.iframe.src = url;
+ //this.iframe.contentWindow.modal = This;
+
+ this._update();
+ this._register();
+ this._doShow();
+ }
+ else {
+ This._ajax({
+ "url": url,
+ "success": function(args) {
+ var html = args.request.responseText;
+ This.showHtml(html);
+ },
+ "error": function(args) {
+ This.showHtml("Error loading the modal dialog");
+ }
+ });
+ }
+
+ };
+
+ this._doShow = function() {
+ if (typeof This.onShow === "function") {
+ var args = {};
+ args.root = This._body();
+ args.modal = This;
+ This.onShow(args);
+ }
+ };
+
+ this._body = function() {
+ return This.iframe ? This.iframe.contentWindow.document : This.div;
+ };
+
+ this._ajax = function(object) {
+ var req = new XMLHttpRequest();
+ if (!req) {
+ return;
+ }
+
+ var method = object.method || "GET";
+ var success = object.success || function() {};
+ var error = object.error || function() {};
+ var data = object.data;
+ var url = object.url;
+
+ req.open(method, url, true);
+ req.setRequestHeader('Content-type', 'text/plain');
+ req.onreadystatechange = function() {
+ if (req.readyState !== 4)
+ return;
+ if (req.status !== 200 && req.status !== 304) {
+ if (error) {
+ var args = {};
+ args.request = req;
+ error(args);
+ }
+ else {
+ if (window.console) { console.log('HTTP error ' + req.status); }
+ }
+ return;
+ }
+ var args = {};
+ args.request = req;
+ success(args);
+ };
+ if (req.readyState === 4) {
+ return;
+ }
+ if (typeof data === 'object') {
+ data = JSON.stringify(data);
+ }
+ req.send(data);
+ };
+
+ this._update = function() {
+
+ delete this.result;
+
+ var win = window;
+ var doc = document;
+
+ var scrollY = win.pageYOffset ? win.pageYOffset : ((doc.documentElement && doc.documentElement.scrollTop) ? doc.documentElement.scrollTop : doc.body.scrollTop);
+
+ var height = function() {
+ return This._windowRect().y;
+ };
+
+ //this.hideDiv.style.height = height() + "px";
+ if (this.theme) {
+ this.hideDiv.className = this.theme + "_background";
+ }
+ if (this.zIndex) {
+ this.hideDiv.style.zIndex = this.zIndex;
+ }
+ this.hideDiv.style.display = '';
+
+ window.setTimeout(function() {
+ if (This.hideDiv) {
+ This.hideDiv.onclick = function() {
+ This.hide({"backgroundClick": true});
+ };
+ }
+ }, 500);
+
+ if (this.theme) {
+ this.div.className = this.theme + "_main";
+ }
+ else {
+ this.div.className = "";
+ }
+
+ if (this.className) {
+ this.div.className += " " + this.className;
+ }
+ if (this.left) {
+ this.div.style.left = this.left + "px";
+ }
+ else {
+ this.div.style.marginLeft = '-' + Math.floor(this.width / 2) + "px"; // '-45%'
+ }
+ this.div.style.position = 'absolute';
+ this.div.style.boxSizing = "content-box";
+ this.div.style.top = (scrollY + this.top) + 'px';
+ this.div.style.width = this.width + 'px'; // '90%'
+ if (this.zIndex) {
+ this.div.style.zIndex = this.zIndex;
+ }
+
+ if (this.height) {
+ if (this.useIframe || !this.autoStretch) {
+ this.div.style.height = this.height + 'px';
+ }
+ else {
+ this.div.style.height = '';
+ }
+ }
+ if (this.useIframe && this.height) {
+ this.iframe.style.height = (this.height) + 'px';
+ }
+
+ this.div.style.display = '';
+
+ this._updateHorizontal();
+
+ // make sure it's there just once
+ DayPilot.ModalStatic.remove(this);
+ DayPilot.ModalStatic.list.push(this);
+
+ /*
+ if (this.iframe) {
+ this.iframe.onload = null;
+ }
+ */
+ };
+
+ this._onIframeLoad = function() {
+ This.iframe.contentWindow.modal = This;
+ if (This.autoStretch) {
+ This.stretch();
+ }
+ };
+
+ this.stretch = function() {
+ var height = function() {
+ return This._windowRect().y;
+ };
+
+ var width = function() {
+ return This._windowRect().x;
+ };
+
+
+ if (this.useIframe) {
+ // width first
+ var maxWidth = width() - 40; // fixed 20px margin
+ for (var w = this.width; w < maxWidth && this._hasHorizontalScrollbar(); w += 10) {
+ //this.iframe.style.width = (w) + 'px';
+ this.div.style.width = w + 'px';
+ this.div.style.marginLeft = '-' + Math.floor(w / 2) + "px"; //
+ }
+
+ // height
+ var maxHeight = this.maxHeight || height() - 2 * this.top;
+ for (var h = this.height; h < maxHeight && this._hasVerticalScrollbar(); h += 10) {
+ this.iframe.style.height = (h) + 'px';
+ this.div.style.height = h + 'px';
+ }
+
+ if (this.autoStretchFirstLoadOnly) {
+ this.ue(this.iframe, "load", this._onIframeLoad);
+ }
+ }
+ else {
+ this.div.style.height = '';
+ }
+
+
+ };
+
+ this._hasHorizontalScrollbar = function() {
+ var document = this.iframe.contentWindow.document;
+ var root = document.compatMode === 'BackCompat' ? document.body : document.documentElement;
+
+ var scrollWidth = root.scrollWidth;
+ var children = document.body.children;
+ for (var i = 0; i < children.length; i++) {
+ var bottom = children[i].offsetLeft + children[i].offsetWidth;
+ scrollWidth = Math.max(scrollWidth, bottom);
+ }
+
+ var isHorizontalScrollbar = scrollWidth > root.clientWidth;
+ return isHorizontalScrollbar;
+
+ };
+
+ this._hasVerticalScrollbar = function() {
+ var document = this.iframe.contentWindow.document;
+ var root = document.compatMode === 'BackCompat' ? document.body : document.documentElement;
+
+ var scrollHeight = root.scrollHeight;
+ var children = document.body.children;
+ for (var i = 0; i < children.length; i++) {
+ var bottom = children[i].offsetTop + children[i].offsetHeight;
+ scrollHeight = Math.max(scrollHeight, bottom);
+ }
+
+ var isVerticalScrollbar = scrollHeight > root.clientHeight;
+ //var isHorizontalScrollbar = root.scrollWidth > root.clientWidth;
+ return isVerticalScrollbar;
+ };
+
+ this._windowRect = function() {
+ var doc = document;
+
+ if (doc.compatMode === "CSS1Compat" && doc.documentElement && doc.documentElement.clientWidth) {
+ var x = doc.documentElement.clientWidth;
+ var y = doc.documentElement.clientHeight;
+ return { x: x, y: y };
+ }
+ else {
+ var x = doc.body.clientWidth;
+ var y = doc.body.clientHeight;
+ return { x: x, y: y };
+ }
+ };
+
+ this._register = function() {
+ if (this._registered) {
+ return;
+ }
+ this.re(window, 'resize', this._onWindowResize);
+ this.re(window, 'scroll', this._onWindowScroll);
+
+ if (this.dragDrop) {
+ this.re(document, 'mousemove', this._onMouseMove);
+ this.re(document, 'mouseup', this._onMouseUp);
+ }
+ this._registered = true;
+ };
+
+ this._unregister = function() {
+ this.ue(window, 'resize', this._onWindowResize);
+ this.ue(window, 'scroll', this._onWindowScroll);
+ if (this.dragDrop) {
+ this.ue(document, 'mousemove', this._onMouseMove);
+ this.ue(document, 'mouseup', this._onMouseUp);
+ }
+ this._registered = false;
+ };
+
+ this._onDragStart = function(e) {
+ if (e.target !== This.div) {
+ return;
+ }
+ e.preventDefault();
+ This.div.style.cursor = "move";
+ This._maskIframe();
+ This._coords = This.mc(e || window.event);
+ This._start = { x: This.div.offsetLeft, y: This.div.offsetTop };
+
+ };
+
+ this._onMouseMove = function(e) {
+ if (!This._coords) {
+ return;
+ }
+
+ var e = e || window.event;
+ var now = This.mc(e);
+
+ var x = now.x - This._coords.x;
+ var y = now.y - This._coords.y;
+
+ //This.iframe.style.display = 'none';
+ This.div.style.marginLeft = '0px';
+ This.div.style.top = (This._start.y + y) + "px";
+ This.div.style.left = (This._start.x + x) + "px";
+
+ };
+
+ this._onMouseUp = function(e) {
+ // no drag&drop
+ if (!This._coords) {
+ return;
+ }
+ //This.iframe.style.display = '';
+ This._unmaskIframe();
+ This.div.style.cursor = null;
+
+ This._coords = null;
+ };
+
+ this._maskIframe = function() {
+ if (!this.useIframe) {
+ return;
+ }
+
+ var opacity = 80;
+
+ var mask = document.createElement("div");
+ mask.style.backgroundColor = "#ffffff";
+ mask.style.filter = "alpha(opacity=" + opacity + ")";
+ mask.style.opacity = "0." + opacity;
+ mask.style.width = "100%";
+ mask.style.height = this.height + "px";
+ mask.style.position = "absolute";
+ mask.style.left = '0px';
+ mask.style.top = '0px';
+
+ this.div.appendChild(mask);
+ this.mask = mask;
+ };
+
+ this._unmaskIframe = function() {
+ if (!this.useIframe) {
+ return;
+ }
+
+ this.div.removeChild(this.mask);
+ this.mask = null;
+ };
+
+ this._onWindowResize = function() {
+ This._updateTop();
+ This._updateHorizontal();
+ };
+
+ this._onWindowScroll = function() {
+ This._updateTop();
+ };
+
+ this._updateHorizontal = function() {
+ if (This.left) {
+ return;
+ }
+
+ if (!This.div) {
+ return;
+ }
+
+ var width = This.div.offsetWidth;
+ This.div.style.marginLeft = '-' + Math.floor(width / 2) + "px"; // '-45%'
+ };
+
+ this._updateTop = function() {
+ if (!This.hideDiv) {
+ return;
+ }
+ if (!This.div) {
+ return;
+ }
+ if (This.hideDiv.style.display === 'none') {
+ return;
+ }
+ if (This.div.style.display === 'none') {
+ return;
+ }
+
+ var scrollY = This._parent.scrollY();
+
+
+ //This.hideDiv.style.height = height() + "px";
+ if (!This.scrollWithPage) {
+ This.div.style.top = (scrollY + This.top) + 'px';
+ }
+ };
+
+ this._parent = {};
+ this._parent.container = function() {
+ return This.container || document.body;
+ };
+ this._parent.scrollY = function() {
+ var c = This._parent.container();
+ if (c === document.body) {
+ return window.pageYOffset ? window.pageYOffset : ((document.documentElement && document.documentElement.scrollTop) ? document.documentElement.scrollTop : document.body.scrollTop);
+ }
+ else {
+ return c.scrollTop;
+ }
+ };
+
+ // already available in common.js but this file should be standalone
+ this.re = function(el, ev, func) {
+ if (el.addEventListener) {
+ el.addEventListener(ev, func, false);
+ } else if (el.attachEvent) {
+ el.attachEvent("on" + ev, func);
+ }
+ };
+
+ // unregister event
+ this.ue = function(el, ev, func) {
+ if (el.removeEventListener) {
+ el.removeEventListener(ev, func, false);
+ } else if (el.detachEvent) {
+ el.detachEvent("on" + ev, func);
+ }
+ };
+
+ // mouse coords
+ this.mc = function(ev) {
+ if (ev.pageX || ev.pageY) {
+ return { x: ev.pageX, y: ev.pageY };
+ }
+ return {
+ x: ev.clientX + document.documentElement.scrollLeft,
+ y: ev.clientY + document.documentElement.scrollTop
+ };
+ };
+
+ // absolute element position on page
+ this.abs = function(element) {
+ var r = {
+ x: element.offsetLeft,
+ y: element.offsetTop
+ };
+
+ while (element.offsetParent) {
+ element = element.offsetParent;
+ r.x += element.offsetLeft;
+ r.y += element.offsetTop;
+ }
+
+ return r;
+ };
+
+ this._create = function() {
+
+ var container = This._parent.container();
+ var isRoot = container === document.body;
+ var position = isRoot ? "fixed" : "absolute";
+
+ var hide = document.createElement("div");
+ hide.id = this.id + "hide";
+ hide.style.position = position;
+ hide.style.left = "0px";
+ hide.style.top = "0px";
+ hide.style.right = "0px";
+ hide.style.bottom = "0px";
+ hide.oncontextmenu = function() { return false; };
+ hide.onmousedown = function() { return false; }; // prevent selecting
+
+ container.appendChild(hide);
+
+ var div = document.createElement("div");
+ div.id = this.id + 'popup';
+ div.style.position = position;
+ div.style.left = '50%';
+ div.style.top = '0px';
+ div.style.backgroundColor = 'white';
+ div.style.width = "50px";
+ div.style.height = "50px";
+ if (this.dragDrop) {
+ div.onmousedown = this._onDragStart;
+ }
+ div.addEventListener("keydown", function(e) {
+ // prevent interaaction with the document using keyboard
+ e.stopPropagation();
+ });
+
+ var defaultHeight = 50;
+
+ var iframe = null;
+ if (this.useIframe) {
+ iframe = document.createElement("iframe");
+ iframe.id = this.id + "iframe";
+ iframe.name = this.id + "iframe";
+ iframe.frameBorder = '0';
+ iframe.style.width = '100%';
+ iframe.style.height = defaultHeight + 'px';
+ div.appendChild(iframe);
+ }
+
+ container.appendChild(div);
+
+ this.div = div;
+ this.iframe = iframe;
+ this.hideDiv = hide;
+ };
+
+ this.setInnerHTML = function(id, innerHTML) {
+ var frame = window.frames[id];
+
+ var doc = frame.contentWindow || frame.document || frame.contentDocument;
+ if (doc.document) {
+ doc = doc.document;
+ }
+
+ if (doc.body == null) { // null in IE
+ doc.write("");
+ }
+
+ if (innerHTML.nodeType) {
+ doc.body.appendChild(innerHTML);
+ }
+ else {
+ doc.body.innerHTML = innerHTML;
+ }
+
+ if (This.autoStretch) {
+ if (!This.autoStretchFirstLoadOnly || !This._stretched) {
+ This.stretch();
+ This._stretched = true;
+ }
+ }
+ };
+
+ this.close = function(result) {
+ this.result = result;
+ this.hide();
+ };
+
+ this.closeSerialized = function() {
+ var ref = This._body();
+ var fields = ref.querySelectorAll("input, textarea, select");
+ var result = {};
+ for (var i = 0; i < fields.length; i++) {
+ var field = fields[i];
+ var name = field.name;
+ if (!name) {
+ continue;
+ }
+ var value = field.value;
+ /*if (field.picker) {
+ value = field.picker.date ? field.picker.date.toString() : null;
+ }
+ else*/ /*if (field.table) {
+ value = field.table.save();
+ }
+ else*/ /*if (field.searchable) {
+ value = field.searchable.selected && field.searchable.selected.id;
+ }
+ else *//*if (field.tagName === "SELECT") {
+ var option = field.options[field.selectedIndex];
+ if (option && typeof option._originalValue !== "undefined") {
+ value = option._originalValue;
+ }
+ }
+ else*/ /*if (field.type === "radio") {
+ if (!field.checked) {
+ continue;
+ }
+ value = field._originalValue;
+ }
+ else*/ /*if (field.type === "checkbox") {
+ value = field.checked;
+ }*/
+ result[name] = value;
+ }
+ This.close(result);
+ };
+
+ this.hide = function(options) {
+
+ options = options || {};
+
+ var args = {};
+ args.backgroundClick = !!options.backgroundClick;
+ args.result = this.result;
+ args.canceled = typeof this.result === "undefined";
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof this.onClose === "function") {
+ this.onClose(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ if (this.div) {
+ this.div.style.display = 'none';
+ this.hideDiv.style.display = 'none';
+ if (!this.useIframe) {
+ this.div.innerHTML = null;
+ }
+ }
+
+ // return focus to the main window (Firefox)
+ window.focus();
+
+ //DayPilot.ModalStatic = null;
+ DayPilot.ModalStatic.remove(this);
+
+ if (typeof this.onClosed === "function") {
+ this.onClosed(args);
+ }
+ else if (this.closed) {
+ this.closed();
+ }
+
+ delete this.result;
+
+ if (this.disposeOnClose) {
+
+ This._unregister();
+
+ This._de(This.div);
+ This._de(This.hideDiv);
+
+ This.div = null;
+ This.hideDiv = null;
+ This.iframe = null;
+ }
+ };
+
+ this._de = function(e) {
+ if (!e) {
+ return;
+ }
+ e.parentNode && e.parentNode.removeChild(e);
+ };
+
+ this._applyOptions = function() {
+ if (!options) {
+ return;
+ }
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+ };
+
+ this._applyOptions();
+
+ };
+
+ DayPilot.Modal.alert = function(message, options) {
+ options = options || {};
+ options.height = options.height || 40;
+ options.useIframe = false;
+
+ var okText = options.okText || "OK";
+ var cancelText = options.cancelText || "Cancel";
+
+ return DayPilot.getPromise(function(success, failure) {
+ options.onClosed = function(args) {
+ success(args);
+ /*
+ if (typeof args.result === "undefined") {
+ failure(args);
+ }
+ else {
+
+ }*/
+ };
+
+ var modal = new DayPilot.Modal(options);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+ // div.style.padding = "10px";
+
+ var text = document.createElement("div");
+ text.className = modal.theme + "_content";
+ text.innerHTML = message;
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+ // buttons.style.margin = "10px 0px";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+ buttonOK.onclick = function(e) {
+ DayPilot.ModalStatic.close("OK");
+ };
+
+ buttons.appendChild(buttonOK);
+
+ div.appendChild(text);
+ div.appendChild(buttons);
+
+ //var buttons = "";
+
+ modal.showHtml(div);
+
+ if (modal.autoFocus) {
+ buttonOK.focus();
+ }
+ });
+
+ };
+
+ DayPilot.Modal.confirm = function(message, options) {
+ options = options || {};
+ options.height = options.height || 40;
+ options.useIframe = false;
+
+ var okText = options.okText || "OK";
+ var cancelText = options.cancelText || "Cancel";
+
+ return DayPilot.getPromise(function(success, failure) {
+ options.onClosed = function(args) {
+ success(args);
+ };
+
+ var modal = new DayPilot.Modal(options);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+ // div.style.padding = "10px";
+
+ var text = document.createElement("div");
+ text.className = modal.theme + "_content";
+ text.innerHTML = message;
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+ // buttons.style.margin = "10px 0px";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+ buttonOK.onclick = function(e) {
+ DayPilot.ModalStatic.close("OK");
+ };
+
+ var space = document.createTextNode(" ");
+
+ var buttonCancel = document.createElement("button");
+ buttonCancel.innerText = cancelText;
+ buttonCancel.className = modal.theme + "_cancel";
+ buttonCancel.onclick = function(e) {
+ DayPilot.ModalStatic.close();
+ };
+
+ buttons.appendChild(buttonOK);
+ buttons.appendChild(space);
+ buttons.appendChild(buttonCancel);
+
+ div.appendChild(text);
+ div.appendChild(buttons);
+
+ modal.showHtml(div);
+
+ if (modal.autoFocus) {
+ buttonOK.focus();
+ }
+
+ });
+ };
+
+
+ DayPilot.Modal.prompt = function(message, defaultValue, options) {
+ if (typeof defaultValue === "object") {
+ options = defaultValue;
+ defaultValue = "";
+ }
+
+ options = options || {};
+ options.height = options.height || 40;
+ options.useIframe = false;
+
+ var okText = options.okText || "OK";
+ var cancelText = options.cancelText || "Cancel";
+
+ var inputText = defaultValue || "";
+
+ return DayPilot.getPromise(function(success, failure) {
+ options.onClosed = function(args) {
+ success(args);
+ };
+
+ var modal = new DayPilot.Modal(options);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+
+ var text = document.createElement("div");
+ text.className = modal.theme + "_content";
+ text.innerHTML = message;
+
+ var inputs = document.createElement("div");
+ inputs.className = modal.theme + "_input";
+ //inputs.style.margin = "10px 0px";
+
+ var input = document.createElement("input");
+ input.value = inputText;
+ input.style.width = "100%";
+ input.onkeydown = function(e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13:
+ modal.close(this.value);
+ break;
+ case 27:
+ modal.close();
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ inputs.appendChild(input);
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+ //buttons.style.margin = "10px 0px";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+ buttonOK.onclick = function(e) {
+ modal.close(input.value);
+ };
+
+ var space = document.createTextNode(" ");
+
+ var buttonCancel = document.createElement("button");
+ buttonCancel.innerText = cancelText;
+ buttonCancel.className = modal.theme + "_cancel";
+ buttonCancel.onclick = function(e) {
+ modal.close();
+ };
+
+ buttons.appendChild(buttonOK);
+ buttons.appendChild(space);
+ buttons.appendChild(buttonCancel);
+
+
+ div.appendChild(text);
+ div.appendChild(inputs);
+ div.appendChild(buttons);
+
+ modal.showHtml(div);
+
+ if (modal.autoFocus) {
+ input.focus();
+ }
+
+ });
+ };
+
+ var isArray = function(arg) {
+ return Object.prototype.toString.call(arg) === '[object Array]';
+ };
+
+ function setPathValue(target, path, value) {
+ var iodot = path.indexOf(".");
+ if (iodot === -1) {
+ if (path !== "__proto__" && path !== "constructor") {
+ target[path] = value;
+ }
+ return;
+ }
+ var segment = path.substring(0, iodot);
+ if (segment === "__proto__" || segment === "constructor") {
+ return;
+ }
+ var remainder = path.substring(iodot + 1);
+ var child = target[segment];
+ if (typeof child !== "object" || child === null) {
+ target[segment] = {};
+ child = target[segment];
+ }
+ setPathValue(child, remainder, value);
+ }
+
+ // form = array of items
+ // data = object with properties
+ // options = standard modal properties
+ // DayPilot.Modal.form([]); // form only - new data
+ // DayPilot.Modal.form({}); // data only - automatic form
+ //
+ DayPilot.Modal.form = function(form, data, options) {
+ if (arguments.length === 1) {
+ var arg = form;
+ var isa = isArray(arg);
+
+ if (isa) {
+ data = {};
+ }
+ else if (typeof arg === "object") {
+ data = form;
+ form = [];
+ for (var name in data) {
+ var item = {};
+ item.name = name;
+ item.id = name;
+ form.push(item);
+ }
+ }
+ else {
+ throw "Invalid DayPilot.Modal.form() parameter";
+ }
+
+ }
+
+ // make a copy
+ var opts = {};
+ for (var name in options) {
+ opts[name] = options[name];
+ }
+
+ // options = options || {};
+ opts.height = opts.height || 40;
+ opts.useIframe = false;
+
+ /* if (typeof opts.autoFocus === "undefined") {
+ opts.autoFocus = false;
+ }*/
+
+ var okText = opts.okText || "OK";
+ var cancelText = opts.cancelText || "Cancel";
+
+ // var message = opts.message || "";
+
+ return DayPilot.getPromise(function(success, failure) {
+ opts.onClosed = function(args) {
+
+ if (args.result) {
+ // deep copy
+ var mergedResult = JSON.parse(JSON.stringify(data));
+
+ // unflatten
+ for (var name in args.result) {
+ setPathValue(mergedResult, name, args.result[name]);
+ }
+ args.result = mergedResult;
+ }
+
+ success(args);
+ };
+
+ var modal = new DayPilot.Modal(opts);
+
+ var div = document.createElement("div");
+ div.className = modal.theme + "_inner";
+
+ var inputs = document.createElement("div");
+ inputs.className = modal.theme + "_input";
+
+ var f = new Form({
+ theme: modal.theme,
+ form: form,
+ data: data,
+ zIndex: modal.zIndex,
+ locale: modal.locale,
+ plugins: modal.plugins,
+ onKey: function(args) {
+ switch (args.key) {
+ case "Enter":
+ // modal.closeSerialized();
+ // modal.close(f.serialize());
+ if (f.validate()) {
+ modal.close(f.serialize());
+ }
+ break;
+ case "Escape":
+ modal.close();
+ break;
+ }
+ },
+ onChange: function(args) {
+ if (typeof modal.onChange === "function") {
+ modal.onChange(args);
+ }
+ }
+ });
+ var el = f.create();
+
+ inputs.append(el);
+
+ var buttons = document.createElement("div");
+ buttons.className = modal.theme + "_buttons";
+
+ var buttonOK = document.createElement("button");
+ buttonOK.innerText = okText;
+ buttonOK.className = modal.theme + "_ok";
+
+ if (opts.okDisabled) {
+ buttonOK.disabled = true;
+ }
+ buttonOK.onclick = function(e) {
+ if (f.validate()) {
+ modal.close(f.serialize());
+ }
+ // modal.closeSerialized();
+ };
+
+ var space = document.createTextNode(" ");
+
+ var buttonCancel = document.createElement("button");
+ buttonCancel.innerText = cancelText;
+ buttonCancel.className = modal.theme + "_cancel";
+ buttonCancel.onclick = function(e) {
+ modal.close();
+ };
+ buttonCancel.onmousedown = function(e) {
+ f.canceling = true;
+ };
+
+ buttons.appendChild(buttonOK);
+ buttons.appendChild(space);
+ buttons.appendChild(buttonCancel);
+
+
+ // div.appendChild(text);
+ div.appendChild(inputs);
+ div.appendChild(buttons);
+
+ modal.showHtml(div);
+
+ modal.div.setAttribute("tabindex", "-1");
+ modal.div.addEventListener("keydown", function(e) {
+ switch (e.keyCode) {
+ case 27:
+ modal.close();
+ break;
+ case 13:
+ if (f.validate()) {
+ modal.close(f.serialize());
+ }
+ // modal.closeSerialized();
+ break;
+ }
+ });
+
+ if (modal.focus) {
+ var toBeFocused = null;
+ if (typeof modal.focus === "object") {
+ var id = modal.focus.id;
+ var value = modal.focus.value;
+ toBeFocused = f.findViewById(id, value);
+ }
+ else if (typeof modal.focus === "string") {
+ toBeFocused = f.findViewById(modal.focus)
+ }
+ if (toBeFocused) {
+ toBeFocused.focus();
+ }
+ }
+ else {
+ var first = f.firstFocusable();
+ if (modal.autoFocus && first) {
+ first.focus();
+ }
+ else {
+ modal.div.focus();
+ }
+ }
+
+ });
+ };
+
+ DayPilot.Modal.close = function(result) {
+ var opener = DayPilot.Modal.opener();
+ if (!opener) {
+ return;
+ }
+ opener.close(result);
+ };
+
+ DayPilot.Modal.stretch = function(result) {
+ var opener = DayPilot.Modal.opener();
+ if (!opener) {
+ throw "Unable to find the opener DayPilot.Modal instance.";
+ }
+ opener.stretch();
+ };
+
+ DayPilot.Modal.closeSerialized = function() {
+ var last = DayPilot.Modal.opener() || DayPilot.ModalStatic.last();
+ if (last) {
+ last.closeSerialized();
+ }
+ };
+
+ DayPilot.Modal.opener = function() {
+ if (typeof DayPilot !== "undefined" && typeof DayPilot.ModalStatic !== "undefined" && DayPilot.ModalStatic.list.length > 0) {
+ return DayPilot.ModalStatic.list[DayPilot.ModalStatic.list.length - 1];
+ }
+ return parent && parent.DayPilot && parent.DayPilot.ModalStatic && parent.DayPilot.ModalStatic.list[parent.DayPilot.ModalStatic.list.length - 1];
+ };
+
+ DayPilot.Modal.Experimental = {};
+ DayPilot.Modal.Experimental.Form = Form;
+
+ if (typeof DayPilot.getPromise === "undefined") {
+ DayPilot.getPromise = function(f) {
+ if (typeof Promise !== 'undefined') {
+ return new Promise(f);
+ }
+
+ DayPilot.Promise = function(f) {
+ var p = this;
+
+ this.then = function(onFulfilled, onRejected) {
+ onFulfilled = onFulfilled || function() {};
+ onRejected = onRejected || function() {};
+ f(onFulfilled, onRejected);
+ return DayPilot.getPromise(f);
+ };
+
+ this['catch'] = function(onRejected) {
+ p.then(null, onRejected);
+ return DayPilot.getPromise(f);
+ };
+ };
+
+ return new DayPilot.Promise(f);
+
+ };
+ }
+
+ // end of modal.js
+
+ // form.js -> 2877
+
+ var Form = function (options) {
+
+ // properties
+ this.form = [];
+ this.data = {};
+ this.theme = "form_default";
+ this.zIndex = 99999;
+ this.locale = "en-us";
+ this.plugins = {};
+
+ // events
+ this.onKey = null;
+
+ // state
+ this._rows = [];
+ this._newRows = null;
+ this.canceling = false;
+
+ this._validationTimeouts = [];
+
+ // view
+ this._views = [];
+ this._div = null;
+
+ options = options || {};
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+ };
+
+ Form.prototype.create = function () {
+ this.load();
+ this.render();
+
+ return this._div;
+ };
+
+ Form.prototype.render = function () {
+ var form = this;
+ this._div = document.createElement("div");
+ this._rows.forEach(function (row) {
+ form.createView(row);
+ });
+ this.applyState();
+ };
+
+ Form.prototype.createView = function (row) {
+ var theme = this.theme;
+ var form = this;
+
+ var div = document.createElement("div");
+ div.className = theme + "_form_item " + theme + "_form_item_level" + row.level;
+ if (!row.interactive && row.type === "title") {
+ /*
+ if (row.type === "title") {
+ div.className += " " + theme + "_form_title";
+ }
+ */
+ div.className += " " + theme + "_form_title";
+ }
+ else {
+ div.className += " " + theme + "_form_item_" + row.type;
+ }
+ if (row.data.cssClass) {
+ div.className += " " + row.data.cssClass;
+ }
+
+ if (!row.isValue) {
+ var label = document.createElement("div");
+ label.className = theme + "_form_item_label";
+ label.innerText = row.text;
+ div.appendChild(label);
+ }
+
+ var interactive = this.createInteractive(row);
+ interactive.onInput = function(options) {
+ options = options || {};
+ form._validateInteractive(interactive, {
+ "debounce": !options.immediate
+ });
+ if (typeof form.onChange === "function") {
+ var args = {};
+ args.result = form.serialize();
+ form.onChange(args);
+ }
+ };
+ interactive.onBlur = function() {
+ if (!form.canceling) {
+ form._validateInteractive(interactive);
+ }
+ };
+ interactive.apply(row);
+ interactive._div = div;
+ interactive.row = row;
+
+ if (interactive.element) {
+ div.appendChild(interactive.element);
+ }
+
+ this._views.push(interactive);
+
+ this._div.appendChild(div);
+
+ };
+
+ Form.prototype.validate = function() {
+ var form = this;
+ var valid = true;
+ this._views.forEach(function(interactive) {
+ var iv = form._validateInteractive(interactive);
+ valid = valid && iv;
+ });
+ return valid;
+ };
+
+ Form.prototype._validateInteractive = function(interactive, options) {
+ options = options || {};
+ var debounce = options.debounce;
+ var silent = options.silent;
+
+
+ var row = interactive.row;
+ var valid = true;
+
+ var onValidate = typeof row.data.onValidate === "function" ? row.data.onValidate : null;
+ var validate = typeof row.data.validate === "function" ? row.data.validate : null; // legacy
+
+ var validateHandler = onValidate || validate;
+
+ if (validateHandler) {
+
+ var args = {};
+ args.valid = true;
+ args.value = interactive.save()[row.field];
+ args.message = "Error";
+ args.values = this.serialize(); // legacy
+ args.result = this.serialize();
+
+ validateHandler(args);
+
+ var cssClassInvalid = this.theme + "_form_item_invalid";
+ var cssClassMessage = this.theme + "_form_item_invalid_message";
+ if (args.valid) {
+ clearTimeout(this._validationTimeouts[row.field]);
+
+ if (interactive._errorMsg) {
+ interactive._errorMsg.remove();
+ interactive._errorMsg = null;
+ }
+ interactive._div.classList.remove(cssClassInvalid);
+ }
+ else {
+
+ function showInvalid() {
+ if (interactive._errorMsg) {
+ interactive._errorMsg.remove();
+ interactive._errorMsg = null;
+ }
+
+ interactive._div.classList.add(cssClassInvalid);
+ var msg = document.createElement("div");
+ msg.classList.add(cssClassMessage);
+ msg.innerText = args.message;
+
+ interactive._errorMsg = msg;
+
+ interactive._div.appendChild(msg);
+ }
+
+ if (!silent) {
+ if (debounce) {
+
+ var debounceDelay = 1000;
+
+ clearTimeout(this._validationTimeouts[row.field]);
+
+ this._validationTimeouts[row.field] = setTimeout(function() {
+ showInvalid();
+ }, debounceDelay);
+ }
+ else {
+ showInvalid();
+ }
+
+ }
+
+ }
+ valid = args.valid;
+ }
+ return valid;
+ };
+
+ Form.prototype.load = function () {
+ // transform this.form + this.data into state (_rows)
+ var t = this;
+ this.form.forEach(function (item) {
+ t.processFormItem(item, 0);
+ });
+
+ var flat;
+ // sanity check (especially for circular references)
+ try {
+ var stringified = JSON.stringify(this.data);
+ var rebuilt = JSON.parse(stringified);
+ flat = flatten(rebuilt);
+ }
+ catch (e) {
+ throw new Error("The 'data' object is not serializable (it may contain circular dependencies): " + e);
+ }
+
+ // set values
+ for (var name in flat) {
+ this.setValue(name, flat[name]);
+ }
+
+ // set state depending on values
+ // this.updateDependentState();
+ };
+
+ Form.prototype.setValue = function (name, value) {
+ this._rows.forEach(function (row) {
+ row.applyValue(name, value);
+ });
+ };
+
+ Form.prototype.updateDependentState = function () {
+ var form = this;
+ var enabled = [true];
+
+ var source = this._newRows ? this._newRows : this._rows;
+
+ source.forEach(function (row) {
+
+ var updatedRow = form.updateState(row, {
+ enabled: enabled[row.level] && !row.data.disabled
+ });
+
+ if (updatedRow.isValue) {
+ enabled[updatedRow.level + 1] = updatedRow.enabled && updatedRow.checked;
+ }
+ });
+ };
+
+ Form.prototype.processFormItem = function (item, level) {
+ var form = this;
+ var type = this.getFieldType(item);
+
+ var rows = [];
+
+ if (type === "radio") {
+
+ if (item.name) {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = "label";
+ row.interactive = false;
+ row.text = item.name;
+ form._rows.push(row);
+ rows.push(row);
+ }
+
+ item.options.forEach(function (option) {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = option;
+ row.level = level;
+ row.type = type;
+ row.isValue = true;
+ row.text = option.name;
+ row.resolved = option.id;
+ form._rows.push(row);
+ rows.push(row);
+
+ if (option.children) {
+ option.children.forEach(function (child) {
+ var childRows = form.processFormItem(child, level + 1);
+ rows = rows.concat(childRows);
+ })
+ }
+ });
+ }
+ else if (type === "title") {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = item.name;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else if (type === "image") {
+ var row = new RowModel();
+ row.isValue = true;
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = null;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else if (type === "html") {
+ var row = new RowModel();
+ row.isValue = true;
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = null;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else if (type === "scrollable") {
+ var row = new RowModel();
+ row.isValue = true;
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.interactive = false;
+ row.text = null;
+ form._rows.push(row);
+ rows.push(row);
+ }
+ else {
+ var row = new RowModel();
+ row.field = item.id;
+ row.data = item;
+ row.level = level;
+ row.type = type;
+ row.text = item.name;
+ row.children = [];
+ form._rows.push(row);
+ rows.push(row);
+ }
+
+ if (type === "checkbox") {
+ row.isValue = true;
+ row.resolved = true;
+
+ if (item.children) {
+ item.children.forEach(function (child) {
+ var childRows = form.processFormItem(child, level + 1);
+ rows = rows.concat(childRows);
+ })
+ }
+ }
+
+ return rows;
+
+ };
+
+ Form.prototype.doOnKey = function (key) {
+ if (typeof this.onKey === "function") {
+ var args = {
+ key: key
+ };
+ this.onKey(args);
+ }
+ };
+
+ Form.prototype.createInteractive = function (row) {
+ var form = this;
+ var views = {
+ "label": function () {
+ return new Interactive();
+ },
+ "title": function () {
+ return new Interactive();
+ },
+ "image": function() {
+ var interactive = new Interactive();
+
+ var image = document.createElement("img");
+ image.src = row.data.image;
+
+ interactive.element = image;
+
+ return interactive;
+ },
+ "html": function() {
+ var interactive = new Interactive();
+
+ var div = document.createElement("div");
+ if (typeof row.data.text === "string") {
+ div.innerText = row.data.text;
+ }
+ else if (typeof row.data.html === "string") {
+ div.innerHTML = row.data.html;
+ }
+
+ interactive.element = div;
+
+ return interactive;
+ },
+ "scrollable": function() {
+ var interactive = new Interactive();
+
+ var scroll = document.createElement("div");
+ scroll.className = form.theme + "_form_item_scrollable_scroll";
+
+ if (row.data.height) {
+ scroll.style.height = row.data.height + "px";
+ }
+
+ var div = document.createElement("div");
+ div.className = form.theme + "_form_item_scrollable_scroll_content";
+ if (typeof row.data.text === "string") {
+ div.innerText = row.data.text;
+ }
+ else if (typeof row.data.html === "string") {
+ div.innerHTML = row.data.html;
+ }
+
+ scroll.appendChild(div);
+
+ interactive.element = scroll;
+
+ return interactive;
+ },
+ "text": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+
+ var input = interactive.element;
+ input.value = row.value;
+ input.disabled = !row.enabled;
+ };
+
+ var input = document.createElement("input");
+ input.name = row.field;
+ input.type = "text";
+ input.autocomplete = "off";
+ input.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13:
+ form.doOnKey("Enter");
+ break;
+ case 27:
+ form.doOnKey("Escape");
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ input.oninput = function(e) {
+ interactive.onInput();
+ };
+ input.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.element = input;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ interactive.element.setSelectionRange(0, interactive.element.value.length);
+ };
+ interactive.save = function() {
+ var result = {};
+ result[row.field] = input.value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "textarea": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+
+ var input = interactive.element;
+ input.value = row.value;
+ input.disabled = !row.enabled;
+ };
+
+ var textarea = document.createElement("textarea");
+ textarea.name = row.field;
+ if (row.data.height) {
+ textarea.style.height = row.data.height + "px";
+ }
+ // input.type = "text";
+ textarea.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13:
+ // form.doOnKey("Enter");
+ if (e.ctrlKey || e.metaKey) {
+ form.doOnKey("Enter");
+ }
+ letcontinue = false;
+ break;
+ case 27:
+ form.doOnKey("Escape");
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ // e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ textarea.oninput = function(e) {
+ interactive.onInput();
+ };
+ textarea.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.element = textarea;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ interactive.element.setSelectionRange(0, 0);
+ };
+ interactive.save = function() {
+ var result = {};
+ result[row.field] = textarea.value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "date": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var input = interactive.element;
+ var picker = interactive.picker;
+
+ if (row.data.dateFormat) {
+ picker.pattern = row.data.dateFormat;
+ }
+ var locale = row.data.locale || form.locale;
+ if (locale) {
+ picker.locale = locale;
+ }
+
+ input.disabled = !row.enabled;
+
+ picker.date = new DayPilot.Date(row.value);
+ var formatted = new DayPilot.Date(row.value).toString(row.data.dateFormat || picker.pattern, picker.locale);
+ input.value = formatted;
+ };
+
+ var input = document.createElement("input");
+ input.name = row.field;
+
+ var picker = new DayPilot.DatePicker({
+ target: input,
+ theme: "navigator_modal",
+ zIndex: form.zIndex + 1,
+ resetTarget: false,
+ targetAlignment: "left",
+ onTimeRangeSelect: function(args) {
+ interactive.onInput({"immediate": true});
+ }
+ });
+
+ // required for serialization - to get the normalized value
+ input.picker = picker;
+ input.className = form.theme + "_input_date";
+ input.type = "text";
+ input.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13: // enter
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Enter");
+ }
+ break;
+ case 27: // escape
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Escape");
+ }
+ break;
+ case 9: // tab
+ picker.close();
+ letcontinue = true;
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ input.onfocus = function () {
+ picker.show();
+ };
+
+ input.onclick = function () {
+ picker.show();
+ };
+
+ /*
+ input.onblur = function () {
+ // input.picker.close();
+ };
+ */
+
+ input.oninput = function(e) {
+ interactive.onInput();
+ };
+ input.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.element = input;
+ interactive.picker = picker;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ };
+ interactive.save = function() {
+ var value = picker.date ? picker.date.toString() : null;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+ return interactive;
+ },
+ "time": function () {
+ return form._createInteractiveTime(row);
+ },
+ "datetime": function() {
+ return form._createInteractiveDateTime(row);
+ },
+ "select": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var select = interactive.element;
+
+ select.value = row.value;
+ select.disabled = !row.enabled;
+ };
+
+ var select = document.createElement("select");
+ select.name = row.field;
+
+ if (row.data.options && row.data.options.forEach) {
+ row.data.options.forEach(function (i) {
+ var option = document.createElement("option");
+ option.innerText = i.name || i.id;
+ option.value = i.id;
+ option._originalValue = i.id;
+ select.appendChild(option);
+ });
+ }
+
+ select.onchange = function(e) {
+ interactive.onInput({"immediate": true});
+ };
+ select.onblur = function(e) {
+ interactive.onBlur();
+ };
+
+ interactive.element = select;
+ interactive.canFocus = function() {
+ return !interactive.element.disabled;
+ };
+ interactive.focus = function () {
+ interactive.element.focus();
+ };
+ interactive.save = function() {
+ var value = null;
+ var option = select.options[select.selectedIndex];
+ if (option && typeof option._originalValue !== "undefined") {
+ value = option._originalValue;
+ }
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+
+ },
+ "searchable": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var searchable = interactive.searchable;
+
+ searchable.disabled = !row.enabled;
+ searchable.select(row.value);
+ };
+
+ var searchable = new Searchable({
+ data: row.data.options || [],
+ name: row.field,
+ theme: form.theme + "_form_item_searchable",
+ listZIndex: form.zIndex + 1,
+ // disabled: !row.enabled
+ onSelect: function(args) {
+ if (args.ui) {
+ interactive.onInput({"immediate": true});
+ }
+ }
+ });
+
+ var element = searchable.create();
+
+ interactive.element = element;
+ interactive.searchable = searchable;
+ interactive.canFocus = function() {
+ return !interactive.searchable.disabled;
+ };
+ interactive.focus = function () {
+ interactive.searchable.focus();
+ };
+ interactive.save = function() {
+ var value = searchable.selected && searchable.selected.id;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "radio": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var radio = interactive.radio;
+
+ radio.checked = row.checked;
+ radio.disabled = !row.enabled;
+ };
+
+ var label = document.createElement("label");
+
+ var radio = document.createElement("input");
+ radio.type = "radio";
+ radio.name = row.field;
+ radio._originalValue = row.resolved;
+
+ radio.onchange = function (ev) {
+ // activation only
+
+ var row = interactive.row;
+ form.findRowsByField(row.field).forEach(function (row) {
+ form.updateState(row, {
+ checked: false
+ });
+ });
+ form.updateState(row, {
+ checked: true
+ });
+
+ form.applyState();
+
+ interactive.onInput({"immediate": true});
+ };
+
+ radio.onblur = function(e) {
+ interactive.onBlur();
+ };
+
+ label.appendChild(radio);
+
+ var text = document.createTextNode(row.text);
+ label.append(text);
+
+ interactive.element = label;
+ interactive.radio = radio;
+ interactive.canFocus = function() {
+ return false;
+ };
+ interactive.focus = function () {
+ interactive.radio.focus();
+ }
+ interactive.save = function() {
+ if (!radio.checked) {
+ return {};
+ }
+ var value = radio._originalValue;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "checkbox": function () {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var checkbox = interactive.checkbox;
+
+ checkbox.checked = row.checked;
+ checkbox.disabled = !row.enabled;
+ };
+
+ var label = document.createElement("label");
+
+ var checkbox = document.createElement("input");
+ checkbox.type = "checkbox";
+ checkbox.name = row.field;
+ checkbox._originalValue = row.resolved;
+
+ checkbox.onchange = function (ev) {
+ var row = interactive.row;
+ form.updateState(row, {
+ checked: this.checked
+ });
+ form.applyState();
+
+ interactive.onInput({"immediate": true});
+ };
+
+ checkbox.onblur = function(e) {
+ interactive.onBlur();
+ };
+
+ label.appendChild(checkbox);
+
+ var text = document.createTextNode(row.text);
+ label.append(text);
+
+ interactive.element = label;
+ interactive.checkbox = checkbox;
+ interactive.canFocus = function() {
+ return false;
+ };
+ interactive.focus = function () {
+ interactive.checkbox.focus();
+ }
+ interactive.save = function() {
+ var value = checkbox.checked;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ },
+ "table": function() {
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var table = interactive.table;
+
+ table.disabled = !row.enabled;
+ table.load(row.value || []);
+
+ /*
+ if (row.value) {
+ table.load(row.value);
+ }
+ */
+ };
+
+ var table = new Table({
+ name: row.field,
+ // form: row.data.columns,
+ form: form,
+ theme: form.theme + "_form_item_tabular",
+ item: row.data,
+ onInput: function(args) {
+ interactive.onInput();
+ }
+ });
+
+ var element = table.create();
+
+ interactive.element = element;
+ interactive.table = table;
+ interactive.canFocus = function() {
+ // return !interactive.table.disabled;
+ return false;
+ };
+ interactive.focus = function () {
+ interactive.table.focus();
+ };
+ interactive.save = function() {
+ var value = table.save();
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+ }
+ };
+ if (form.plugins && form.plugins[row.type]) {
+ return form.plugins[row.type](row);
+ }
+ return views[row.type]();
+ };
+
+ Form.prototype._createInteractiveTime = function(row) {
+ var form = this;
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var searchable = interactive.searchable;
+
+ searchable.disabled = !row.enabled;
+ searchable.select(row.value);
+ };
+
+ var data = [];
+
+ var interval = row.data.timeInterval || 15; // allowed values: 1, 5, 10, 15, 20, 30, 60
+ var allowedIntervals = [1, 5, 10, 15, 20, 30, 60];
+ if (!allowedIntervals.includes(interval)) {
+ interval = 15;
+ }
+ var perHour = 60 / interval;
+
+ var localeStr = row.data.locale || form.locale;
+ var locale = DayPilot.Locale.find(localeStr) || DayPilot.Locale.US;
+
+ var date = DayPilot.Date.today();
+
+ for (var i = 0; i < 24*perHour; i++) {
+ var time = date.addMinutes(interval*i);
+ var item = {};
+ item.name = time.toString(row.data.timeFormat || locale.timePattern, locale);
+ item.id = time.toString("HH:mm");
+ data.push(item);
+ }
+
+ var searchable = new Searchable({
+ data: data,
+ name: row.field,
+ theme: form.theme + "_form_item_time",
+ listZIndex: form.zIndex + 1,
+ strategy: "startsWith",
+ // disabled: !row.enabled
+ onSelect: function(args) {
+ if (args.ui) {
+ interactive.onInput({"immediate": true});
+ }
+ }
+ });
+
+ var element = searchable.create();
+
+ interactive.element = element;
+ interactive.searchable = searchable;
+ interactive.canFocus = function() {
+ return !interactive.searchable.disabled;
+ };
+ interactive.focus = function () {
+ interactive.searchable.focus();
+ };
+ interactive.save = function() {
+ var value = searchable.selected && searchable.selected.id;
+ var result = {};
+ result[row.field] = value;
+ return result;
+ };
+
+ return interactive;
+
+ };
+
+ Form.prototype._createInteractiveDateTime = function(row) {
+ var form = this;
+ var interactive = new Interactive();
+ interactive.apply = function (row) {
+ interactive.row = row;
+ var searchable = interactive.searchable;
+
+ searchable.disabled = !row.enabled;
+
+ var timePart = new DayPilot.Date(row.value).toString("HH:mm");
+
+ searchable.select(timePart);
+
+ var input = interactive.dateInput;
+ var picker = interactive.picker;
+
+ if (row.data.dateFormat) {
+ picker.pattern = row.data.dateFormat;
+ }
+ var localeStr = row.data.locale || form.locale;
+ if (localeStr) {
+ var locale = DayPilot.Locale.find(localeStr) || DayPilot.Locale.US;
+ picker.locale = localeStr;
+ picker.pattern = locale.datePattern;
+ }
+
+ input.disabled = !row.enabled;
+
+ picker.date = new DayPilot.Date(row.value);
+ var formatted = new DayPilot.Date(row.value).toString(row.data.dateFormat || picker.pattern, picker.locale);
+ input.value = formatted;
+
+ };
+
+ var dateElement = (function createDatePicker() {
+ var input = document.createElement("input");
+ input.name = row.field;
+
+ var picker = new DayPilot.DatePicker({
+ target: input,
+ theme: "navigator_modal",
+ zIndex: form.zIndex + 1,
+ resetTarget: false,
+ targetAlignment: "left",
+ onTimeRangeSelect: function(args) {
+ interactive.onInput({"immediate": true});
+ }
+ });
+
+ // required for serialization - to get the normalized value
+ input.picker = picker;
+ input.className = form.theme + "_input_date";
+ input.type = "text";
+ input.onkeydown = function (e) {
+ var letcontinue = false;
+ switch (e.keyCode) {
+ case 13: // enter
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Enter");
+ }
+ break;
+ case 27: // escape
+ if (picker.visible) {
+ picker.close();
+ } else {
+ form.doOnKey("Escape");
+ }
+ break;
+ case 9: // tab
+ picker.close();
+ letcontinue = true;
+ break;
+ default:
+ letcontinue = true;
+ break;
+ }
+ if (!letcontinue) {
+ e.preventDefault();
+ e.stopPropagation();
+ }
+ };
+
+ input.onfocus = function () {
+ picker.show();
+ };
+
+ input.onclick = function () {
+ picker.show();
+ };
+
+ input.oninput = function(e) {
+ interactive.onInput();
+ };
+ input.onblur = function(e) {
+ interactive.onBlur();
+ }
+
+ interactive.dateInput = input;
+ interactive.picker = picker;
+
+ return input;
+ })();
+
+
+ var timeElement = (function createTimePicker() {
+ var data = [];
+
+ var interval = row.data.timeInterval || 15; // allowed values: 1, 5, 10, 15, 20, 30, 60
+ var allowedIntervals = [1, 5, 10, 15, 20, 30, 60];
+ if (!allowedIntervals.includes(interval)) {
+ interval = 15;
+ }
+ var perHour = 60 / interval;
+
+ var localeStr = row.data.locale || form.locale;
+ var locale = DayPilot.Locale.find(localeStr) || DayPilot.Locale.US;
+
+ var date = DayPilot.Date.today();
+
+ for (var i = 0; i < 24*perHour; i++) {
+ var time = date.addMinutes(interval*i);
+ var item = {};
+ item.name = time.toString(row.data.timeFormat || locale.timePattern, locale);
+ item.id = time.toString("HH:mm");
+ data.push(item);
+ }
+
+ var searchable = new Searchable({
+ data: data,
+ name: row.field,
+ theme: form.theme + "_form_item_time",
+ listZIndex: form.zIndex + 1,
+ strategy: "startsWith",
+ // disabled: !row.enabled
+ onSelect: function(args) {
+ if (args.ui) {
+ interactive.onInput({"immediate": true});
+ }
+ }
+ });
+
+ interactive.searchable = searchable;
+ return searchable.create();
+
+ })();
+
+
+ var element = document.createElement("div");
+ element.className = form.theme + "_form_item_datetime_parent";
+ element.appendChild(dateElement);
+ element.appendChild(timeElement);
+
+ interactive.element = element;
+ interactive.canFocus = function() {
+ return !interactive.searchable.disabled;
+ };
+ interactive.focus = function () {
+ interactive.dateInput.focus();
+ };
+ interactive.save = function() {
+ var timeValue = interactive.searchable.selected && interactive.searchable.selected.id;
+ var dateValue = interactive.picker.date ? interactive.picker.date.toString() : null;
+
+ var date = new DayPilot.Date(dateValue).getDatePart();
+ var value = DayPilot.Date.parse(date.toString("yyyy-dd-MM ") + timeValue, "yyyy-dd-MM HH:mm");
+
+ var result = {};
+ result[row.field] = value;
+ return result;
+
+ };
+
+ return interactive;
+
+ };
+
+ Form.prototype.findRowsByField = function (field) {
+ return this._rows.filter(function (row) {
+ return row.field === field;
+ });
+ };
+
+ // ooptional value param applies to radio which can have multiple views, one for each value
+ Form.prototype.findViewById = function (id, value) {
+ return this._views.find(function (v) {
+ if (v.row.field === id) {
+ if (v.row.type === "radio") {
+ return v.row.resolved === value;
+ } else {
+ return true;
+ }
+ }
+ return false;
+ });
+ };
+
+ Form.prototype.firstFocusable = function () {
+ return this._views.find(function (v) {
+ return v.canFocus && v.canFocus();
+ });
+ };
+
+ Form.prototype.updateState = function (row, props) {
+ var source = this._newRows ? this._newRows : this._rows;
+ var index = source.indexOf(row);
+ this._newRows = source.map(function (srow) {
+ if (srow !== row) {
+ return srow;
+ }
+ // our row
+ if (row.propsEqual(props)) {
+ return row;
+ }
+ var cloned = row.clone();
+ for (var name in props) {
+ cloned[name] = props[name];
+ }
+ return cloned;
+ });
+ return this._newRows[index];
+ };
+
+ // dirty row received, a member of _newRows
+ Form.prototype.updateInteractive = function (row) {
+ var index = this._newRows.indexOf(row);
+ this._views[index].apply(row);
+ };
+
+ Form.prototype.applyState = function () {
+ var form = this;
+
+ this.updateDependentState();
+
+ if (!this._newRows) {
+ return;
+ }
+
+ var dirtyRows = this._newRows.filter(function (row, i) {
+ return form._rows[i] !== row;
+ });
+
+ dirtyRows.forEach(function (row) {
+ form.updateInteractive(row);
+ });
+
+ this._rows = this._newRows;
+
+ this._newRows = null;
+ };
+
+ Form.prototype.getFieldType = function (item) {
+
+ var known = ["text", "date", "select", "searchable", "radio", "checkbox", "table", "title", "image", "html", "textarea", "scrollable", "time", "datetime"];
+ if (known.indexOf(item.type) !== -1) {
+ return item.type;
+ }
+
+ if (item.type && this.plugins && this.plugins[item.type]) {
+ return item.type;
+ }
+
+ if (item.image) {
+ return "image";
+ }
+
+ if (item.html || item.text) {
+ return "html";
+ }
+
+ if (!item.id) {
+ return "title";
+ }
+
+ if (item.options) {
+ return "searchable";
+ }
+
+ if (item.dateFormat) {
+ return "date";
+ }
+
+ if (item.columns) {
+ return "table";
+ }
+
+ return "text";
+ };
+
+ Form.prototype.serialize = function() {
+ var result = {};
+
+ this._views.forEach(function(interactive) {
+ var out = interactive.save();
+ for (var name in out) {
+ result[name] = out[name];
+ }
+ });
+
+ return result;
+ };
+
+ var RowModel = function () {
+ this.id = this.guid();
+ this.field = null;
+ this.data = null;
+ this.type = null;
+ this.level = 0;
+ this.enabled = true;
+ this.value = null;
+ this.text = null;
+
+ this.interactive = true;
+
+ this.isValue = false;
+ this.checked = false;
+ this.resolved = null; // value resolves to
+ };
+
+ RowModel.prototype.clone = function () {
+ var rm = new RowModel();
+ for (var name in this) {
+ if (name === "id") {
+ continue;
+ }
+ rm[name] = this[name];
+ }
+ return rm;
+ };
+
+ RowModel.prototype.propsEqual = function (props) {
+ for (var name in props) {
+ if (this[name] !== props[name]) {
+ return false;
+ }
+ }
+ return true;
+ };
+
+ RowModel.prototype.guid = function () {
+ var S4 = function () {
+ return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
+ };
+ return ("" + S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4());
+ };
+
+ RowModel.prototype.applyValue = function (name, value) {
+ if (this.field !== name) {
+ return;
+ }
+ this.value = value;
+ if (this.isValue && value === this.resolved) {
+ this.checked = true;
+ }
+ };
+
+ var Interactive = function () {
+ // this.row = row;
+ this.element = null;
+
+ // only used for autofocus
+ this.canFocus = function() {
+ return false;
+ };
+ this.apply = function (row) {
+ };
+ this.focus = function () {
+ };
+ this.save = function() {
+ return {};
+ }
+ };
+
+ function flatten(object, result, prefix) {
+ result = result || {};
+ prefix = prefix || "";
+ for (var name in object) {
+ var src = object[name];
+ if (typeof src === "object") {
+ if (Object.prototype.toString.call(src) === '[object Array]') {
+ result[prefix + name] = src;
+ }
+ else if (src && src.toJSON) {
+ result[prefix + name] = src.toJSON();
+ }
+ else {
+ flatten(src, result, prefix + name + ".");
+ }
+ }
+ else {
+ result[prefix + name] = src;
+ }
+ }
+ return result;
+ }
+
+ // end of form.js
+
+ // searchable.js -> 3265
+ var Searchable = function(options) {
+
+ // properties
+ this.data = [];
+ this.name = null;
+ this.theme = "searchable_default";
+ this._disabled = false;
+ this.listZIndex = 100000;
+
+
+ // events
+ this.onSelect = null;
+
+ // state
+ this._selected = null;
+ this._highlighted = null;
+ this._collapsed = false;
+ // this._focused = false;
+ // this._filter = null;
+
+ // view
+ this._input = null;
+ this._list = null;
+ this._options = [];
+ this._hidden = null;
+
+ options = options || {};
+
+ var t = this;
+
+ var specialHandling = {
+ "selected": {
+ post: function(val) {
+ if (typeof val === "object" && val.id) {
+ t._selected = val;
+ }
+ else if (typeof val === "string" || typeof val === "number") {
+ t.select(val);
+ }
+ }
+ }
+ };
+
+ Object.defineProperty(this, "selected", {
+ get: function() {
+ return this._selected;
+ },
+ });
+
+ Object.defineProperty(this, "disabled", {
+ get: function() {
+ return this._disabled;
+ },
+ set: function(val) {
+ this._disabled = val;
+ if (this._input) {
+ this._input.disabled = val;
+ if (val) {
+ // verify
+ this._cancel();
+ }
+ }
+ }
+ });
+
+ for (var name in options) {
+ if (!specialHandling[name]) {
+ this[name] = options[name];
+ }
+ }
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ specialHandling[name].post(options[name]);
+ }
+ }
+
+ };
+
+ Searchable.prototype.select = function(id) {
+ this._selected = this.data.find(function(item) { return item.id === id });
+ this._doOnSelect(false);
+ return this;
+ };
+
+
+ Searchable.prototype.create = function() {
+ var component = this;
+ var t = this;
+
+ var div = document.createElement("div");
+ div.className = this.theme + "_main";
+ div.style.position = "relative";
+
+ var icon = document.createElement("div");
+ icon.className = this.theme + "_icon";
+ icon.style.position = "absolute";
+ icon.style.right = "0";
+ icon.style.top = "0";
+ icon.style.bottom = "0";
+ icon.style.width = "20px";
+ icon.addEventListener("mousedown", function(ev) {
+ ev.preventDefault();
+ if (component._collapsed) {
+ component.focus();
+ expand();
+ }
+ else {
+ cancel();
+ collapse();
+ }
+ });
+
+ var list = document.createElement("div");
+ list.className = this.theme + "_list";
+ list.style.display = "none";
+ list.style.position = "absolute";
+ list.style.zIndex = this.listZIndex;
+
+ var hidden = document.createElement("input");
+ hidden.type = "hidden";
+ hidden.name = this.name;
+ hidden.searchable = t;
+ this._hidden = hidden;
+
+ var input = document.createElement("input");
+ input.type = "text";
+ input.className = this.theme + "_input";
+ input.disabled = this._disabled;
+ input.addEventListener("click", function(ev) {
+ // console.log("click");
+ expand();
+ });
+ input.addEventListener("focus", function(ev) {
+ // do not call expand() here, the readonly status has to stay
+ // input.value = "";
+ // console.log("focus");
+ filter("all");
+ });
+ input.addEventListener("input", function(ev) {
+ filter();
+ });
+ input.addEventListener("blur", function(ev) {
+ input.removeAttribute("readonly");
+ cancel();
+ });
+
+ input.addEventListener("keydown", function(ev) {
+ if (component._collapsed) {
+ if (ev.key === "Enter") {
+ return;
+ }
+ if (ev.key === "Esc" || ev.key === "Escape") {
+ return;
+ }
+ expand();
+ }
+
+ if (ev.key === "ArrowDown") {
+ var index = t._options.indexOf(t._highlighted);
+ if (index + 1 < t._options.length) {
+ t._highlighted = t._options[index + 1];
+ }
+ updateHiglight();
+ } else if (ev.key === "ArrowUp") {
+ var index = t._options.indexOf(t._highlighted);
+ if (index - 1 >= 0) {
+ t._highlighted = t._options[index - 1];
+ }
+ updateHiglight();
+ } else if (ev.key === "Enter") {
+ if (component._highlighted) {
+ ev.stopPropagation();
+ selectOption(component._highlighted);
+ }
+ else {
+ ev.stopPropagation();
+ // t._cancel();
+ cancel();
+ collapse();
+ }
+ } else if (ev.key === "Esc" || ev.key === "Escape") {
+ ev.stopPropagation();
+ cancel();
+ collapse();
+ }
+ });
+ this._input = input;
+ this._list = list;
+
+ if (!this._selected) {
+ this._selected = this.data[0];
+ if (this._selected) {
+ input.value = this._selected.name;
+ }
+ }
+
+ function filter(strategy) {
+ // component._collapsed = false;
+
+ var defaultStrategy = component.strategy;
+ if (component.strategy !== "includes" && component.strategy !== "startsWith") {
+ defaultStrategy = "includes";
+ }
+
+ strategy = strategy || defaultStrategy || "includes";
+
+ list.style.display = "";
+ list.style.top = input.offsetHeight + "px";
+ list.style.left = "0px";
+ list.style.width = input.offsetWidth + "px";
+ list.innerHTML = "";
+
+ // allow scrollbar access
+ list.addEventListener("mousedown", function(ev) {
+ ev.preventDefault();
+ });
+
+ component._highlighted = null;
+
+ component._options = [];
+
+ var first = null;
+
+ component.data.forEach(function(item) {
+
+ var name = item.name || item.id;
+
+ if (strategy === "includes") {
+ if (name.toLowerCase().indexOf(input.value.toLowerCase()) === -1) {
+ return;
+ }
+ }
+ else if (strategy === "startsWith") {
+ if (name.toLowerCase().indexOf(input.value.toLowerCase()) !== 0) {
+ return;
+ }
+ }
+ else if (strategy === "all") {
+ // don't skip
+ }
+
+
+ var option = document.createElement("div");
+ option.className = component.theme + "_list_item";
+ option.innerText = name;
+
+ option.item = item;
+
+ if (item === component._selected) {
+ component._highlighted = option;
+ }
+ if (!first) {
+ first = option;
+ }
+
+ // "mousedown" goes before blur, "click" goes after
+ option.addEventListener("mousedown", function(ev) {
+ selectOption(option);
+ ev.preventDefault();
+ });
+
+ option.addEventListener("mousemove", function(ev) {
+ if (component._highlighted === option) {
+ return;
+ }
+ component._highlighted = option;
+ updateHiglight({dontScroll: true});
+ });
+
+ list.appendChild(option);
+
+ component._options.push(option);
+ });
+
+ if (!component._highlighted) {
+ component._highlighted = first;
+ }
+ updateHiglight();
+ }
+
+ function updateHiglight(options) {
+ options = options || {};
+ var scrollIntoView = !options.dontScroll;
+
+ // redo, avoid selectors
+ var previous = document.querySelectorAll("." + component.theme + "_list_item_highlight");
+ previous.forEach(function(p) {
+ p.className = p.className.replace(component.theme + "_list_item_highlight", "");
+ });
+
+ if (component._highlighted) {
+ component._highlighted.className += " " + component.theme + "_list_item_highlight";
+
+ if (scrollIntoView && !isScrolledIntoView(component._highlighted, list)) {
+ component._highlighted.scrollIntoView();
+ }
+ }
+ }
+
+ function isScrolledIntoView(target, viewport) {
+ var tRect = target.getBoundingClientRect();
+ var vRect = viewport.getBoundingClientRect();
+ return tRect.top >= vRect.top && tRect.bottom <= vRect.bottom;
+ }
+
+ function selectOption(option) {
+ var item = option.item;
+
+ // input.value = option.innerText;
+ component._selected = item;
+ component._doOnSelect(true);
+ hide();
+ collapse();
+ }
+
+ function cancel() {
+ component._cancel();
+ }
+
+ function hide() {
+ component._hide();
+ }
+
+ function collapse() {
+ component._collapsed = true;
+ input.setAttribute("readonly", "readonly");
+ input.focus();
+ }
+
+ function expand() {
+ component._collapsed = false;
+ input.removeAttribute("readonly");
+ // input.value = "";
+ input.select();
+ filter("all");
+ }
+
+ div.appendChild(input);
+ div.appendChild(icon);
+ div.appendChild(hidden);
+ div.appendChild(list);
+ return div;
+ };
+
+ Searchable.prototype._cancel = function() {
+ this._hide();
+ if (!this._selected) {
+ this._input.value = "";
+ // not sure about this:
+ this._doOnSelect(true);
+ } else {
+ this._input.value = this._selected.name;
+ }
+ // collapse();
+ }
+
+ Searchable.prototype.focus = function() {
+ // this._input.focus();
+
+ this._collapsed = true;
+ this._input.setAttribute("readonly", "readonly");
+ this._input.focus();
+ this._cancel();
+ };
+
+ Searchable.prototype._hide = function() {
+ this._list.style.display = "none";
+ };
+
+ Searchable.prototype._doOnSelect = function(byUser) {
+ this._hidden.value = this.selected ? this.selected.id : null;
+ if (this._selected) {
+ this._input.value = this._selected.name;
+ }
+ else {
+ this._input.value = "";
+ }
+
+ if (typeof this.onSelect === "function") {
+ var args = {
+ control: this,
+ ui: byUser
+ };
+ this.onSelect(args);
+ }
+ };
+
+ // end of searchable.js
+
+ // table.js -> 3641
+ var Table = function(options) {
+
+ this.form = null; // Form instance
+ this.item = null; // table def from form
+ this.data = null;
+ this.name = null;
+ this.theme = "edit_table_default";
+ this.onInput = null;
+
+ this.nav = {};
+
+ this._activeEdit = null;
+ this._rows = [];
+
+ options = options || {};
+
+ for (var name in options) {
+ this[name] = options[name];
+ }
+
+ };
+
+ Table.prototype.create = function() {
+ var table = this;
+
+ var div = document.createElement("div");
+ div.className = this.theme + "_main";
+ div.style.position = "relative";
+
+ var hidden = document.createElement("input");
+ hidden.type = "hidden";
+ hidden.name = table.name;
+ hidden.table = this;
+ div.appendChild(hidden);
+
+ var tableElement = document.createElement("div");
+ tableElement.className = this.theme + "_table";
+ var header = this._createHeader();
+ tableElement.appendChild(header);
+
+ var spacerRow = table._createRowState({});
+ spacerRow.spacer = true;
+ var spacer = this._renderRow(spacerRow);
+ // spacer.style.visibility = "hidden";
+ spacer.classList.add(table.theme + "_spacer");
+ tableElement.appendChild(spacer);
+
+ var body = document.createElement("div");
+ body.className = table.theme + "_tbody";
+ tableElement.appendChild(body);
+
+ div.appendChild(tableElement);
+
+ var after = document.createElement("div");
+ div.appendChild(after);
+
+ this.nav.body = body;
+ this.nav.table = tableElement;
+ this.nav.main = div;
+ this.nav.after = after;
+
+ var add = document.createElement("div");
+ var plus = document.createElement("span");
+ plus.className = this.theme + "_plus";
+
+ plus.addEventListener("click", function(ev) {
+ if (table.disabled) {
+ return;
+ }
+
+ var generate = table.item.onNewRow;
+ var value = {};
+ if (typeof generate === "function") {
+ var args = {};
+ args.result = table.form.serialize();
+ args.value = {};
+ generate(args);
+ value = args.value;
+ }
+
+ var row = table._createRowState(value);
+ table._rows.push(row);
+ table._render();
+ table._doOnInput();
+ });
+
+ this.nav.plus = plus;
+
+ add.appendChild(plus);
+ div.appendChild(add);
+
+ return div;
+ };
+
+ Table.prototype._createHeader = function() {
+ var table = this;
+ var row = document.createElement("div");
+ row.classList.add(this.theme + "_row");
+ row.classList.add(this.theme + "_header");
+ this.item.columns.forEach(function(item) {
+ var cell = document.createElement("div");
+ cell.classList.add(table.theme + "_cell");
+ cell.innerText = item.name;
+ row.appendChild(cell);
+ });
+ return row;
+ };
+
+ Table.prototype._maxRowsReached = function() {
+ var max = this.item.max || 0;
+ if (max && this._rows.length >= max) {
+ return true;
+ }
+ return false;
+ };
+
+ Table.prototype.save = function() {
+ var table = this;
+ var data = [];
+
+ table._rows.forEach(function(row) {
+ var item = {};
+ row.cells.forEach(function(cell) {
+ item[cell.id] = cell.value;
+ });
+ data.push(item);
+ });
+
+ return data;
+ };
+
+ Table.prototype.load = function(data) {
+ var table = this;
+
+ var isArray = Object.prototype.toString.call(data) === '[object Array]';
+ if (!isArray) {
+ throw new Error("Array expected");
+ }
+
+ this.data = data;
+
+ this._createState();
+ this._render();
+ };
+
+ Table.prototype._updateCss = function() {
+ if (this.disabled) {
+ this.nav.main.classList.add(this.theme + "_disabled");
+ }
+ else {
+ this.nav.main.classList.remove(this.theme + "_disabled");
+ }
+
+ var maxReached = this._maxRowsReached();
+ if (maxReached) {
+ this.nav.plus.classList.add(this.theme + "_plus_max");
+ }
+ else {
+ this.nav.plus.classList.remove(this.theme + "_plus_max");
+ }
+ };
+
+ Table.prototype._createState = function() {
+ var table = this;
+ this._rows = [];
+ this.data.forEach(function(dataRow) {
+ var row = table._createRowState(dataRow);
+ table._rows.push(row);
+ });
+ };
+
+ Table.prototype._removeRow = function(row) {
+ var table = this;
+
+ var index = table._rows.indexOf(row);
+ table._rows.splice(index, 1);
+ };
+
+ Table.prototype._createRowState = function(dataRow) {
+ var table = this;
+
+ var row = {};
+ row.data = dataRow;
+ row.cells = [];
+
+ table.item.columns.forEach(function(formItem) {
+ var id = formItem.id;
+ var value = dataRow[id];
+
+ var type = table._formItemType(formItem);
+ if (typeof value === "undefined") {
+ if (type === "text") {
+ value = "";
+ }
+ else if (type === "number") {
+ value = 0;
+ }
+ else if (type === "select") {
+ var options = formItem.options;
+ value = options && options[0].id;
+ }
+ }
+
+ var cell = {};
+ cell.id = id;
+ cell.value = value;
+ cell.type = type;
+ cell.data = formItem;
+ row.cells.push(cell);
+ });
+
+ return row;
+ };
+
+ Table.prototype._formItemType = function(formItem) {
+ var type = formItem.type;
+ if (!type) {
+ if (formItem.options) {
+ type = "select";
+ }
+ else {
+ type = "text";
+ }
+ }
+ return type;
+ };
+
+ Table.prototype._render = function() {
+
+ var table = this;
+ this.nav.body.innerHTML = "";
+ this.nav.after.innerHTML = "";
+
+ this._rows.forEach(function(row) {
+ var el = table._renderRow(row);
+ table.nav.body.appendChild(el);
+ });
+
+ if (this._rows.length === 0) {
+ var el = table._renderEmpty();
+ table.nav.after.appendChild(el);
+ }
+
+ this._updateCss();
+ };
+
+ Table.prototype._renderEmpty = function() {
+ var div = document.createElement("div");
+ div.className = this.theme + "_empty";
+
+ return div;
+ };
+
+ Table.prototype._renderRow = function(row) {
+ var table = this;
+ var el = document.createElement("div");
+ el.className = table.theme + "_row";
+
+ row.cells.forEach(function(cell) {
+
+ var cellEl = document.createElement("div");
+ cellEl.className = table.theme + "_cell";
+ var interactive = table._renderCell(cell);
+
+ if (row.spacer) {
+ var wrap = document.createElement("div");
+ wrap.style.height = "0px";
+ wrap.style.overflow = "hidden";
+ wrap.appendChild(interactive);
+
+ cellEl.appendChild(wrap);
+ }
+ else {
+ cellEl.appendChild(interactive);
+
+ }
+
+ el.appendChild(cellEl);
+ });
+
+ var cell = document.createElement("div");
+ cell.classList.add(table.theme + "_cell");
+ cell.classList.add(table.theme + "_rowaction");
+
+ var span = document.createElement("span");
+ span.className = this.theme + "_delete";
+
+ span.addEventListener("click", function(ev) {
+ if (table.disabled) {
+ return;
+ }
+ table._removeRow(row);
+ table._render();
+ table._doOnInput();
+ });
+
+ if (!row.spacer) {
+ cell.appendChild(span);
+ }
+
+
+ el.appendChild(cell);
+
+ return el;
+ };
+
+ Table.prototype._doOnInput = function() {
+ var table = this;
+ if (typeof table.onInput === "function") {
+ var args = {}
+ table.onInput(args);
+ }
+ };
+
+ Table.prototype._renderCell = function(cell) {
+ var table = this;
+ var type = cell.type;
+ if (type === "text" || type === "number") {
+ var input = document.createElement("input");
+ input.type = type;
+ if (table.disabled) {
+ input.disabled = true;
+ }
+ if (cell.value) {
+ input.value = cell.value;
+ }
+ input.addEventListener("keyup", function(ev) {
+ if (type === "number") {
+ cell.value = Number(this.value);
+ }
+ else {
+ cell.value = this.value;
+ }
+ table._doOnInput();
+ });
+ return input;
+ }
+ else if (type === "select") {
+ var select = document.createElement("select");
+ if (table.disabled) {
+ select.disabled = true;
+ }
+
+ cell.data.options.forEach(function(item) {
+ var option = document.createElement("option");
+ option.innerText = item.name;
+ option.value = item.id;
+ option._originalValue = item.id;
+
+ select.appendChild(option);
+
+ if (cell.value === item.id) {
+ option.setAttribute("selected", true);
+ }
+ });
+
+ select.addEventListener("change", function(ev) {
+ var option = select.options[select.selectedIndex];
+ if (option && typeof option._originalValue !== "undefined") {
+ cell.value = option._originalValue;
+ }
+ table._doOnInput();
+ });
+
+ return select;
+ }
+
+ throw new Error("Unsupported item type: " + type);
+ };
+
+ Table.prototype.focus = function() {
+ };
+
+ // end of table.js
+
+})(DayPilot);
diff --git a/static/src/js/src/daypilot-month.src.js b/static/src/js/src/daypilot-month.src.js
new file mode 100644
index 0000000..e851482
--- /dev/null
+++ b/static/src/js/src/daypilot-month.src.js
@@ -0,0 +1,2440 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function() {
+
+ var doNothing = function() { };
+
+ if (typeof DayPilot.Month !== 'undefined' && DayPilot.Month.events) {
+ return;
+ }
+
+ var DayPilotMonth = {};
+
+ DayPilotMonth.Month = function(placeholder, options) {
+ this.v = '2024.3.539-lite';
+ this.nav = {};
+
+ var calendar = this;
+
+ this.id = placeholder;
+ this.isMonth = true;
+
+ // this.angularAutoApply = false;
+ this.api = 2;
+ this.backendUrl = null;
+ this.cellHeaderHeight = 24;
+ this.cellHeight = 100; // default cell height is 100 pixels (it's a minCellHeight, it will be extended if needed)
+ this.cellMarginBottom = 0;
+ this.contextMenu = null;
+ this.cssClassPrefix = "month_default";
+ this.eventBarVisible = true;
+ this.eventHeight = 25;
+ this.eventsLoadMethod = "GET";
+ this.headerHeight = 30;
+ this.hideUntilInit = true;
+ this.lineSpace = 1;
+ this.locale = "en-us";
+ this.showToolTip = true;
+ this.startDate = new DayPilot.Date(); // today
+ this.theme = null;
+ this.visible = true;
+ this.weekStarts = "Auto";
+ this.width = '100%'; // default width is 100%
+ this.xssProtection = "Enabled";
+
+ this.afterRender = function() { };
+
+ this.cellHeaderClickHandling = "Enabled";
+ this.eventClickHandling = "Enabled";
+ this.eventDeleteHandling = "Disabled";
+ this.eventMoveHandling = "Update";
+ this.eventResizeHandling = "Update";
+ this.eventRightClickHandling = "ContextMenu";
+ this.headerClickHandling = "Enabled";
+ this.timeRangeSelectedHandling = "Enabled";
+
+ this.onCellHeaderClick = null;
+ this.onCellHeaderClicked = null;
+ this.onEventClick = null;
+ this.onEventClicked = null;
+ this.onEventDelete = null;
+ this.onEventDeleted = null;
+ this.onEventMove = null;
+ this.onEventMoved = null;
+ this.onEventResize = null;
+ this.onEventResized = null;
+ this.onEventRightClick = null;
+ this.onEventRightClicked = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+
+ this.onBeforeEventRender = null;
+ this.onBeforeCellRender = null;
+
+ this.cellEvents = [];
+
+ this.elements = {};
+ this.elements.events = [];
+
+ this.cache = {};
+
+ this._disposed = false;
+
+ this._updateView = function(result, context) {
+
+ var result = JSON.parse(result);
+
+ if (result.CallBackRedirect) {
+ document.location.href = result.CallBackRedirect;
+ return;
+ }
+
+ if (result.UpdateType === "None") {
+ calendar.fireAfterRenderDetached(result.CallBackData, true);
+ return;
+ }
+
+ calendar.events.list = result.Events;
+
+ if (result.UpdateType === "Full") {
+
+ // properties
+ calendar.startDate = result.StartDate;
+ // calendar.headerBackColor = result.HeaderBackColor ? result.HeaderBackColor : calendar.headerBackColor;
+ // calendar.backColor = result.BackColor ? result.BackColor : calendar.backColor;
+ // calendar.nonBusinessBackColor = result.NonBusinessBackColor ? result.NonBusinessBackColor : calendar.nonBusinessBackColor;
+ calendar.timeFormat = result.TimeFormat ? result.TimeFormat : calendar.timeFormat;
+ if (typeof result.WeekStarts !== 'undefined') { calendar.weekStarts = result.WeekStarts; } // number, can be 0
+
+ calendar.hashes = result.Hashes;
+ }
+
+ calendar._deleteEvents();
+ calendar._prepareRows();
+ calendar._loadEvents();
+
+ if (result.UpdateType === "Full") {
+ calendar._clearTable();
+ calendar._drawTable();
+ }
+ calendar._updateHeight();
+
+ calendar.show();
+
+ calendar._drawEvents();
+
+ calendar.fireAfterRenderDetached(result.CallBackData, true);
+
+ };
+
+ this.fireAfterRenderDetached = function(data, isCallBack) {
+ var afterRenderDelayed = function(data, isc) {
+ return function() {
+ if (calendar.afterRender) {
+ calendar.afterRender(data, isc);
+ }
+ };
+ };
+
+ window.setTimeout(afterRenderDelayed(data, isCallBack), 0);
+ };
+
+ this.lineHeight = function() {
+ return this.eventHeight + this.lineSpace;
+ };
+
+ this.events = {};
+ //this.events.list = [];
+
+ this.events.add = function(e) {
+
+ var data = null;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ data = e;
+ }
+ else {
+ throw "DayPilot.Month.events.add() expects an object or DayPilot.Event instance.";
+ }
+
+ // e.calendar = calendar;
+
+ if (!calendar.events.list) {
+ calendar.events.list = [];
+ }
+
+ calendar.events.list.push(data);
+ calendar.update();
+ calendar._angular.notify();
+ };
+
+ this.events.find = function(id) {
+ if (!calendar.events.list) {
+ return null;
+ }
+
+ if (typeof id === "function") {
+ var fn = id;
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var e = new DayPilot.Event(calendar.events.list[i], calendar);
+ if (fn(e)) {
+ return e;
+ }
+ }
+ return null;
+ }
+
+ for (var i = 0; i < calendar.events.list.length; i++) {
+ var data = calendar.events.list[i];
+ if (data.id === id) {
+ return new DayPilot.Event(data, calendar);
+ }
+ }
+ return null;
+ };
+
+ this.events.update = function(e) {
+ if (e instanceof DayPilot.Event) {
+ e.commit();
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ var index = DayPilot.indexOf(calendar.events.list, target.data);
+ calendar.events.list.splice(index, 1, e);
+ }
+ }
+
+ calendar.update();
+ calendar._angular.notify();
+ };
+
+ this.events.remove = function(e) {
+ var data;
+ if (e instanceof DayPilot.Event) {
+ data = e.data;
+ }
+ else if (typeof e === "object") {
+ var target = calendar.events.find(e.id);
+ if (target) {
+ data = target.data;
+ }
+ }
+ else if (typeof e === "string" || typeof e === "number") {
+ var target = calendar.events.find(e);
+ if (target) {
+ data = target.data;
+ }
+ }
+
+ var index = DayPilot.indexOf(calendar.events.list, data);
+ calendar.events.list.splice(index, 1);
+ calendar.update();
+ calendar._angular.notify();
+ };
+
+ this.events.load = function(url, success, error) {
+ var onError = function (args) {
+ var largs = {};
+ largs.exception = args.exception;
+ largs.request = args.request;
+
+ if (typeof error === 'function') {
+ error(largs);
+ }
+ };
+
+ var onSuccess = function (args) {
+ var r = args.request;
+ var data;
+
+ // it's supposed to be JSON
+ try {
+ data = JSON.parse(r.responseText);
+ }
+ catch (e) {
+ var fargs = {};
+ fargs.exception = e;
+ onError(fargs);
+ return;
+ }
+
+ if (DayPilot.isArray(data)) {
+ var sargs = {};
+ sargs.preventDefault = function () {
+ this.preventDefault.value = true;
+ };
+ sargs.data = data;
+ if (typeof success === "function") {
+ success(sargs);
+ }
+
+ if (sargs.preventDefault.value) {
+ return;
+ }
+
+ calendar.events.list = data;
+ if (calendar._initialized) {
+ calendar.update();
+ }
+ }
+ };
+
+ var usePost = calendar.eventsLoadMethod && calendar.eventsLoadMethod.toUpperCase() === "POST";
+
+ if (usePost) {
+ DayPilot.Http.ajax({
+ "method": "POST",
+ "data": {"start": calendar.visibleStart().toString(), "end": calendar.visibleEnd().toString()},
+ "url": url,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ else {
+ var fullUrl = url;
+ var queryString = "start=" + calendar.visibleStart().toString() + "&end=" + calendar.visibleEnd().toString();
+ if (fullUrl.indexOf("?") > -1) {
+ fullUrl += "&" + queryString;
+ }
+ else {
+ fullUrl += "?" + queryString;
+ }
+
+ DayPilot.Http.ajax({
+ "method": "GET",
+ "url": fullUrl,
+ "success": onSuccess,
+ "error": onError
+ });
+ }
+ };
+
+ this.update = function(options) {
+
+ calendar._loadOptions(options);
+
+ if (!this._initialized) {
+ return;
+ }
+
+ if (calendar._disposed) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Month instance that has been disposed.");
+ }
+
+ if (!this.cells) {
+ return;
+ }
+
+ var full = true;
+
+ calendar._deleteEvents();
+ calendar._prepareRows();
+ calendar._loadEvents();
+
+ if (full) {
+ calendar._clearTable();
+ calendar._drawTable();
+ }
+ calendar._updateHeight();
+ calendar._show();
+ calendar._drawEvents();
+
+ if (this.visible) {
+ this.show();
+ }
+ else {
+ this.hide();
+ }
+ };
+
+ this._specialHandling = null;
+ this._loadOptions = function(options) {
+ if (!options) {
+ return;
+ }
+ var specialHandling = {
+ "events": {
+ "preInit": function() {
+ var events = this.data || [];
+ if (DayPilot.isArray(events.list)) {
+ calendar.events.list = events.list;
+ }
+ else {
+ calendar.events.list = events;
+ }
+ }
+ }
+ };
+ this._specialHandling = specialHandling;
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ var item = specialHandling[name];
+ item.data = options[name];
+ if (item.preInit) {
+ item.preInit();
+ }
+ }
+ else {
+ calendar[name] = options[name];
+ }
+ }
+
+ };
+
+ this._postInit = function() {
+ var specialHandling = this._specialHandling;
+ for (var name in specialHandling) {
+ var item = specialHandling[name];
+ if (item.postInit) {
+ item.postInit();
+ }
+ }
+ };
+
+ this._cache = {};
+ this._cache.events = [];
+
+ this._doBeforeEventRender = function(i) {
+ var cache = this._cache.events;
+ var data = this.events.list[i];
+ var evc = {};
+
+ // make a copy
+ for (var name in data) {
+ evc[name] = data[name];
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ var args = {};
+ args.control = calendar;
+ args.data = evc;
+ this.onBeforeEventRender(args);
+ }
+
+ cache[i] = evc;
+
+ };
+
+ this._loadEvents = function() {
+ var events = this.events.list;
+
+ if (!events) {
+ return;
+ }
+
+ if (!DayPilot.isArray(events)) {
+ throw new DayPilot.Exception("DayPilot.Month.events.list expects an array object. You supplied: " + (typeof events));
+ }
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ for (var i = 0; i < events.length; i++) {
+ this._doBeforeEventRender(i);
+ }
+ }
+
+ // prepare rows and columns
+ for (var x = 0; x < events.length; x++) {
+ var data = events[x];
+
+ if (typeof data !== "object") {
+ throw new DayPilot.Exception("Event data item must be an object");
+ }
+ if (!data.start) {
+ throw new DayPilot.Exception("Event data item must specify 'start' property");
+ }
+ if (!data.end) {
+ throw new DayPilot.Exception("Event data item must specify 'end' property");
+ }
+
+ var start = new DayPilot.Date(data.start);
+ var end = new DayPilot.Date(data.end);
+ if (start.getTime() > end.getTime()) { // skip invalid events, zero duration allowed
+ continue;
+ }
+ for (var i = 0; i < this.rows.length; i++) {
+ var row = this.rows[i];
+ var ep = new DayPilot.Event(data, this);
+ if (row.belongsHere(ep)) {
+ row.events.push(ep);
+
+ if (typeof this.onBeforeEventRender === 'function') {
+ ep.cache = this._cache.events[x];
+ }
+ }
+ }
+ }
+
+ // arrange events into lines
+ for (var ri = 0; ri < this.rows.length; ri++) {
+ var row = this.rows[ri];
+ row.events.sort(this._eventComparer);
+
+ for (var ei = 0; ei < this.rows[ri].events.length; ei++) {
+ var ev = row.events[ei];
+ var colStart = row.getStartColumn(ev);
+ var colWidth = row.getWidth(ev);
+ var line = row.putIntoLine(ev, colStart, colWidth, ri);
+ }
+ }
+ };
+
+
+ this._deleteEvents = function() {
+ for (var i = 0; i < this.elements.events.length; i++) {
+ var e = this.elements.events[i];
+ e.event = null;
+ e.click = null;
+ e.parentNode.removeChild(e);
+ }
+
+ this.elements.events = [];
+
+ };
+
+ this._drawEvents = function() {
+ //this.cache.events = {}; // reset DayPilotMonth.Event object cache
+
+ this._drawEventsRows();
+ };
+
+ this._drawEventsRows = function() {
+ this.elements.events = [];
+
+ // draw events
+ for (var ri = 0; ri < this.rows.length; ri++) {
+ var row = this.rows[ri];
+
+ for (var li = 0; li < row.lines.length; li++) {
+ var line = row.lines[li];
+
+ for (var pi = 0; pi < line.length; pi++) {
+ this._drawEvent(line[pi]);
+ }
+ }
+ }
+
+ };
+
+ this._eventComparer = function(a, b) {
+ if (!a || !b || !a.start || !b.start) {
+ return 0; // no sorting, invalid arguments
+ }
+
+ var byStart = a.start().getTime() - b.start().getTime();
+ if (byStart !== 0) {
+ return byStart;
+ }
+
+ var byEnd = b.end().getTime() - a.end().getTime(); // desc
+ return byEnd;
+ };
+
+ this.drawShadow = function(x, y, line, width, offset, e) {
+
+ if (!offset) {
+ offset = 0;
+ }
+
+ var remains = width;
+
+ this.shadow = {};
+ this.shadow.list = [];
+ this.shadow.start = { x: x, y: y };
+ this.shadow.width = width;
+
+ // something before the first day
+ var hidden = y * 7 + x - offset;
+ if (hidden < 0) {
+ //document.title = hidden + ' ' + new Date();
+ remains += hidden;
+ x = 0;
+ y = 0;
+ }
+
+ var remainingOffset = offset;
+ while (remainingOffset >= 7) {
+ y--;
+ remainingOffset -= 7;
+ }
+ if (remainingOffset > x) {
+ var plus = 7 - this.getColCount();
+ if (remainingOffset > (x + plus)) {
+ y--;
+ x = x + 7 - remainingOffset;
+ }
+ else {
+ remains = remains - remainingOffset + x;
+ x = 0;
+ }
+ }
+ else {
+ x -= remainingOffset;
+ }
+
+ if (y < 0) {
+ y = 0;
+ x = 0;
+ }
+
+ var cursor = null;
+ if (DayPilotMonth.resizingEvent) {
+ cursor = 'w-resize';
+ }
+ else if (DayPilotMonth.movingEvent) {
+ cursor = "move";
+ }
+
+ this.nav.top.style.cursor = cursor;
+
+ while (remains > 0 && y < this.rows.length) {
+ var drawNow = Math.min(this.getColCount() - x, remains);
+ var row = this.rows[y];
+
+
+ var top = this.getRowTop(y);
+ var height = row.getHeight();
+
+ var shadow = document.createElement("div");
+ shadow.setAttribute("unselectable", "on");
+ shadow.style.position = 'absolute';
+ shadow.style.left = (this.getCellWidth() * x) + '%';
+ shadow.style.width = (this.getCellWidth() * drawNow) + '%';
+ shadow.style.top = (top) + 'px';
+ shadow.style.height = (height) + 'px';
+ shadow.style.cursor = cursor;
+ shadow.classList.add(calendar._prefixCssClass("_shadow"));
+
+ var inside = document.createElement("div");
+ inside.setAttribute("unselectable", "on");
+ shadow.appendChild(inside);
+
+ inside.style.position = "absolute";
+ inside.style.top = "0px";
+ inside.style.right = "0px";
+ inside.style.left = "0px";
+ inside.style.bottom = "0px";
+ inside.classList.add(calendar._prefixCssClass("_shadow_inner"));
+
+ // inside.style.backgroundColor = "#aaaaaa";
+ // inside.style.opacity = 0.5;
+ // inside.style.filter = "alpha(opacity=50)";
+ //inside.style.border = '2px solid #aaaaaa';
+ /*
+ if (e) {
+ inside.style.overflow = 'hidden';
+ inside.style.fontSize = this.eventFontSize;
+ inside.style.fontFamily = this.eventFontFamily;
+ inside.style.color = this.eventFontColor;
+ inside.innerHTML = e.client.html();
+ }
+ */
+
+ this.nav.top.appendChild(shadow);
+ this.shadow.list.push(shadow);
+
+ remains -= (drawNow + 7 - this.getColCount());
+ x = 0;
+ y++;
+ }
+
+ };
+
+ this.clearShadow = function() {
+ if (this.shadow) {
+ for (var i = 0; i < this.shadow.list.length; i++) {
+ this.nav.top.removeChild(this.shadow.list[i]);
+ }
+ this.shadow = null;
+ this.nav.top.style.cursor = '';
+ }
+ };
+
+ this.getEventTop = function(row, line) {
+ var top = this.headerHeight;
+ for (var i = 0; i < row; i++) {
+ top += this.rows[i].getHeight();
+ }
+ top += this.cellHeaderHeight; // space on top
+ top += line * this.lineHeight();
+ return top;
+ };
+
+ this.getDateFromCell = function(x, y) {
+ //return DayPilot.Date.addDays(this.firstDate, y * 7 + x);
+ return this.firstDate.addDays(y * 7 + x);
+ };
+
+ this._drawEvent = function(e) {
+ /*
+
+ supported properties:
+ * cssClass
+ * backColor
+ * borderColor
+ * barColor
+ * barHidden
+ * fontColor
+ * html
+ * toolTip
+ *
+ * clickEnabled
+ */
+
+ var data = e.cache || e.data;
+
+ //var ev = eventPart.event;
+ var row = e.part.row;
+ var line = e.part.line;
+ var colStart = e.part.colStart;
+ var colWidth = e.part.colWidth;
+
+ var left = this.getCellWidth() * (colStart);
+ var width = this.getCellWidth() * (colWidth);
+ var top = this.getEventTop(row, line);
+
+ var div = document.createElement("div");
+ div.setAttribute("unselectable", "on");
+ div.style.height = this.eventHeight + 'px';
+ div.className = this._prefixCssClass("_event");
+
+ if (data.cssClass) {
+ DayPilot.Util.addClass(div, data.cssClass);
+ }
+
+ if (!e.part.startsHere) {
+ DayPilot.Util.addClass(div, this._prefixCssClass("_event_continueleft"));
+ }
+
+ if (!e.part.endsHere) {
+ DayPilot.Util.addClass(div, this._prefixCssClass("_event_continueright"));
+ }
+
+ div.event = e;
+
+ div.style.width = width + '%';
+ div.style.position = 'absolute';
+ div.style.left = left + '%';
+ div.style.top = top + 'px'; // plus space on top
+
+ if (this.showToolTip && e.client.toolTip()) {
+ div.title = e.client.toolTip();
+ }
+
+ div.onclick = calendar._eventClickDispatch;
+ div.oncontextmenu = calendar._onEventContextMenu;
+ div.onmousedown = function(ev) {
+ ev = ev || window.event;
+ var button = ev.which || ev.button;
+
+ ev.cancelBubble = true;
+ if (ev.stopPropagation) {
+ ev.stopPropagation();
+ }
+
+ if (button === 1) {
+
+ DayPilotMonth.movingEvent = null;
+ if (this.style.cursor === 'w-resize' || this.style.cursor === 'e-resize') {
+ var resizing = {};
+ resizing.start = {};
+ resizing.start.x = colStart;
+ resizing.start.y = row;
+ resizing.event = div.event;
+ resizing.width = DayPilot.DateUtil.daysSpan(resizing.event.start(), resizing.event.end()) + 1;
+ resizing.direction = this.style.cursor;
+ DayPilotMonth.resizingEvent = resizing;
+ }
+ else if (this.style.cursor === 'move' || e.client.moveEnabled()) {
+ calendar.clearShadow();
+
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+ if (!coords) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+
+ var hidden = DayPilot.DateUtil.daysDiff(e.start(), calendar.rows[row].start);
+ var offset = (cell.y * 7 + cell.x) - (row * 7 + colStart);
+ if (hidden) {
+ offset += hidden;
+ }
+
+ var moving = {};
+ moving.start = {};
+ moving.start.x = colStart;
+ moving.start.y = row;
+ moving.start.line = line;
+ moving.offset = calendar.eventMoveToPosition ? 0 : offset;
+ moving.colWidth = colWidth;
+ moving.event = div.event;
+ moving.coords = coords;
+ DayPilotMonth.movingEvent = moving;
+
+ }
+
+ }
+ };
+
+ div.onmousemove = function(ev) {
+ if (typeof (DayPilotMonth) === 'undefined') {
+ return;
+ }
+
+ if (DayPilotMonth.movingEvent || DayPilotMonth.resizingEvent) {
+ return;
+ }
+
+ // position
+ var offset = DayPilot.mo3(div, ev);
+ if (!offset) {
+ return;
+ }
+
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "";
+ }
+
+ var resizeMargin = 6;
+
+ if (offset.x <= resizeMargin && e.client.resizeEnabled()) {
+ if (e.part.startsHere) {
+ div.style.cursor = "w-resize";
+ div.dpBorder = 'left';
+ }
+ else {
+ div.style.cursor = 'not-allowed';
+ }
+ }
+ else if (div.clientWidth - offset.x <= resizeMargin && e.client.resizeEnabled()) {
+ if (e.part.endsHere) {
+ div.style.cursor = "e-resize";
+ div.dpBorder = 'right';
+ }
+ else {
+ div.style.cursor = 'not-allowed';
+ }
+ }
+ else if (e.client.clickEnabled()) {
+ div.style.cursor = "pointer";
+ }
+ else {
+ div.style.cursor = 'default';
+ }
+
+ };
+
+ div.onmouseleave = function(ev) {
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "none";
+ }
+ div.style.cursor = '';
+ };
+
+ div.onmouseenter = function(ev) {
+ if (div.deleteIcon) {
+ div.deleteIcon.style.display = "";
+ }
+ };
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.className = this._prefixCssClass("_event_inner");
+
+ if (data.borderColor === "darker" && data.backColor) {
+ inner.style.borderColor = DayPilot.ColorUtil.darker(data.backColor, 2);
+ }
+ else {
+ inner.style.borderColor = data.borderColor;
+ }
+
+ if (data.backColor) {
+ inner.style.background = data.backColor;
+ }
+
+ if (data.fontColor) {
+ inner.style.color = data.fontColor;
+ }
+
+
+ inner.innerHTML = e.client.html();
+
+ div.appendChild(inner);
+
+ if (e.client.barVisible()) {
+
+ var bar = document.createElement("div");
+ bar.setAttribute("unselectable", "on");
+ bar.className = this._prefixCssClass("_event_bar");
+ bar.style.position = "absolute";
+
+ var barInner = document.createElement("div");
+ barInner.setAttribute("unselectable", "on");
+ barInner.className = this._prefixCssClass("_event_bar_inner");
+ barInner.style.top = "0%";
+ barInner.style.height = "100%";
+
+ if (data.barColor) {
+ barInner.style.backgroundColor = data.barColor;
+ }
+
+ bar.appendChild(barInner);
+ div.appendChild(bar);
+ }
+
+
+ if (e.client.deleteEnabled()) {
+ var dheight = 18;
+ var dwidth = 18;
+ var dtop = Math.floor(calendar.eventHeight / 2 - dheight / 2);
+
+ var del = document.createElement("div");
+ del.style.position = "absolute";
+ del.style.right = "2px";
+ del.style.top = dtop + "px";
+ del.style.width = dwidth + "px";
+ del.style.height = dheight + "px";
+ del.className = calendar._prefixCssClass("_event_delete");
+ del.onmousedown = function(ev) {
+ ev.stopPropagation();
+ };
+ del.onclick = function(ev) {
+ ev.stopPropagation();
+ var e = this.parentNode.event;
+ if (e) {
+ calendar._eventDeleteDispatch(e);
+ }
+ };
+ del.style.display = "none";
+ div.deleteIcon = del;
+ div.appendChild(del);
+ }
+
+ // areas
+ var areas = data.areas ? DayPilot.Areas.copy(data.areas) : [];
+ DayPilot.Areas.attach(div, e, {"areas": areas});
+
+ if (typeof calendar.onAfterEventRender === 'function') {
+ var args = {};
+ args.e = div.event;
+ args.div = div;
+
+ calendar.onAfterEventRender(args);
+ }
+
+ this.elements.events.push(div);
+
+ this.nav.events.appendChild(div);
+ };
+
+
+ // returns DayPilot.Date object
+ this.lastVisibleDayOfMonth = function() {
+ return this.startDate.lastDayOfMonth();
+ };
+
+ this._prepareRows = function() {
+
+ if (typeof this.startDate === 'string') {
+ this.startDate = new DayPilot.Date(this.startDate);
+ }
+ this.startDate = this.startDate.firstDayOfMonth();
+
+ this.firstDate = this.startDate.firstDayOfWeek(this.getWeekStart());
+
+ var firstDayOfMonth = this.startDate;
+
+ var rowCount;
+
+ var lastVisibleDayOfMonth = this.lastVisibleDayOfMonth();
+ var count = DayPilot.DateUtil.daysDiff(this.firstDate, lastVisibleDayOfMonth) + 1;
+ rowCount = Math.ceil(count / 7);
+
+ this.days = rowCount * 7;
+
+ this.rows = [];
+ for (var x = 0; x < rowCount; x++) {
+ var r = {};
+ r.start = this.firstDate.addDays( x * 7); // start point
+ r.end = r.start.addDays(this.getColCount()); // end point
+ r.events = []; // collection of events
+ r.lines = []; // collection of lines
+ r.index = x; // row index
+ r.minHeight = this.cellHeight; // default, can be extended during events loading
+ r.calendar = this;
+
+ r.belongsHere = function(ev) {
+ if (ev.end().getTime() === ev.start().getTime() && ev.start().getTime() === this.start.getTime()) {
+ return true;
+ }
+ return !(ev.end().getTime() <= this.start.getTime() || ev.start().getTime() >= this.end.getTime());
+ };
+
+ r.getPartStart = function(ev) {
+ return DayPilot.DateUtil.max(this.start, ev.start());
+ };
+
+ r.getPartEnd = function(ev) {
+ return DayPilot.DateUtil.min(this.end, ev.end());
+ };
+
+ r.getStartColumn = function(ev) {
+ var partStart = this.getPartStart(ev);
+ return DayPilot.DateUtil.daysDiff(this.start, partStart);
+ };
+
+ r.getWidth = function(ev) {
+ return DayPilot.DateUtil.daysSpan(this.getPartStart(ev), this.getPartEnd(ev)) + 1;
+ };
+
+ r.putIntoLine = function(ev, colStart, colWidth, row) {
+ var thisRow = this;
+
+ for (var i = 0; i < this.lines.length; i++) {
+ var line = this.lines[i];
+ if (line.isFree(colStart, colWidth)) {
+ line.addEvent(ev, colStart, colWidth, row, i);
+ return i;
+ }
+ }
+
+ var line = [];
+ line.isFree = function(colStart, colWidth) {
+ var free = true;
+
+ for (var i = 0; i < this.length; i++) {
+ if (!(colStart + colWidth - 1 < this[i].part.colStart || colStart > this[i].part.colStart + this[i].part.colWidth - 1)) {
+ free = false;
+ }
+ }
+
+ return free;
+ };
+
+ line.addEvent = function(ep, colStart, colWidth, row, index) {
+ //var eventPart = {};
+ //eventPart.event = ev;
+ ep.part.colStart = colStart;
+ ep.part.colWidth = colWidth;
+ ep.part.row = row;
+ ep.part.line = index;
+ ep.part.startsHere = thisRow.start.getTime() <= ep.start().getTime();
+ //if (confirm('r.start: ' + thisRow.start + ' ev.Start: ' + ev.Start)) thisRow = null;
+ ep.part.endsHere = thisRow.end.getTime() >= ep.end().getTime();
+
+ this.push(ep);
+ };
+
+ line.addEvent(ev, colStart, colWidth, row, this.lines.length);
+
+ this.lines.push(line);
+
+ return this.lines.length - 1;
+ };
+
+ r.getStart = function() {
+ var start = 0;
+ for (var i = 0; i < calendar.rows.length && i < this.index; i++) {
+ start += calendar.rows[i].getHeight();
+ }
+ };
+
+ r.getHeight = function() {
+ return Math.max(this.lines.length * calendar.lineHeight() + calendar.cellHeaderHeight + calendar.cellMarginBottom, this.calendar.cellHeight);
+ };
+
+ this.rows.push(r);
+ }
+
+ //this.endDate = DayPilot.Date.addDays(this.firstDate, rowCount * 7);
+ this.endDate = this.firstDate.addDays(rowCount * 7);
+ };
+
+ this.visibleStart = function() {
+ return calendar.firstDate;
+ };
+
+ this.visibleEnd = function() {
+ return calendar.endDate;
+ };
+
+ this.getHeight = function() {
+ var height = this.headerHeight;
+ for (var i = 0; i < this.rows.length; i++) {
+ height += this.rows[i].getHeight();
+ }
+ return height;
+ };
+
+ this.getWidth = function(start, end) {
+ var diff = (end.y * 7 + end.x) - (start.y * 7 + start.x);
+ return diff + 1;
+ };
+
+ this.getMinCoords = function(first, second) {
+ if ((first.y * 7 + first.x) < (second.y * 7 + second.x)) {
+ return first;
+ }
+ else {
+ return second;
+ }
+ };
+
+ this._prefixCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ if (prefix) {
+ return prefix + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this._drawTop = function() {
+ var relative = this.nav.top;
+ //this.nav.top = relative;
+ relative.setAttribute("unselectable", "on");
+ relative.style.MozUserSelect = 'none';
+ relative.style.KhtmlUserSelect = 'none';
+ relative.style.WebkitUserSelect = 'none';
+ relative.style.position = 'relative';
+ if (this.width) {
+ relative.style.width = this.width;
+ }
+ relative.style.height = this.getHeight() + 'px';
+ relative.onselectstart = function(e) { return false; }; // prevent text cursor in Chrome during drag&drop
+
+
+ if (this.hideUntilInit) {
+ relative.style.visibility = 'hidden';
+ }
+
+ if (!this.visible) {
+ relative.style.display = "none";
+ }
+
+ relative.className = this._prefixCssClass("_main");
+
+ var cells = document.createElement("div");
+ this.nav.cells = cells;
+ cells.style.position = "absolute";
+ cells.style.left = "0px";
+ cells.style.right = "0px";
+ cells.setAttribute("unselectable", "on");
+ relative.appendChild(cells);
+
+ var events = document.createElement("div");
+ this.nav.events = events;
+ events.style.position = "absolute";
+ events.style.left = "0px";
+ events.style.right = "0px";
+ events.setAttribute("unselectable", "on");
+ relative.appendChild(events);
+
+ relative.onmousemove = function(ev) {
+
+ if (DayPilotMonth.resizingEvent) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ if (!coords) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+ calendar.clearShadow();
+ var resizing = DayPilotMonth.resizingEvent;
+
+ var original = resizing.start;
+ var width, start;
+
+ if (resizing.direction === 'w-resize') {
+ start = cell;
+
+ var endDate = resizing.event.end();
+ //if (DayPilot.Date.getDate(endDate).getTime() === endDate.getTime()) {
+ if (endDate.getDatePart() === endDate) {
+ endDate = endDate.addDays(-1);
+ }
+
+ var end = calendar.getCellFromDate(endDate);
+ width = calendar.getWidth(cell, end);
+ }
+ else {
+ start = calendar.getCellFromDate(resizing.event.start());
+ width = calendar.getWidth(start, cell);
+ }
+
+ if (width < 1) {
+ width = 1;
+ }
+
+ calendar.drawShadow(start.x, start.y, 0, width);
+
+ }
+ else if (DayPilotMonth.movingEvent) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ if (!coords) {
+ return;
+ }
+
+ // not actually moved, Chrome bug
+ if (coords.x === DayPilotMonth.movingEvent.coords.x && coords.y === DayPilotMonth.movingEvent.coords.y) {
+ return;
+ }
+
+ var minDistance = 3;
+ var distance = Math.abs(coords.x - DayPilotMonth.movingEvent.coords.x) + Math.abs(coords.y - DayPilotMonth.movingEvent.coords.y);
+ if (distance <= minDistance) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+
+ calendar.clearShadow();
+
+ var event = DayPilotMonth.movingEvent.event;
+ var offset = DayPilotMonth.movingEvent.offset;
+ var width = calendar.cellMode ? 1 : DayPilot.DateUtil.daysSpan(event.start(), event.end()) + 1;
+
+ if (width < 1) {
+ width = 1;
+ }
+ calendar.drawShadow(cell.x, cell.y, 0, width, offset, event);
+ }
+ else if (DayPilotMonth.timeRangeSelecting) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ if (!coords) {
+ return;
+ }
+
+ var cell = calendar.getCellBelowPoint(coords.x, coords.y);
+
+ calendar.clearShadow();
+
+ var start = DayPilotMonth.timeRangeSelecting;
+
+ var startIndex = start.y * 7 + start.x;
+ var cellIndex = cell.y * 7 + cell.x;
+
+ var width = Math.abs(cellIndex - startIndex) + 1;
+
+ if (width < 1) {
+ width = 1;
+ }
+
+ var shadowStart = startIndex < cellIndex ? start : cell;
+
+ DayPilotMonth.timeRangeSelecting.from = { x: shadowStart.x, y: shadowStart.y };
+ DayPilotMonth.timeRangeSelecting.width = width;
+ DayPilotMonth.timeRangeSelecting.moved = true;
+
+ calendar.drawShadow(shadowStart.x, shadowStart.y, 0, width, 0, null);
+
+ }
+
+ };
+
+ //this.nav.top.appendChild(this.vsph);
+ };
+
+ this._updateHeight = function() {
+ this.nav.top.style.height = this.getHeight() + 'px';
+
+ for (var x = 0; x < this.cells.length; x++) {
+ for (var y = 0; y < this.cells[x].length; y++) {
+ this.cells[x][y].style.top = this.getRowTop(y) + 'px';
+ this.cells[x][y].style.height = this.rows[y].getHeight() + 'px';
+ }
+ }
+ };
+
+ this.getCellBelowPoint = function(x, y) {
+ var columnWidth = Math.floor(this.nav.top.clientWidth / this.getColCount());
+ var column = Math.min(Math.floor(x / columnWidth), this.getColCount() - 1);
+
+ var row = null;
+
+ var height = this.headerHeight;
+ var relativeY = 0;
+ for (var i = 0; i < this.rows.length; i++) {
+ var baseHeight = height;
+ height += this.rows[i].getHeight();
+ if (y < height) {
+ relativeY = y - baseHeight;
+ row = i;
+ break;
+ }
+ }
+ if (row === null) {
+ row = this.rows.length - 1; // might be a pixel below the last line
+ }
+
+ var cell = {};
+ cell.x = column;
+ cell.y = row;
+ cell.relativeY = relativeY;
+
+ return cell;
+ };
+
+ this.getCellFromDate = function(date) {
+ var width = DayPilot.DateUtil.daysDiff(this.firstDate, date);
+ var cell = { x: 0, y: 0 };
+ while (width >= 7) {
+ cell.y++;
+ width -= 7;
+ }
+ cell.x = width;
+ return cell;
+ };
+
+ this._drawTable = function() {
+
+ var table = document.createElement("div");
+ table.oncontextmenu = function() { return false; };
+ this.nav.cells.appendChild(table);
+
+ this.cells = [];
+
+ for (var x = 0; x < this.getColCount(); x++) {
+
+ this.cells[x] = [];
+ var headerProperties = null;
+
+ var header = document.createElement("div");
+ header.setAttribute("unselectable", "on");
+ header.style.position = 'absolute';
+
+ header.style.left = (this.getCellWidth() * x) + '%';
+ header.style.width = (this.getCellWidth()) + '%';
+ header.style.top = '0px';
+ header.style.height = (this.headerHeight) + 'px';
+
+ var dayIndex = x + this.getWeekStart();
+ if (dayIndex > 6) {
+ dayIndex -= 7;
+ }
+
+ header.className = this._prefixCssClass("_header");
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ inner.innerHTML = resolved.locale().dayNames[dayIndex];
+
+ header.appendChild(inner);
+
+ inner.style.position = "absolute";
+ inner.style.top = "0px";
+ inner.style.bottom = "0px";
+ inner.style.left = "0px";
+ inner.style.right = "0px";
+ inner.className = this._prefixCssClass("_header_inner");
+
+ inner.innerHTML = resolved.locale().dayNames[dayIndex];
+
+ table.appendChild(header);
+
+ for (var y = 0; y < this.rows.length; y++) {
+ this._drawCell(x, y, table);
+ }
+
+ }
+
+ };
+
+ this._clearTable = function() {
+
+ // clear event handlers
+ for (var x = 0; x < this.cells.length; x++) {
+ for (var y = 0; y < this.cells[x].length; y++) {
+ this.cells[x][y].onclick = null;
+ }
+ }
+
+ this.nav.cells.innerHTML = '';
+
+ };
+
+ this._api2 = function() {
+ return calendar.api === 2;
+ };
+ this._drawCell = function(x, y, table) {
+
+ var row = this.rows[y];
+ var d = this.firstDate.addDays(y * 7 + x);
+
+ var date = d.getDay();
+ var headerHtml = null;
+ if (date === 1) {
+ headerHtml = resolved.locale().monthNames[d.getMonth()] + ' ' + date;
+ }
+ else {
+ headerHtml = "" + date;
+ }
+ var business = !calendar.isWeekend(d);
+
+ var args = {};
+ args.control = calendar;
+ args.cell = {};
+ args.cell.start = d;
+ args.cell.end = d.addDays(1);
+ args.cell.properties = {
+ "headerHtml": headerHtml,
+ "backColor": null,
+ "business": business,
+ "html": null
+ };
+ if (typeof calendar.onBeforeCellRender === "function") {
+ calendar.onBeforeCellRender(args);
+ }
+
+ var props = args.cell.properties;
+
+ var cell = document.createElement("div");
+ cell.setAttribute("unselectable", "on");
+ cell.style.position = 'absolute';
+ cell.style.cursor = 'default';
+ cell.style.left = (this.getCellWidth() * x) + '%';
+ cell.style.width = (this.getCellWidth()) + '%';
+ cell.style.top = (this.getRowTop(y)) + 'px';
+ cell.style.height = (row.getHeight()) + 'px';
+ cell.className = this._prefixCssClass("_cell");
+
+ if (props.business) {
+ var business = this._prefixCssClass("_cell_business");
+ DayPilot.Util.addClass(cell, business);
+ }
+
+ var previousMonth = this.startDate.addMonths(-1).getMonth();
+ var nextMonth = this.startDate.addMonths(1).getMonth();
+
+ var thisMonth = this.startDate.getMonth();
+
+ var inner = document.createElement("div");
+ inner.setAttribute("unselectable", "on");
+ cell.appendChild(inner);
+
+ inner.style.position = "absolute";
+ inner.style.left = "0px";
+ inner.style.right = "0px";
+ inner.style.top = "0px";
+ inner.style.bottom = "0px";
+ inner.className = this._prefixCssClass("_cell_inner");
+
+ if (props.backColor) {
+ inner.style.backgroundColor = args.cell.properties.backColor;
+ }
+
+ cell.onmousedown = function(e) {
+ if (calendar.timeRangeSelectedHandling !== 'Disabled') {
+ calendar.clearShadow();
+ DayPilotMonth.timeRangeSelecting = { "root": calendar, "x": x, "y": y, "from": { x: x, y: y }, "width": 1 };
+ }
+ };
+
+ cell.onclick = function() {
+
+ var single = function(d) {
+ var start = new DayPilot.Date(d);
+ var end = start.addDays(1);
+ calendar._timeRangeSelectedDispatch(start, end);
+ };
+
+ if (calendar.timeRangeSelectedHandling !== 'Disabled') {
+ single(d);
+ return;
+ }
+
+ };
+
+ var day = document.createElement("div");
+ day.setAttribute("unselectable", "on");
+ day.style.height = this.cellHeaderHeight + "px";
+ day.className = this._prefixCssClass("_cell_header");
+
+ day.onclick = function(ev) {
+
+ if (calendar.cellHeaderClickHandling !== "Enabled") {
+ return;
+ }
+
+ ev.stopPropagation();
+
+ var args = {};
+ args.control = calendar;
+ args.start = d;
+ args.end = d.addDays(1);
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onCellHeaderClick === "function") {
+ calendar.onCellHeaderClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ if (typeof calendar.onCellHeaderClicked === "function") {
+ calendar.onCellHeaderClicked(args);
+ }
+
+ };
+
+ day.innerHTML = props.headerHtml;
+
+ inner.appendChild(day);
+
+ if (props.html) {
+ var html = document.createElement("div");
+ html.style.height = (row.getHeight() - this.cellHeaderHeight) + 'px';
+ html.style.overflow = 'hidden';
+ html.innerHTML = props.html;
+ inner.appendChild(html);
+ }
+
+ this.cells[x][y] = cell;
+
+ table.appendChild(cell);
+ };
+
+ this.getWeekStart = function() {
+ if (calendar.weekStarts === 'Auto') {
+ var locale = resolved.locale();
+ if (locale) {
+ return locale.weekStarts;
+ }
+ else {
+ return 0; // Sunday
+ }
+ }
+ else {
+ return calendar.weekStarts || 0;
+ }
+ return resolved.locale().weekStarts;
+ };
+
+ this.getColCount = function() {
+ return 7;
+ };
+
+ this.getCellWidth = function() {
+ return 14.285;
+ };
+
+ this.getRowTop = function(index) {
+ var top = this.headerHeight;
+ for (var i = 0; i < index; i++) {
+ top += this.rows[i].getHeight();
+ }
+ return top;
+ };
+
+ this._callBack2 = function(action, data, parameters) {
+
+ var envelope = {};
+
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + DayPilot.JSON.stringify(envelope);
+
+ if (this.backendUrl) {
+ DayPilot.request(this.backendUrl, this._callBackResponse, commandstring, this.ajaxError);
+ }
+ };
+
+ this._callBackResponse = function(response) {
+ calendar._updateView(response.responseText);
+ };
+
+ this._getCallBackHeader = function() {
+ var h = {};
+ h.control = "dpm";
+ h.id = this.id;
+ h.v = this.v;
+
+ h.visibleStart = new DayPilot.Date(this.firstDate);
+ h.visibleEnd = h.visibleStart.addDays(this.days);
+
+ h.startDate = calendar.startDate;
+ // h.headerBackColor = this.headerBackColor;
+ // h.backColor = this.backColor;
+ // h.nonBusinessBackColor = this.nonBusinessBackColor;
+ h.timeFormat = this.timeFormat;
+ h.weekStarts = this.weekStarts;
+
+ return h;
+ };
+
+ this.eventClickCallBack = function(e, data) {
+ this._callBack2('EventClick', data, e);
+ };
+
+ this._eventClickDispatch = function(e) {
+
+ DayPilotMonth.movingEvent = null;
+ DayPilotMonth.resizingEvent = null;
+
+ var div = this;
+
+ var e = e || window.event;
+ var ctrlKey = e.ctrlKey;
+
+ e.cancelBubble = true;
+ if (e.stopPropagation) {
+ e.stopPropagation();
+ }
+
+ calendar.eventClickSingle(div, e);
+ };
+
+
+ this.eventClickSingle = function(div, ev) {
+ var e = div.event;
+ if (!e) {
+ return;
+ }
+ if (!e.client.clickEnabled()) {
+ return;
+ }
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.control = calendar;
+ args.div = div;
+ args.originalEvent = ev;
+ args.meta = ev.metaKey;
+ args.ctrl = ev.ctrlKey;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventClick === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClick(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventClickHandling) {
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(e);
+ }
+ }
+ break; }
+
+ if (typeof calendar.onEventClicked === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventClicked(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventClickHandling) {
+ case 'CallBack':
+ calendar.eventClickCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventClick(e);
+ break;
+ }
+ }
+ };
+
+ this._onEventContextMenu = function() {
+ var e = this;
+ calendar._eventRightClickDispatch(e.event);
+ return false;
+ };
+
+ this._eventRightClickDispatch = function(e) {
+
+ this.event = e;
+
+ if (!e.client.rightClickEnabled()) {
+ return false;
+ }
+
+ var args = {};
+ args.e = e;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventRightClick === 'function') {
+ calendar.onEventRightClick(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventRightClickHandling) {
+ case 'ContextMenu':
+ var menu = e.client.contextMenu();
+ if (menu) {
+ menu.show(e);
+ }
+ else {
+ if (calendar.contextMenu) {
+ calendar.contextMenu.show(this.event);
+ }
+ }
+ break;
+ }
+
+ if (typeof calendar.onEventRightClicked === 'function') {
+ calendar.onEventRightClicked(args);
+ }
+
+
+ return false;
+ };
+
+
+ this._eventDeleteDispatch = function (e) {
+ if (calendar._api2()) {
+
+ var args = {};
+ args.e = e;
+ args.control = calendar;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventDelete === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDelete(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventDeleteHandling) {
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'Update':
+ calendar.events.remove(e);
+ break;
+ }
+
+ if (typeof calendar.onEventDeleted === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventDeleted(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventDeleteHandling) {
+ case 'PostBack':
+ calendar.eventDeletePostBack(e);
+ break;
+ case 'CallBack':
+ calendar.eventDeleteCallBack(e);
+ break;
+ case 'JavaScript':
+ calendar.onEventDelete(e);
+ break;
+ }
+ }
+
+ };
+
+ this.eventDeleteCallBack = function(e, data) {
+ this._callBack2('EventDelete', data, e);
+ };
+
+ this.eventDeletePostBack = function(e, data) {
+ this._postBack2('EventDelete', data, e);
+ };
+
+ this.eventMoveCallBack = function(e, newStart, newEnd, data, position) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+ params.position = position;
+
+ this._callBack2('EventMove', data, params);
+ };
+
+ this._eventMoveDispatch = function(e, x, y, offset, ev, position) {
+
+ var startOffset = e.start().getTimePart();
+
+ var endDate = e.end().getDatePart();
+ if (endDate.getTime() !== e.end().getTime()) {
+ endDate = endDate.addDays(1);
+ }
+ var endOffset = DayPilot.DateUtil.diff(e.end(), endDate);
+
+ var boxStart = this.getDateFromCell(x, y);
+ boxStart = boxStart.addDays(-offset);
+ var width = DayPilot.DateUtil.daysSpan(e.start(), e.end()) + 1;
+
+ var boxEnd = boxStart.addDays(width);
+
+ var newStart = boxStart.addTime(startOffset);
+ var newEnd = boxEnd.addTime(endOffset);
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.control = calendar;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.ctrl = ev.ctrlKey;
+ args.shift = ev.shiftKey;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventMove === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMove(args);
+ });
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventMoveHandling) {
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventMoved === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventMoved(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventMoveHandling) {
+ case 'CallBack':
+ calendar.eventMoveCallBack(e, newStart, newEnd);
+ break;
+ case 'JavaScript':
+ calendar.onEventMove(e, newStart, newEnd);
+ break;
+ }
+ }
+ };
+
+ this.eventResizeCallBack = function(e, newStart, newEnd, data) {
+ if (!newStart)
+ throw 'newStart is null';
+ if (!newEnd)
+ throw 'newEnd is null';
+
+ var params = {};
+ params.e = e;
+ params.newStart = newStart;
+ params.newEnd = newEnd;
+
+ this._callBack2('EventResize', data, params);
+ };
+
+ this._eventResizeDispatch = function(e, start, width) {
+ var startOffset = e.start().getTimePart();
+
+ var endDate = e.end().getDatePart();
+ if (endDate.getTime() !== e.end().getTime()) {
+ endDate = endDate.addDays(1);
+ }
+ var endOffset = DayPilot.DateUtil.diff(e.end(), endDate);
+
+ var boxStart = this.getDateFromCell(start.x, start.y);
+ //var width = DayPilot.Date.daysSpan(e.start(), e.end()) + 1;
+ var boxEnd = boxStart.addDays(width);
+
+ var newStart = boxStart.addTime(startOffset);
+ var newEnd = boxEnd.addTime(endOffset);
+
+ if (calendar._api2()) {
+ // API v2
+ var args = {};
+
+ args.e = e;
+ args.control = calendar;
+ args.newStart = newStart;
+ args.newEnd = newEnd;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onEventResize === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResize(args);
+ });
+
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ switch (calendar.eventResizeHandling) {
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'Update':
+ e.start(newStart);
+ e.end(newEnd);
+ calendar.events.update(e);
+ break;
+ }
+
+ if (typeof calendar.onEventResized === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onEventResized(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.eventResizeHandling) {
+ case 'CallBack':
+ calendar.eventResizeCallBack(e, newStart, newEnd);
+ break;
+ case 'JavaScript':
+ calendar.onEventResize(e, newStart, newEnd);
+ break;
+ }
+ }
+
+ };
+
+
+ this.timeRangeSelectedCallBack = function(start, end, data) {
+
+ var range = {};
+ range.start = start;
+ range.end = end;
+
+ this._callBack2('TimeRangeSelected', data, range);
+ };
+
+ this._timeRangeSelectedDispatch = function(start, end) {
+ if (this._api2()) {
+ var args = {};
+ args.control = calendar;
+ args.start = start;
+ args.end = end;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onTimeRangeSelect === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelect(args);
+ });
+
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ }
+
+ if (typeof calendar.onTimeRangeSelected === 'function') {
+ calendar._angular.apply(function() {
+ calendar.onTimeRangeSelected(args);
+ });
+ }
+ }
+ else {
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'CallBack':
+ calendar.timeRangeSelectedCallBack(start, end);
+ break;
+ case 'JavaScript':
+ calendar.onTimeRangeSelected(start, end);
+ break;
+ }
+ }
+ };
+
+ this._angular = {};
+ this._angular.scope = null;
+ this._angular.notify = function() {
+ if (calendar._angular.scope) {
+ calendar._angular.scope["$apply"]();
+ }
+ };
+ this._angular.apply = function(f) {
+ f();
+
+ /*
+ if (calendar.angularAutoApply && calendar._angular.scope) {
+ calendar._angular.scope["$apply"](f);
+ }
+ else {
+ f();
+ }*/
+ };
+
+ this.clearSelection = function() {
+ calendar.clearShadow();
+ };
+
+ this.commandCallBack = function(command, data) {
+
+ var params = {};
+ params.command = command;
+
+ this._callBack2('Command', data, params);
+ };
+
+ this.isWeekend = function(date) {
+ date = new DayPilot.Date(date);
+
+ var sunday = 0;
+ var saturday = 6;
+
+ if (date.dayOfWeek() === sunday) {
+ return true;
+ }
+ if (date.dayOfWeek() === saturday) {
+ return true;
+ }
+ return false;
+ };
+
+ this._resolved = {};
+ this._resolved.locale = function() {
+ var found = DayPilot.Locale.find(calendar.locale);
+ if (!found) {
+ return DayPilot.Locale.US;
+ }
+ return found;
+ };
+
+ this._resolved._xssProtectionEnabled = function() {
+ return calendar.xssProtection !== "Disabled";
+ };
+
+
+ var resolved = this._resolved;
+
+ this.debug = function(msg, append) {
+ if (!this.debuggingEnabled) {
+ return;
+ }
+
+ if (!calendar.debugMessages) {
+ calendar.debugMessages = [];
+ }
+ calendar.debugMessages.push(msg);
+
+ if (typeof console !== 'undefined') {
+ console.log(msg);
+ }
+ };
+
+ this.dispose = function() {
+ var c = calendar;
+
+ if (c._disposed) {
+ return;
+ }
+ c._disposed = true;
+
+ c._deleteEvents();
+
+ c.nav.top.removeAttribute("style");
+ c.nav.top.removeAttribute("class");
+ c.nav.top.innerHTML = '';
+ c.nav.top.dp = null;
+
+ c.nav.top.onmousemove = null;
+ c.nav.top = null;
+
+ };
+
+ this.disposed = function() {
+ return this._disposed;
+ };
+
+ this._registerGlobalHandlers = function() {
+ if (!DayPilotMonth.globalHandlers) {
+ DayPilotMonth.globalHandlers = true;
+ DayPilot.re(document, 'mouseup', DayPilotMonth.gMouseUp);
+ }
+ };
+
+ this.loadFromServer = function() {
+ // make sure it has a place to ask
+ if (this.backendUrl || typeof WebForm_DoCallback === 'function') {
+ return (typeof calendar.events.list === 'undefined') || (!calendar.events.list);
+ }
+ else {
+ return false;
+ }
+
+ };
+
+ this._show = function() {
+ if (this.nav.top.style.visibility === 'hidden') {
+ this.nav.top.style.visibility = 'visible';
+ }
+ };
+
+ this.show = function() {
+ calendar.visible = true;
+ calendar.nav.top.style.display = '';
+ };
+
+ this.hide = function() {
+ calendar.visible = false;
+ calendar.nav.top.style.display = 'none';
+ };
+
+ this._loadTop = function() {
+ if (this.id && this.id.tagName) {
+ this.nav.top = this.id;
+ }
+ else if (typeof this.id === "string") {
+ this.nav.top = document.getElementById(this.id);
+ if (!this.nav.top) {
+ throw "DayPilot.Month: The placeholder element not found: '" + id + "'.";
+ }
+ }
+ else {
+ throw "DayPilot.Month() constructor requires the target element or its ID as a parameter";
+ }
+ };
+
+ this._initShort = function() {
+
+ this._prepareRows();
+ this._drawTop();
+ this._drawTable();
+ this._registerGlobalHandlers();
+ this._callBack2('Init'); // load events
+ };
+
+ this._xssTextHtml = function(text, html) {
+
+ if (calendar._resolved._xssProtectionEnabled()) {
+ return DayPilot.Util.escapeTextHtml(text, html);
+ }
+
+ if (!DayPilot.Util.isNullOrUndefined(html)) {
+ return html;
+ }
+ if (DayPilot.Util.isNullOrUndefined(text)) {
+ return "";
+ }
+ return text;
+ };
+
+ this.internal = {};
+ this.internal.loadOptions = this._loadOptions;
+ this.internal.xssTextHtml = calendar._xssTextHtml;
+
+ this.init = function() {
+ this._loadTop();
+
+ var loadFromServer = this.loadFromServer();
+
+ if (loadFromServer) {
+ this._initShort();
+ return;
+ }
+
+ this._prepareRows();
+ this._loadEvents();
+ this._drawTop();
+ this._drawTable();
+ this._show();
+ this._drawEvents();
+
+ this._registerGlobalHandlers();
+
+ this.fireAfterRenderDetached(null, false);
+
+ this._initialized = true;
+
+ return this;
+ };
+
+ this.Init = this.init;
+
+ // API compatibility (common)
+ Object.defineProperty(this, 'durationBarVisible', { get: function() { return calendar.eventBarVisible; } });
+
+ this._loadOptions(options);
+ };
+
+ DayPilotMonth.gMouseUp = function(ev) {
+
+ if (DayPilotMonth.movingEvent) {
+ var src = DayPilotMonth.movingEvent;
+
+ if (!src.event) {
+ return;
+ }
+ if (!src.event.calendar) {
+ return;
+ }
+ if (!src.event.calendar.shadow) {
+ return;
+ }
+ if (!src.event.calendar.shadow.start) {
+ return;
+ }
+
+ // load ref
+ var calendar = DayPilotMonth.movingEvent.event.calendar;
+ var e = DayPilotMonth.movingEvent.event;
+ var start = calendar.shadow.start;
+ var position = calendar.shadow.position;
+ var offset = DayPilotMonth.movingEvent.offset;
+
+ // cleanup
+ calendar.clearShadow();
+ DayPilotMonth.movingEvent = null;
+
+ var ev = ev || window.event;
+
+ // fire the event
+ calendar._eventMoveDispatch(e, start.x, start.y, offset, ev, position);
+
+ ev.cancelBubble = true;
+ if (ev.stopPropagation) {
+ ev.stopPropagation();
+ }
+ DayPilotMonth.movingEvent = null;
+ return false;
+ }
+ else if (DayPilotMonth.resizingEvent) {
+ var src = DayPilotMonth.resizingEvent;
+
+ if (!src.event) {
+ return;
+ }
+ if (!src.event.calendar) {
+ return;
+ }
+ if (!src.event.calendar.shadow) {
+ return;
+ }
+ if (!src.event.calendar.shadow.start) {
+ return;
+ }
+
+ // load ref
+ var calendar = DayPilotMonth.resizingEvent.event.calendar;
+
+ var e = DayPilotMonth.resizingEvent.event;
+ var start = calendar.shadow.start;
+ var width = calendar.shadow.width;
+
+ // cleanup
+ calendar.clearShadow();
+ DayPilotMonth.resizingEvent = null;
+
+ // fire the event
+ calendar._eventResizeDispatch(e, start, width);
+
+ ev.cancelBubble = true;
+ DayPilotMonth.resizingEvent = null;
+ return false;
+ }
+ else if (DayPilotMonth.timeRangeSelecting) {
+ if (DayPilotMonth.timeRangeSelecting.moved) {
+ var sel = DayPilotMonth.timeRangeSelecting;
+ var calendar = sel.root;
+
+ var start = new DayPilot.Date(calendar.getDateFromCell(sel.from.x, sel.from.y));
+ var end = start.addDays(sel.width);
+ calendar._timeRangeSelectedDispatch(start, end);
+
+ calendar.clearShadow();
+ }
+ DayPilotMonth.timeRangeSelecting = null;
+ }
+ };
+
+ // publish the API
+
+ // current
+ DayPilot.Month = DayPilotMonth.Month;
+
+ // experimental jQuery bindings
+ if (typeof jQuery !== 'undefined') {
+ (function($) {
+ $.fn.daypilotMonth = function(options) {
+ var first = null;
+ var j = this.each(function() {
+ if (this.daypilot) { // already initialized
+ return;
+ };
+
+ var daypilot = new DayPilot.Month(this.id);
+ this.daypilot = daypilot;
+ for (name in options) {
+ daypilot[name] = options[name];
+ }
+ daypilot.Init();
+ if (!first) {
+ first = daypilot;
+ }
+ });
+ if (this.length === 1) {
+ return first;
+ }
+ else {
+ return j;
+ }
+ };
+ })(jQuery);
+ }
+
+ (function registerAngularModule() {
+
+ var app = DayPilot.am();
+
+ if (!app) {
+ return;
+ }
+
+ app.directive("daypilotMonth", ['$parse', function($parse) {
+ return {
+ "restrict": "E",
+ "template": "",
+ "replace": true,
+ "link": function (scope, element, attrs) {
+
+ var calendar = new DayPilot.Month(element[0]);
+ calendar._angular.scope = scope;
+ calendar.init();
+
+ var oattr = attrs["id"];
+ if (oattr) {
+ scope[oattr] = calendar;
+ }
+
+ // save DayPilot.Calendar object in the specified variable
+ var pas = attrs["publishAs"];
+ if (pas) {
+ var getter = $parse(pas);
+ var setter = getter.assign;
+ setter(scope, calendar);
+ }
+
+ // bind event handlers from attributes starting with "on"
+ for (var name in attrs) {
+ if (name.indexOf("on") === 0) { // event handler
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ scope["$apply"](function() {
+ f(scope, {"args": args});
+ });
+ };
+ })(name);
+ }
+ }
+
+ var watch = scope["$watch"];
+ var config = attrs["config"] || attrs["daypilotConfig"];
+ var events = attrs["events"] || attrs["daypilotEvents"];
+
+ //var watch = scope["$watch"];
+
+ watch.call(scope, config, function (value) {
+ for (var name in value) {
+ calendar[name] = value[name];
+ }
+ calendar.update();
+ }, true);
+
+ watch.call(scope, events, function(value) {
+ calendar.events.list = value;
+ calendar.update();
+ }, true);
+
+ }
+ };
+ }]);
+ })();
+
+
+ if (typeof Sys !== 'undefined' && Sys.Application && Sys.Application.notifyScriptLoaded) {
+ Sys.Application.notifyScriptLoaded();
+ }
+
+
+})();
diff --git a/static/src/js/src/daypilot-navigator.src.js b/static/src/js/src/daypilot-navigator.src.js
new file mode 100644
index 0000000..c13bfc6
--- /dev/null
+++ b/static/src/js/src/daypilot-navigator.src.js
@@ -0,0 +1,1977 @@
+/*
+Copyright © 2024 Annpoint, s.r.o.
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+
+-------------------------------------------------------------------------
+
+NOTE: Requires the following acknowledgement (see also NOTICE):
+This software includes DayPilot (https://www.daypilot.org).
+*/
+
+if (typeof DayPilot === 'undefined') {
+ var DayPilot = {};
+}
+
+if (typeof DayPilot.Global === 'undefined') {
+ DayPilot.Global = {};
+}
+
+(function() {
+
+ if (typeof DayPilot.Navigator !== 'undefined' && DayPilot.Navigator.nav) {
+ return;
+ }
+
+ var DayPilotNavigator = {};
+ DayPilot.Navigator = function(id, options) {
+ this.v = '2024.3.539-lite';
+ var calendar = this;
+ this.id = id;
+ this.api = 2;
+ this.isNavigator = true;
+
+ this.autoFocusOnClick = true;
+ this.weekStarts = 'Auto'; // 0 = Sunday, 1 = Monday, ... 'Auto' = according to locale
+ this.selectMode = 'Day'; // day/week/month/none
+ this.titleHeight = 30;
+ this.dayHeaderHeight = 30;
+ this.bound = null;
+ this.cellWidth = 30;
+ this.cellHeight = 30;
+ this.cssClassPrefix = "navigator_default";
+ this.freeHandSelectionEnabled = false;
+ this.selectionStart = new DayPilot.Date().getDatePart(); // today
+ this.selectionEnd = null;
+ this.selectionDay = null;
+ this.showMonths = 1;
+ this.skipMonths = 1;
+ this.command = "navigate";
+ this.year = new DayPilot.Date().getYear();
+ this.month = new DayPilot.Date().getMonth() + 1;
+ this.showWeekNumbers = false;
+ this.weekNumberAlgorithm = 'Auto';
+ this.rowsPerMonth = 'Six'; // Six, Auto
+ this.orientation = "Vertical";
+ this.locale = "en-us";
+ this.rtl = false;
+ this.visible = true;
+
+ this.timeRangeSelectedHandling = "Bind";
+ this.visibleRangeChangedHandling = "Enabled";
+
+ this.onVisibleRangeChange = null;
+ this.onVisibleRangeChanged = null;
+ this.onTimeRangeSelect = null;
+ this.onTimeRangeSelected = null;
+
+ this.nav = {};
+
+ this._cache = {};
+
+ this._prepare = function() {
+
+ this.root.dp = this;
+
+ this.root.className = this._prefixCssClass('_main');
+
+ if (this.orientation === "Horizontal") {
+ this.root.style.width = this.showMonths * (resolved.cellWidth() * 7 + this._weekNumberWidth()) + 'px';
+ this.root.style.height = (this.cellHeight*6 + this.titleHeight + this.dayHeaderHeight) + 'px';
+ }
+ else {
+ this.root.style.width = (resolved.cellWidth() * 7 + this._weekNumberWidth()) + 'px';
+ }
+ //this.root.style.height = (this.showMonths*(this.cellHeight*6 + this.titleHeight + this.dayHeaderHeight)) + 'px';
+
+ if (this.rtl) {
+ this.root.style.direction = "rtl";
+ }
+
+ this.root.style.position = "relative";
+
+ if (!this.visible) {
+ this.root.style.display = "none";
+ }
+
+ var vsph = document.createElement("input");
+ vsph.type = 'hidden';
+ vsph.name = calendar.id + "_state";
+ vsph.id = vsph.name;
+ //vsph.value = result.VsUpdate;
+ this.root.appendChild(vsph);
+ this.state = vsph;
+
+ if (!this.startDate) {
+ this.startDate = DayPilot.Date.fromYearMonthDay(this.year, this.month);
+ }
+ else { // make sure it's the first day
+ this.startDate = new DayPilot.Date(this.startDate).firstDayOfMonth();
+ }
+
+ this.calendars = [];
+ this.selected = [];
+ this.months = [];
+ };
+
+ this._api2 = function() {
+ return calendar.api === 2;
+ };
+
+ this._clearTable = function() {
+ // TODO do something smarter here
+ this.root.innerHTML = '';
+ };
+
+ this._prefixCssClass = function(part) {
+ var prefix = this.theme || this.cssClassPrefix;
+ if (prefix) {
+ return prefix + part;
+ }
+ else {
+ return "";
+ }
+ };
+
+ this._addClass = function(object, name) {
+ var fullName = this._prefixCssClass("_" + name);
+ DayPilot.Util.addClass(object, fullName);
+ };
+
+ this._removeClass = function(object, name) {
+ var fullName = this._prefixCssClass("_" + name);
+ DayPilot.Util.removeClass(object, fullName);
+ };
+
+ this._drawTable = function(j, showLinks) {
+ var month = {};
+ month.cells = [];
+ month.days = [];
+ month.weeks = [];
+
+ var startDate = this.startDate.addMonths(j);
+
+ var showBefore = showLinks.before;
+ var showAfter = showLinks.after;
+
+ var firstOfMonth = startDate.firstDayOfMonth();
+ var first = firstOfMonth.firstDayOfWeek(resolved.weekStarts());
+
+ var last = firstOfMonth.addMonths(1);
+ var days = DayPilot.DateUtil.daysDiff(first, last);
+
+ var rowCount = (this.rowsPerMonth === "Auto") ? Math.ceil(days / 7) : 6;
+ month.rowCount = rowCount;
+ var today = (new DayPilot.Date()).getDatePart();
+
+ var width = resolved.cellWidth() * 7 + this._weekNumberWidth();
+ month.width = width;
+ var height = this.cellHeight * rowCount + this.titleHeight + this.dayHeaderHeight;
+ month.height = height;
+
+ var main = document.createElement("div");
+ main.style.width = (width) + 'px';
+ main.style.height = (height) + 'px';
+
+
+ if (this.orientation === "Horizontal") {
+ main.style.position = "absolute";
+ main.style.left = (width * j) + "px";
+ main.style.top = "0px";
+
+ month.top = 0;
+ month.left = width * j;
+ }
+ else {
+ main.style.position = 'relative';
+
+ var above = j > 0 ? calendar.months[j - 1].top + calendar.months[j - 1].height : 0;
+ month.top = above;
+
+ month.left = 0;
+ }
+
+ main.className = this._prefixCssClass('_month');
+ main.style.cursor = 'default';
+ main.style.MozUserSelect = 'none';
+ main.style.KhtmlUserSelect = 'none';
+ main.style.WebkitUserSelect = 'none';
+
+ main.month = month;
+
+ this.root.appendChild(main);
+
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+
+ // title left
+ var tl = document.createElement("div");
+ tl.style.position = 'absolute';
+ tl.style.left = '0px';
+ tl.style.right = '0px'; // rtl
+ tl.style.top = '0px';
+ tl.style.width = resolved.cellWidth() + 'px';
+ tl.style.height = this.titleHeight + 'px';
+ tl.style.lineHeight = this.titleHeight + 'px';
+ //tl.style.textAlign = 'left';
+ tl.setAttribute("unselectable", "on");
+ tl.className = this._prefixCssClass('_titleleft');
+ if (showLinks.left) {
+ tl.style.cursor = 'pointer';
+ tl.innerHTML = "<";
+ tl.onclick = this._clickLeft;
+ }
+ main.appendChild(tl);
+ this.tl = tl;
+
+ // title center
+ var ti = document.createElement("div");
+ ti.style.position = 'absolute';
+ ti.style.left = resolved.cellWidth() + 'px';
+ ti.style.top = '0px';
+ ti.style.width = (resolved.cellWidth() * 5 + this._weekNumberWidth()) + 'px';
+ ti.style.height = this.titleHeight + 'px';
+ ti.style.lineHeight = this.titleHeight + 'px';
+ //ti.style.textAlign = 'center';
+ ti.setAttribute("unselectable", "on");
+ ti.className = this._prefixCssClass('_title');
+ ti.innerHTML = resolved.locale().monthNames[startDate.getMonth()] + ' ' + startDate.getYear();
+ main.appendChild(ti);
+ this.ti = ti;
+
+ // title right
+ var tr = document.createElement("div");
+ tr.style.position = 'absolute';
+ tr.style.left = (resolved.cellWidth() * 6 + this._weekNumberWidth()) + 'px';
+ tr.style.right = (resolved.cellWidth() * 6 + this._weekNumberWidth()) + 'px'; // rtl
+ tr.style.top = '0px';
+ tr.style.width = resolved.cellWidth() + 'px';
+ tr.style.height = this.titleHeight + 'px';
+ tr.style.lineHeight = this.titleHeight + 'px';
+ //tr.style.textAlign = 'right';
+ tr.setAttribute("unselectable", "on");
+ tr.className = this._prefixCssClass('_titleright');
+ if (showLinks.right) {
+ tr.style.cursor = 'pointer';
+ tr.innerHTML = ">";
+ tr.onclick = this._clickRight;
+ }
+ main.appendChild(tr);
+ this.tr = tr;
+
+
+ var xOffset = this._weekNumberWidth();
+ if (this.showWeekNumbers) {
+ for (var y = 0; y < rowCount; y++) {
+ var day = first.addDays(y * 7);
+ var weekNumber = null;
+ switch (this.weekNumberAlgorithm) {
+ case "Auto":
+ weekNumber = (resolved.weekStarts() === 1) ? day.weekNumberISO() : day.weekNumber();
+ break;
+ case "US":
+ weekNumber = day.weekNumber();
+ break;
+ case "ISO8601":
+ weekNumber = day.weekNumberISO();
+ break;
+ default:
+ throw "Unknown weekNumberAlgorithm value.";
+ }
+
+ var dh = document.createElement("div");
+ dh.style.position = 'absolute';
+ dh.style.left = (0) + 'px';
+ dh.style.right = (0) + 'px';
+ dh.style.top = (y * this.cellHeight + totalHeaderHeight) + 'px';
+ dh.style.width = resolved.cellWidth() + 'px';
+ dh.style.height = this.cellHeight + 'px';
+ dh.style.lineHeight = this.cellHeight + 'px';
+ //dh.style.textAlign = 'right';
+ dh.setAttribute("unselectable", "on");
+ dh.className = this._prefixCssClass('_weeknumber');
+ //dh.innerHTML = "" + weekNumber + "";
+ dh.innerHTML = "" + weekNumber + "";
+ main.appendChild(dh);
+ month.weeks.push(dh);
+ }
+ }
+
+
+ for (var x = 0; x < 7; x++) {
+ month.cells[x] = [];
+
+ // day header
+ var dh = document.createElement("div");
+ dh.style.position = 'absolute';
+ dh.style.left = (x * resolved.cellWidth() + xOffset) + 'px';
+ dh.style.right = (x * resolved.cellWidth() + xOffset) + 'px'; // rtl
+ dh.style.top = this.titleHeight + 'px';
+ dh.style.width = resolved.cellWidth() + 'px';
+ dh.style.height = this.dayHeaderHeight + 'px';
+ dh.style.lineHeight = this.dayHeaderHeight + 'px';
+ //dh.style.textAlign = 'right';
+ dh.setAttribute("unselectable", "on");
+ dh.className = this._prefixCssClass('_dayheader');
+ dh.innerHTML = "" + this._getDayName(x) + "";
+ main.appendChild(dh);
+ month.days.push(dh);
+
+ for (var y = 0; y < rowCount; y++) {
+ var day = first.addDays(y * 7 + x);
+
+ var isSelected = this._isSelected(day) && this._selectModeLowerCase() !== 'none';
+ var isCurrentMonth = day.firstDayOfMonth() === startDate;
+ var isPrevMonth = day < startDate;
+ var isNextMonth = day >= startDate.addMonths(1);
+
+ if (this._selectModeLowerCase() === "month") {
+ isSelected = isSelected && isCurrentMonth;
+ }
+ else if (this._selectModeLowerCase() === "day") {
+ isSelected = isSelected && (isCurrentMonth || (showBefore && isPrevMonth) || (showAfter && isNextMonth));
+ }
+ else if (this._selectModeLowerCase() === "week") {
+ //var sd = this.selectionDay || this.selectionStart;
+ var isSelectionCurrentMonth = day.firstDayOfMonth() === startDate;
+ isSelected = isSelected && (isSelectionCurrentMonth || (showBefore && isPrevMonth) || (showAfter && isNextMonth));
+ }
+
+ var dc = document.createElement("div");
+ month.cells[x][y] = dc;
+
+ var cellPos = calendar._cellRelativeCoords(x, y);
+ var left = cellPos.x;
+ var top = cellPos.y;
+
+ dc.day = day;
+ dc.x = x;
+ dc.y = y;
+ dc.left = left;
+ dc.top = top;
+ dc.isCurrentMonth = isCurrentMonth;
+ dc.isNextMonth = isNextMonth;
+ dc.isPrevMonth = isPrevMonth;
+ dc.showBefore = showBefore;
+ dc.showAfter = showAfter;
+ dc.className = this._prefixCssClass((isCurrentMonth ? '_day' : '_dayother'));
+ calendar._addClass(dc, "cell");
+ if (day.getTime() === today.getTime() && isCurrentMonth) {
+ this._addClass(dc, 'today');
+ }
+ if (day.dayOfWeek() === 0 || day.dayOfWeek() === 6) {
+ this._addClass(dc, 'weekend');
+ }
+
+ dc.style.position = 'absolute';
+ dc.style.left = (left) + 'px';
+ dc.style.right = (left) + 'px'; // rtl
+ dc.style.top = (top) + 'px';
+ dc.style.width = resolved.cellWidth() + 'px';
+ dc.style.height = this.cellHeight + 'px';
+ dc.style.lineHeight = this.cellHeight + 'px'; // vertical alignment
+ //dc.style.textAlign = 'right';
+ //dc.style.border = '1px solid white';
+
+ var inner = document.createElement("div");
+ inner.style.position = 'absolute';
+ inner.className = (day.getTime() === today.getTime() && isCurrentMonth) ? this._prefixCssClass('_todaybox') : this._prefixCssClass('_daybox');
+ calendar._addClass(inner, "cell_box");
+ //inner.style.boxSizing = "border-box";
+ inner.style.left = '0px';
+ inner.style.top = '0px';
+ inner.style.right = '0px';
+ inner.style.bottom = '0px';
+ //inner.style.width = (this.cellWidth - 2) + 'px';
+ //inner.style.height = (this.cellHeight - 2) + 'px';
+ dc.appendChild(inner);
+
+ /*
+ if (isCurrentMonth) {
+ dc.style.cursor = 'pointer';
+ }
+ */
+
+ var cell = null;
+ if (this.cells && this.cells[day.toStringSortable()]) {
+ cell = this.cells[day.toStringSortable()];
+ }
+
+ if (typeof calendar.onBeforeCellRender === "function") {
+ var args = {};
+ args.cell = cell || {};
+ args.cell.day = day;
+ args.cell.isCurrentMonth = isCurrentMonth;
+ args.cell.isToday = day.getTime() === today.getTime() && isCurrentMonth;
+ args.cell.isWeekend = day.dayOfWeek() === 0 || day.dayOfWeek() === 6;
+ if (cell) {
+ args.cell.html = cell.html || day.getDay();
+ args.cell.cssClass = cell.css;
+ }
+ else {
+ args.cell.html = day.getDay();
+ args.cell.cssClass = null;
+ }
+
+ calendar.onBeforeCellRender(args);
+
+ cell = args.cell;
+ }
+
+ if (cell) {
+ DayPilot.Util.addClass(dc, cell.cssClass || cell.css);
+ }
+
+ //var span = null;
+ if (isCurrentMonth || (showBefore && isPrevMonth) || (showAfter && isNextMonth)) {
+ var text = document.createElement("div");
+ text.innerHTML = day.getDay();
+ text.style.position = "absolute";
+ text.style.left = '0px';
+ text.style.top = '0px';
+ text.style.right = '0px';
+ text.style.bottom = '0px';
+ calendar._addClass(text, "cell_text");
+
+ // dc.style.cursor = 'pointer';
+ dc.isClickable = true;
+
+ if (cell && cell.html) {
+ text.innerHTML = cell.html;
+ }
+
+ dc.appendChild(text);
+ }
+
+
+ dc.setAttribute("unselectable", "on");
+
+ dc.onclick = this._cellClick;
+
+ main.appendChild(dc);
+
+ if (isSelected) {
+ calendar._cellSelect(main, x, y);
+ this.selected.push(dc);
+ }
+
+ }
+ }
+
+ var line = document.createElement("div");
+ line.style.position = 'absolute';
+ line.style.left = '0px';
+ line.style.top = (totalHeaderHeight - 2) + 'px';
+ line.style.width = (resolved.cellWidth() * 7 + this._weekNumberWidth()) + 'px';
+ line.style.height = '1px';
+ line.style.fontSize = '1px';
+ line.style.lineHeight = '1px';
+ line.className = this._prefixCssClass("_line");
+
+ main.appendChild(line);
+ this.months.push(month);
+ };
+
+ this._cellRelativeCoords = function(x, y) {
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+ var xOffset = this._weekNumberWidth();
+ var left = x * resolved.cellWidth() + xOffset;
+ var top = y * this.cellHeight + totalHeaderHeight;
+
+ return {
+ "x": left,
+ "y": top
+ };
+
+ };
+
+ this._cellSelect = function(main, x, y) {
+ var div = main.month.cells[x][y];
+
+ calendar._addClass(div, 'select');
+
+ /*
+ if (div.selectDiv) {
+ return;
+ }
+
+ var xOffset = this._weekNumberWidth();
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+
+ // overlay select
+ var od = document.createElement("div");
+ od.style.position = 'absolute';
+ od.style.left = (x * this.cellWidth + xOffset) + 'px';
+ od.style.top = (y * this.cellHeight + totalHeaderHeight) + 'px';
+ od.style.width = this.cellWidth + 'px';
+ od.style.height = this.cellHeight + 'px';
+ od.className = calendar._prefixCssClass("_selected_overlay");
+
+ div.selectDiv = od;
+
+ main.appendChild(od);
+ */
+ };
+
+ this._cellUnselect = function(main, x, y) {
+ var div = main.month.cells[x][y];
+
+ calendar._removeClass(div, 'select');
+
+ /*
+ DayPilot.de(div.selectDiv);
+ div.selectDiv = null;
+ */
+ };
+
+ this._weekNumberWidth = function() {
+ if (this.showWeekNumbers) {
+ return resolved.cellWidth();
+ }
+ return 0;
+ };
+
+ this._updateFreeBusy = function() {
+ if (!this.items) {
+ return;
+ }
+
+ for (var j = 0; j < this.showMonths; j++) {
+ for (var x = 0; x < 7; x++) {
+ for (var y = 0; y < 6; y++) {
+ var cell = this.months[j].cells[x][y];
+ if (!cell) {
+ continue;
+ }
+ if (this.items[cell.day.toStringSortable()] === 1) {
+ this._addClass(cell, 'busy');
+ this._removeClass(cell, 'free');
+ }
+ else {
+ this._removeClass(cell, 'busy');
+ this._addClass(cell, 'free');
+ }
+ }
+ }
+ }
+ };
+
+ this._saveState = function() {
+ var s = {};
+ s.startDate = calendar.startDate;
+ s.selectionStart = calendar.selectionStart;
+ s.selectionEnd = calendar.selectionEnd.addDays(1);
+ calendar.state.value = JSON.stringify(s);
+ };
+
+ this._selectModeLowerCase = function() {
+ var selectMode = this.selectMode || "";
+ return selectMode.toLowerCase();
+ };
+
+ this._adjustSelection = function() {
+
+ var input = this.selectionDay || this.selectionStart; // selectionDay is preferred
+ if (!input) {
+ input = DayPilot.Date.today();
+ }
+ input = new DayPilot.Date(input); // make sure it's DayPilot.Date
+
+ // ignores selectionEnd
+ // uses selectMode
+ switch (this._selectModeLowerCase()) {
+ case 'day':
+ this.selectionStart = input;
+ this.selectionDay = input;
+ this.selectionEnd = input;
+ break;
+ case 'week':
+ this.selectionDay = input;
+ this.selectionStart = input.firstDayOfWeek(resolved.weekStarts());
+ this.selectionEnd = this.selectionStart.addDays(6);
+ break;
+ case 'month':
+ this.selectionDay = input;
+ this.selectionStart = input.firstDayOfMonth();
+ this.selectionEnd = this.selectionStart.lastDayOfMonth();
+ break;
+ case 'none':
+ this.selectionEnd = input;
+ break;
+ default:
+ throw "Unknown selectMode value.";
+ }
+
+ };
+
+ this._postponedSelect = null;
+
+ // options.dontFocus, options.dontNotify
+ this.select = function(a1, a2, a3) {
+
+ var a2IsDate = a2 && (a2 instanceof DayPilot.Date || typeof a2 === "string");
+ var a2IsOptions = (a2 && typeof a2 === "object") || typeof a2 === "boolean";
+
+ var date1 = a1;
+ var date2 = a2IsDate ? a2 : null;
+ var options = a2IsOptions ? a2 : a3;
+
+ if (!this._initialized) {
+ this._postponedSelect = {
+ "date1": date1,
+ "date2": date2,
+ "options": options
+ };
+ return;
+ }
+
+ var focus = true;
+ var notify = true; // fire the timeRangeSelected event
+
+ if (options && typeof options === "object") {
+ if (options.dontFocus) {
+ focus = false;
+ }
+ if (options.dontNotify) {
+ notify = false;
+ }
+ }
+ else if (typeof options === "boolean") {
+ focus = !options;
+ }
+
+ var originalStart = this.selectionStart;
+ var originalEnd = this.selectionEnd;
+
+ this.selectionStart = new DayPilot.Date(date1).getDatePart();
+ this.selectionDay = this.selectionStart;
+
+ var startChanged = false;
+ if (focus) {
+
+ var newStart = this.startDate;
+ if (this.selectionStart < this._activeStart() || this.selectionStart >= this._activeEnd()) {
+ newStart = this.selectionStart.firstDayOfMonth();
+ }
+
+ if (newStart.toStringSortable() !== this.startDate.toStringSortable()) {
+ startChanged = true;
+ }
+
+ this.startDate = newStart;
+ }
+
+ if (date2 && calendar.freeHandSelectionEnabled) {
+ calendar.selectionEnd = new DayPilot.Date(date2);
+ }
+ else {
+ this._adjustSelection();
+ }
+
+ // redraw
+ this._clearTable();
+ this._prepare();
+ this._drawMonths();
+ this._updateFreeBusy();
+ this._saveState();
+
+ if (notify && (!originalStart.equals(this.selectionStart) || !originalEnd.equals(this.selectionEnd))) {
+ //alert('time range');
+ this._timeRangeSelectedDispatch();
+ }
+
+ if (startChanged) {
+ //alert('visible range');
+ this._visibleRangeChangedDispatch();
+ }
+ };
+
+ this.update = function(options) {
+
+ calendar._loadOptions(options);
+
+/*
+ if (!calendar._initialized) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Navigator instance that hasn't been initialized yet.");
+ }
+*/
+ if (!this._initialized) {
+ return;
+ }
+
+ if (calendar._disposed) {
+ throw new DayPilot.Exception("You are trying to update a DayPilot.Navigator instance that has been disposed.");
+ }
+
+ calendar._clearCache();
+
+ var os = {
+ "day": calendar.selectionDay,
+ "start": calendar.selectionStart,
+ "end": calendar.selectionEnd
+ };
+
+ calendar._update();
+
+ if (os.start !== calendar.selectionStart || os.end != calendar.selectionEnd || os.day !== calendar.selectionDay) {
+ calendar._timeRangeSelectedDispatch();
+ }
+ };
+
+ this._update = function () {
+ // redraw
+ this._clearTable();
+ this._prepare();
+ this._adjustSelection();
+ this._drawMonths();
+ this._loadEvents();
+ this._updateFreeBusy();
+ this._saveState();
+
+ if (this.visible) {
+ this.show();
+ }
+ else {
+ this.hide();
+ }
+ };
+
+ this._clearCache = function() {
+ calendar._cache = {};
+ };
+
+ this._specialHandling = null;
+ this._loadOptions = function(options) {
+ if (!options) {
+ return;
+ }
+ var specialHandling = {
+ "events": {
+ "preInit": function() {
+ var events = this.data || [];
+ if (DayPilot.isArray(events.list)) {
+ calendar.events.list = events.list;
+ }
+ else {
+ calendar.events.list = events;
+ }
+ }
+ }
+ };
+ this._specialHandling = specialHandling;
+
+ for (var name in options) {
+ if (specialHandling[name]) {
+ var item = specialHandling[name];
+ item.data = options[name];
+ if (item.preInit) {
+ item.preInit();
+ }
+ }
+ else {
+ calendar[name] = options[name];
+ }
+ }
+
+ };
+
+ this._postInit = function() {
+ var specialHandling = this._specialHandling;
+ for (var name in specialHandling) {
+ var item = specialHandling[name];
+ if (item.postInit) {
+ item.postInit();
+ }
+ }
+ };
+
+
+ this._callBack2 = function(action, data, parameters) {
+ var envelope = {};
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + JSON.stringify(envelope);
+
+ var context = null;
+ if (this.backendUrl) {
+ DayPilot.request(this.backendUrl, this._callBackResponse, commandstring, this._ajaxError);
+ }
+ else {
+ WebForm_DoCallback(this.uniqueID, commandstring, this._updateView, context, this.callbackError, true);
+ }
+
+ };
+
+ this._ajaxError = function(req) {
+ if (typeof calendar.onAjaxError === 'function') {
+ var args = {};
+ args.request = req;
+ calendar.onAjaxError(args);
+ }
+ else if (typeof calendar.ajaxError === 'function') { // backwards compatibility
+ calendar.ajaxError(req);
+ }
+ };
+
+ this._callBackResponse = function(response) {
+ calendar._updateView(response.responseText);
+ };
+
+ this._postBack2 = function(action, data, parameters) {
+ var envelope = {};
+ envelope.action = action;
+ envelope.parameters = parameters;
+ envelope.data = data;
+ envelope.header = this._getCallBackHeader();
+
+ var commandstring = "JSON" + JSON.stringify(envelope);
+ __doPostBack(calendar.uniqueID, commandstring);
+ };
+
+ this._getCallBackHeader = function() {
+ var h = {};
+ h.v = this.v;
+ h.startDate = this.startDate;
+ h.selectionStart = this.selectionStart;
+ h.showMonths = this.showMonths;
+ return h;
+ };
+
+ this._listen = function(action, data) {
+ if (action === 'refresh') {
+ this._visibleRangeChangedDispatch();
+ }
+ };
+
+ this._getDayName = function(i) {
+ var x = i + resolved.weekStarts();
+ if (x > 6) {
+ x -= 7;
+ }
+ return resolved.locale().dayNamesShort[x];
+
+ };
+
+ this._isSelected = function(date) {
+ if (this.selectionStart === null || this.selectionEnd === null) {
+ return false;
+ }
+
+ if (this.selectionStart.getTime() <= date.getTime() && date.getTime() <= this.selectionEnd.getTime()) {
+ return true;
+ }
+
+ return false;
+ };
+
+ this._getMonthFromCoords = function(coords) {
+ var month = 0;
+ //debugger;
+ for (var i = 0; i < calendar.months.length; i++) {
+ var m = calendar.months[i];
+ if (!m) {
+ return null;
+ }
+ if (coords.x < m.left || m.width < coords.x ) {
+ return null;
+ }
+
+ var height = calendar.months[i].height;
+ if (m.top <= coords.y && coords.y < m.top + m.height) {
+ return i;
+ }
+ }
+ return null;
+
+ };
+
+ this._getPosition = function(ev) {
+
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ var monthIndex = calendar._getMonthFromCoords(coords);
+ if (monthIndex === null) {
+ return null;
+ }
+ var month = calendar.months[monthIndex];
+
+ var totalHeaderHeight = this.titleHeight + this.dayHeaderHeight;
+ if (month.top <= coords.y && coords.y < month.top + totalHeaderHeight) {
+ return {
+ "month": monthIndex,
+ "x": 0,
+ "y": 0,
+ "coords": coords,
+ "header": true
+ };
+ }
+
+ for (var x = 0; x < month.cells.length; x++) {
+ for (var y = 0; y < month.cells[x].length; y++) {
+ var cell = month.cells[x][y];
+ var top = cell.top + month.top;
+ var left = cell.left + month.left;
+ if (left <= coords.x && coords.x < left + calendar.cellWidth) {
+ if (top <= coords.y && coords.y < top + calendar.cellHeight) {
+ return {
+ "month": monthIndex,
+ "x": x,
+ "y": y,
+ "coords": coords
+ };
+ }
+ }
+ }
+ }
+
+ return null;
+ };
+
+ this._onTopMouseDown = function(ev) {
+ var freeHandSelection = calendar.freeHandSelectionEnabled;
+ if (!freeHandSelection) {
+ return;
+ }
+ var start = calendar._getPosition(ev);
+ if (start && !start.header) {
+ ps.start = start;
+ }
+ var cell = calendar.months[start.month].cells[start.x][start.y];
+
+ ev.preventDefault();
+
+ };
+
+ this._onTopMouseMove = function(ev) {
+ if (!ps.start) {
+ return;
+ }
+ var end = calendar._getPosition(ev);
+ if (ps.end) {
+ ps.end = end;
+ }
+ else if (end) {
+ var requiredDistance = 3;
+ var distance = DayPilot.distance(ps.start.coords, end.coords);
+
+ if (distance > requiredDistance) {
+ ps.end = end;
+ }
+ }
+
+ if (ps.end) {
+ ps.clear();
+ ps.draw();
+ }
+ //ps.end = end;
+ };
+
+ this._preselection = {};
+ var ps = this._preselection;
+ ps.start = null;
+
+ ps.drawCell = function(pos) {
+ var month = calendar.months[pos.month];
+
+ var cellPos = calendar._cellRelativeCoords(pos.x, pos.y);
+
+ var top = month.top + cellPos.y;
+ var left = month.left + cellPos.x;
+
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+ div.style.left = left + "px";
+ div.style.top = top + "px";
+ div.style.height = calendar.cellHeight + "px";
+ div.style.width = calendar.cellWidth + "px";
+ div.style.backgroundColor = "#ccc";
+ div.style.opacity = 0.5;
+ div.style.cursor = "default";
+
+ calendar.nav.preselection.appendChild(div);
+ ps.cells.push(div);
+ };
+
+ ps.clear = function() {
+ if (!ps.cells) {
+ return;
+ }
+
+ for(var i = 0; i < ps.cells.length; i++) {
+ calendar.nav.preselection.removeChild(ps.cells[i]);
+ }
+ ps.cells = [];
+ };
+
+ ps.draw = function() {
+ var ordered = ps.ordered();
+
+ var position = new Position(ordered.start);
+ var end = ordered.end;
+
+ if (!end) {
+ return;
+ }
+
+ if (end === ps.end && end.header) {
+ if (end.month > 0) {
+ end.month -= 1;
+ var month = calendar.months[end.month];
+ end.x = 6;
+ end.y = month.rowCount - 1;
+ }
+ }
+
+ ps.cells = [];
+
+ while (!position.is(end)) {
+ ps.drawCell(position);
+ var next = new Position(position).next();
+ if (!next) {
+ return;
+ }
+ position.month = next.month;
+ position.x = next.x;
+ position.y = next.y;
+ }
+
+ // the last one
+ ps.drawCell(position);
+
+ };
+
+ ps.ordered = function() {
+ //var position = new Position(start);
+ var start = ps.start;
+ var end = ps.end;
+
+ var result = {};
+ if (!end || new Position(start).before(end)) {
+ result.start = start;
+ result.end = end;
+ }
+ else {
+ result.start = end;
+ result.end = start;
+ }
+ return result;
+ };
+
+ var Position = function(month, x, y) {
+
+ if (month instanceof Position) {
+ return month;
+ }
+ if (typeof month === "object") {
+ var ref = month;
+ this.month = ref.month;
+ this.x = ref.x;
+ this.y = ref.y;
+ }
+ else {
+ this.month = month;
+ this.x = x;
+ this.y = y;
+ }
+ this.is = function(ref) {
+ return this.month === ref.month && this.x === ref.x && this.y === ref.y;
+ };
+
+ this.next = function() {
+ var start = this;
+ if (start.x < 6) {
+ return {
+ "month": start.month,
+ "x": start.x + 1,
+ "y": start.y
+ };
+ }
+ var month = calendar.months[start.month];
+ if (start.y < month.rowCount - 1) {
+ return {
+ "month": start.month,
+ "x": 0,
+ "y": start.y + 1
+ };
+ }
+ if (start.month < calendar.months.length - 1) {
+ return {
+ "month": start.month + 1,
+ "x": 0,
+ "y": 0
+ };
+ }
+ return null;
+ };
+
+ this.visible = function() {
+ var cell = this.cell();
+ if (cell.isCurrentMonth) {
+ return true;
+ }
+ if (cell.isPrevMonth && cell.showBefore) {
+ return true;
+ }
+ if (cell.isNextMonth && cell.showAfter) {
+ return true;
+ }
+ return false;
+ };
+
+ this.nextVisible = function() {
+ var pos = this;
+ while (!pos.visible()) {
+ var next = pos.next();
+ if (!next) {
+ return null;
+ }
+ pos = new Position(next);
+ }
+ return pos;
+ };
+
+ this.previous = function() {
+ var start = this;
+ if (start.x > 0) {
+ return {
+ "month": start.month,
+ "x": start.x - 1,
+ "y": start.y
+ };
+ }
+ var month = calendar.months[start.month];
+ if (start.y > 0) {
+ return {
+ "month": start.month,
+ "x": 6,
+ "y": start.y - 1
+ };
+ }
+ if (start.month > 0) {
+ var m = calendar.months[start.month - 1];
+ return {
+ "month": start.month - 1,
+ "x": 6,
+ "y": m.rowCount - 1
+ };
+ }
+ return null;
+
+ };
+
+ this.previousVisible = function() {
+ var pos = this;
+ while (!pos.visible()) {
+ var previous = pos.previous();
+ if (!previous) {
+ return null;
+ }
+ pos = new Position(previous);
+ }
+ return pos;
+ };
+
+ this.cell = function() {
+ return calendar.months[this.month].cells[this.x][this.y];
+ };
+
+ this.date = function() {
+ return this.cell().day;
+ };
+
+ this.before = function(ref) {
+ var thisDate = this.date();
+ var refDate = new Position(ref).date();
+ return thisDate < refDate;
+ };
+ };
+
+
+ this._cellClick = function(ev) {
+ var main = this.parentNode;
+ var month = this.parentNode.month;
+
+ var x = this.x;
+ var y = this.y;
+ var day = month.cells[x][y].day;
+
+ if (!month.cells[x][y].isClickable) {
+ return;
+ }
+
+ calendar.clearSelection();
+
+ calendar.selectionDay = day;
+
+ var day = calendar.selectionDay;
+ switch (calendar._selectModeLowerCase()) {
+ case 'none':
+ //var s = month.cells[x][y];
+ calendar.selectionStart = day;
+ calendar.selectionEnd = day;
+ break;
+ case 'day':
+ if (calendar.autoFocusOnClick) {
+ var start = day;
+ if (day < calendar._activeStart() ||
+ day >= calendar._activeEnd()) {
+ calendar.select(day);
+ return;
+ }
+ }
+
+ var s = month.cells[x][y];
+ calendar._cellSelect(main, x, y);
+ //calendar._addClass(s, 'select');
+ calendar.selected.push(s);
+ calendar.selectionStart = s.day;
+ calendar.selectionEnd = s.day;
+
+ break;
+ case 'week':
+ if (calendar.autoFocusOnClick) {
+ var start = month.cells[0][y].day;
+ var end = month.cells[6][y].day;
+ if (start.firstDayOfMonth() === end.firstDayOfMonth()) {
+ if (start < calendar._activeStart() ||
+ end >= calendar._activeEnd()) {
+ calendar.select(day);
+ return;
+ }
+ }
+ }
+ for (var j = 0; j < 7; j++) {
+ calendar._cellSelect(main, j, y);
+ //calendar._addClass(month.cells[j][y], 'select');
+ calendar.selected.push(month.cells[j][y]);
+ }
+ calendar.selectionStart = month.cells[0][y].day;
+ calendar.selectionEnd = month.cells[6][y].day;
+
+ break;
+ case 'month':
+ if (calendar.autoFocusOnClick) {
+ var start = day;
+ if (day < calendar._activeStart() ||
+ day >= calendar._activeEnd()) {
+ calendar.select(day);
+ return;
+ }
+ }
+
+ var start = null;
+ var end = null;
+ for (var y = 0; y < 6; y++) {
+ for (var x = 0; x < 7; x++) {
+ var s = month.cells[x][y];
+ if (!s) {
+ continue;
+ }
+ if (s.day.getYear() === day.getYear() && s.day.getMonth() === day.getMonth()) {
+ calendar._cellSelect(main, x, y);
+ //calendar._addClass(s, 'select');
+ calendar.selected.push(s);
+ if (start === null) {
+ start = s.day;
+ }
+ end = s.day;
+ }
+ }
+ }
+ calendar.selectionStart = start;
+ calendar.selectionEnd = end;
+ break;
+ default:
+ throw 'unknown selectMode';
+ }
+
+ calendar._saveState();
+
+ calendar._timeRangeSelectedDispatch();
+ };
+
+ this._timeRangeSelectedDispatch = function(options) {
+ var start = calendar.selectionStart;
+ var end = calendar.selectionEnd.addDays(1);
+ var days = DayPilot.DateUtil.daysDiff(start, end);
+ var day = calendar.selectionDay;
+
+ options = options || {};
+
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.day = day;
+ args.days = days;
+ args.mode = options.mode || calendar.selectMode;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onTimeRangeSelect === 'function') {
+ calendar.onTimeRangeSelect(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'Bind':
+ if (typeof bound === "object") {
+ var selection = {};
+ selection.start = start;
+ selection.end = end;
+ selection.days = days;
+ selection.day = day;
+ bound.commandCallBack(calendar.command, selection);
+ }
+ break;
+ case 'None':
+ break;
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end, day);
+ break;
+ }
+
+ if (typeof calendar.onTimeRangeSelected === 'function') {
+ calendar.onTimeRangeSelected(args);
+ }
+
+ }
+ else {
+ switch (calendar.timeRangeSelectedHandling) {
+ case 'Bind':
+ if (typeof bound === "object") {
+ var selection = {};
+ selection.start = start;
+ selection.end = end;
+ selection.days = days;
+ selection.day = day;
+ bound.commandCallBack(calendar.command, selection);
+ }
+ break;
+ case 'JavaScript':
+ calendar.onTimeRangeSelected(start, end, day);
+ break;
+ case 'None':
+ break;
+ case 'PostBack':
+ calendar.timeRangeSelectedPostBack(start, end, day);
+ break;
+ }
+ }
+
+
+ };
+
+ this.timeRangeSelectedPostBack = function(start, end, data, day) {
+ var params = {};
+ params.start = start;
+ params.end = end;
+ params.day = day;
+
+ this._postBack2('TimeRangeSelected', data, params);
+ };
+
+ this._clickRight = function(ev) {
+ calendar._moveMonth(calendar.skipMonths);
+ };
+
+ this._clickLeft = function(ev) {
+ calendar._moveMonth(-calendar.skipMonths);
+ };
+
+ this._moveMonth = function(i) {
+ this.startDate = this.startDate.addMonths(i);
+ this._clearTable();
+ this._prepare();
+ this._drawMonths();
+
+ this._saveState();
+
+ this._visibleRangeChangedDispatch();
+ this._updateFreeBusy();
+ };
+
+ this._activeStart = function() {
+ return calendar.startDate.firstDayOfMonth();
+ };
+
+ this._activeEnd = function() {
+ return calendar.startDate.firstDayOfMonth().addMonths(this.showMonths);
+ };
+
+ this.visibleStart = function() {
+ return calendar.startDate.firstDayOfMonth().firstDayOfWeek(resolved.weekStarts());
+ };
+
+ this.visibleEnd = function() {
+ return calendar.startDate.firstDayOfMonth().addMonths(this.showMonths - 1).firstDayOfWeek(resolved.weekStarts()).addDays(42);
+ };
+
+ this._visibleRangeChangedDispatch = function() {
+ var start = this.visibleStart();
+ var end = this.visibleEnd();
+
+ if (calendar._api2()) {
+
+ var args = {};
+ args.start = start;
+ args.end = end;
+ args.preventDefault = function() {
+ this.preventDefault.value = true;
+ };
+
+ if (typeof calendar.onVisibleRangeChange === 'function') {
+ calendar.onVisibleRangeChange(args);
+ if (args.preventDefault.value) {
+ return;
+ }
+ }
+
+ // now perform the default builtin action
+ switch (this.visibleRangeChangedHandling) {
+ case "CallBack":
+ this.visibleRangeChangedCallBack(null);
+ break;
+ case "PostBack":
+ this.visibleRangeChangedPostBack(null);
+ break;
+ case "Disabled":
+ break;
+ }
+
+ if (typeof calendar.onVisibleRangeChanged === 'function') {
+ calendar.onVisibleRangeChanged(args);
+ }
+
+ }
+ else {
+ switch (this.visibleRangeChangedHandling) {
+ case "CallBack":
+ this.visibleRangeChangedCallBack(null);
+ break;
+ case "PostBack":
+ this.visibleRangeChangedPostBack(null);
+ break;
+ case "JavaScript":
+ this.onVisibleRangeChanged(start, end);
+ break;
+ case "Disabled":
+ break;
+ }
+ }
+
+ /*
+ switch (this.visibleRangeChangedHandling) {
+ case "CallBack":
+ this.visibleRangeChangedCallBack(null);
+ break;
+ case "PostBack":
+ this.visibleRangeChangedPostBack(null);
+ break;
+ case "JavaScript":
+ this.onVisibleRangeChanged(start, end);
+ break;
+ case "Disabled":
+ break;
+ }
+ */
+ };
+
+
+ this.visibleRangeChangedCallBack = function(data) {
+ var parameters = {};
+ this._callBack2("Visible", data, parameters);
+ };
+
+ this.visibleRangeChangedPostBack = function(data) {
+ var parameters = {};
+ this._postBack2("Visible", data, parameters);
+ };
+
+ this._updateView = function(result, context) {
+ var result = JSON.parse(result);
+ calendar.items = result.Items;
+ calendar.cells = result.Cells;
+
+ if (calendar.cells) {
+ calendar.update();
+ }
+ else {
+ calendar._updateFreeBusy();
+ }
+ };
+
+ this._drawMonths = function() {
+ for (var j = 0; j < this.showMonths; j++) {
+ var showLinks = this._getShowLinks(j);
+ this._drawTable(j, showLinks);
+ }
+
+ this.root.style.height = this._getHeight() + "px";
+
+ this.nav.preselection = document.createElement("div");
+ this.nav.preselection.style.position = "absolute";
+ this.nav.preselection.style.left = "0px";
+ this.nav.preselection.style.top = "0px";
+ this.root.appendChild(this.nav.preselection);
+
+ /*
+ var div = document.createElement("div");
+ div.style.clear = "left";
+ div.style.height = "0px";
+ div.style.width = "0px";
+ this.root.appendChild(div);
+ */
+
+ };
+
+
+ this._getHeight = function() {
+ if (this.orientation === "Horizontal") {
+ var max = 0;
+ for (var i = 0; i < this.months.length; i++) {
+ var month = this.months[i];
+ if (month.height > max) {
+ max = month.height;
+ }
+ }
+ return max;
+ }
+ else {
+ var total = 0;
+ for (var i = 0; i < this.months.length; i++) {
+ var month = this.months[i];
+ //total += this.showMonths*(this.cellHeight*month.rowCount + this.titleHeight + this.dayHeaderHeight);
+ total += month.height;
+ }
+ return total;
+ }
+ };
+
+ this._getShowLinks = function(j) {
+ if (this.internal.showLinks) {
+ return this.internal.showLinks;
+ }
+
+ var showLinks = {};
+ showLinks.left = (j === 0);
+ showLinks.right = (j === 0);
+ showLinks.before = j === 0;
+ showLinks.after = j === this.showMonths - 1;
+
+ if (this.orientation === "Horizontal") {
+ showLinks.right = (j === this.showMonths - 1);
+ }
+
+ return showLinks;
+ };
+
+ // not used at the moment - no internal changes to data
+
+ this._angular = {};
+ this._angular.scope = null;
+ this._angular.notify = function() {
+ if (calendar._angular.scope) {
+ calendar._angular.scope["$apply"]();
+ }
+ };
+
+ this.internal = {};
+ this.internal.loadOptions = calendar._loadOptions;
+ // ASP.NET
+ this.internal.initialized = function() {
+ return calendar._initialized;
+ };
+
+ this._resolved = {};
+ var resolved = this._resolved;
+
+ resolved.locale = function() {
+ return DayPilot.Locale.find(calendar.locale);
+ };
+
+ resolved.weekStarts = function() {
+ if (calendar.weekStarts === 'Auto') {
+ var locale = resolved.locale();
+ if (locale) {
+ return locale.weekStarts;
+ }
+ else {
+ return 0; // Sunday
+ }
+ }
+ else {
+ return calendar.weekStarts;
+ }
+ };
+
+ resolved.cellWidth = function() {
+ if (calendar._cache.cellWidth) {
+ return calendar._cache.cellWidth;
+ }
+ var width = calendar._getDimensionsFromCss("_cell_dimensions").width;
+ if (!width) {
+ width = calendar.cellWidth;
+ }
+ calendar._cache.cellWidth = width;
+ return width;
+ };
+
+ this.clearSelection = function() {
+ for (var j = 0; j < this.selected.length; j++) {
+ //this._removeClass(this.selected[j], 'select');
+ var div = this.selected[j];
+ calendar._cellUnselect(div.parentNode, div.x, div.y);
+ }
+ this.selected = [];
+ };
+
+ this._isShortInit = function() {
+ // make sure it has a place to ask
+ if (this.backendUrl) {
+ return (typeof calendar.items === 'undefined') || (!calendar.items);
+ }
+ else {
+ return false;
+ }
+ };
+
+ this.events = {};
+
+ this._loadEvents = function() {
+ if (!DayPilot.isArray(this.events.list)) {
+ return;
+ }
+
+ this.items = {};
+
+ for(var i = 0; i < this.events.list.length; i++) {
+ var e = this.events.list[i];
+ if (e.hidden) {
+ continue;
+ }
+ var days = this._eventDays(e);
+ for(var name in days) {
+ this.items[name] = 1;
+ }
+ }
+ };
+
+ this._getDimensionsFromCss = function(className) {
+ var div = document.createElement("div");
+ div.style.position = "absolute";
+ div.style.top = "-2000px";
+ div.style.left = "-2000px";
+ div.className = this._prefixCssClass(className);
+
+ var container = calendar.root || document.body;
+
+ container.appendChild(div);
+ var height = div.offsetHeight;
+ var width = div.offsetWidth;
+ container.removeChild(div);
+
+ var result = {};
+ result.height = height;
+ result.width = width;
+ return result;
+ };
+
+ this._eventDays = function(e) {
+ var start = new DayPilot.Date(e.start);
+ var end = new DayPilot.Date(e.end);
+
+ var days = {};
+
+ var d = start.getDatePart();
+ while (d.getTime() <= end.getTime()) {
+ days[d.toStringSortable()] = 1;
+ d = d.addDays(1);
+ }
+
+ return days;
+ };
+
+ this.show = function() {
+ calendar.visible = true;
+ calendar.root.style.display = '';
+ };
+
+ this.hide = function() {
+ calendar.visible = false;
+ calendar.root.style.display = 'none';
+ };
+
+ this._loadTop = function() {
+ if (this.id && this.id.tagName) {
+ this.nav.top = this.id;
+ }
+ else if (typeof this.id === "string") {
+ this.nav.top = document.getElementById(this.id);
+ if (!this.nav.top) {
+ throw "DayPilot.Navigator: The placeholder element not found: '" + id + "'.";
+ }
+ }
+ else {
+ throw "DayPilot.Navigator() constructor requires the target element or its ID as a parameter";
+ }
+
+ this.root = this.nav.top;
+
+ };
+
+ this.init = function() {
+ this._loadTop();
+
+ if (this.root.dp) { // already initialized
+ return;
+ }
+
+ this._adjustSelection();
+ this._prepare();
+ this._drawMonths();
+ this._loadEvents();
+ this._updateFreeBusy();
+ this._registerDispose();
+ this._registerTopHandlers();
+ this._registerGlobalHandlers();
+
+ //this.select(this.selectionStart);
+
+ var loadFromServer = this._isShortInit();
+ if (loadFromServer) {
+ this._visibleRangeChangedDispatch(); // TODO change to "Init"?
+ }
+ this._initialized = true;
+ this._postInit();
+
+ if (this._postponedSelect) {
+ var params = this._postponedSelect;
+ this.select(params.date1, params.date2, params.options);
+ this._postponedSelect = null;
+ }
+
+ return this;
+ };
+
+ this._registerTopHandlers = function() {
+ calendar.nav.top.onmousedown = this._onTopMouseDown;
+ calendar.nav.top.onmousemove = this._onTopMouseMove;
+ };
+
+ this._registerGlobalHandlers = function() {
+ DayPilot.re(document, 'mouseup', calendar._gMouseUp);
+ };
+
+ this._gMouseUp = function(ev) {
+ if (ps.start && ps.end) {
+ var coords = DayPilot.mo3(calendar.nav.top, ev);
+
+ // click, cancel
+ if (coords.x === ps.start.coords.x && coords.y === ps.start.coords.y) {
+ ps.start = null;
+ ps.clear();
+ return;
+ }
+
+ ps.clear();
+
+ var ordered = ps.ordered();
+
+ ordered.start = new Position(ordered.start).nextVisible();
+ ordered.end = new Position(ordered.end).previousVisible();
+
+
+ calendar.selectionDay = new Position(ordered.start).date();
+ calendar.selectionStart = calendar.selectionDay;
+ calendar.selectionEnd = new Position(ordered.end).date();
+
+ ps.start = null;
+ ps.end = null;
+
+ // redraw
+ calendar._clearTable();
+ calendar._prepare();
+ calendar._drawMonths();
+ calendar._updateFreeBusy();
+ calendar._saveState();
+
+ var notify = true;
+ if (notify) {
+ calendar._timeRangeSelectedDispatch({"mode": "FreeHand"});
+ }
+
+ }
+
+ // clear in either case
+ ps.start = null;
+ ps.end = null;
+
+ };
+
+ this.dispose = function() {
+ var c = calendar;
+
+ if (!c.root) {
+ return;
+ }
+
+ c.root.removeAttribute("style");
+ c.root.removeAttribute("class");
+ c.root.dp = null;
+ c.root.innerHTML = null;
+ c.root = null;
+
+ c._disposed = true;
+
+ };
+
+ this._registerDispose = function() {
+ //var root = document.getElementById(id);
+ this.root.dispose = this.dispose;
+ };
+
+ this.Init = this.init;
+
+ this._loadOptions(options);
+ };
+
+ // jQuery plugin
+ if (typeof jQuery !== 'undefined') {
+ (function($) {
+ $.fn.daypilotNavigator = function(options) {
+ var first = null;
+ var j = this.each(function() {
+ if (this.daypilot) { // already initialized
+ return;
+ };
+
+ var daypilot = new DayPilot.Navigator(this.id);
+ this.daypilot = daypilot;
+ for (var name in options) {
+ daypilot[name] = options[name];
+ }
+ daypilot.Init();
+ if (!first) {
+ first = daypilot;
+ }
+ });
+ if (this.length === 1) {
+ return first;
+ }
+ else {
+ return j;
+ }
+ };
+ })(jQuery);
+ }
+
+ // AngularJS plugin
+ (function registerAngularModule() {
+ var app = DayPilot.am();
+
+ if (!app) {
+ return;
+ }
+
+
+ app.directive("daypilotNavigator", ['$parse', function($parse) {
+ return {
+ "restrict": "E",
+ "template": "",
+ "compile": function compile(element, attrs) {
+ element.replaceWith(this["template"].replace("{{id}}", attrs["id"]));
+
+ return function link(scope, element, attrs) {
+ var calendar = new DayPilot.Navigator(element[0]);
+ calendar._angular.scope = scope;
+ calendar.init();
+
+ var oattr = attrs["id"];
+ if (oattr) {
+ scope[oattr] = calendar;
+ }
+
+ // save DayPilot.Calendar object in the specified variable
+ var pas = attrs["publishAs"];
+ if (pas) {
+ var getter = $parse(pas);
+ var setter = getter.assign;
+ setter(scope, calendar);
+ }
+
+ // bind event handlers from attributes starting with "on"
+ for (var name in attrs) {
+ if (name.indexOf("on") === 0) { // event handler
+ var apply = DayPilot.Util.shouldApply(name);
+
+ if (apply) {
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ scope["$apply"](function() {
+ f(scope, {"args": args});
+ });
+ };
+ })(name);
+ }
+ else {
+ (function(name) {
+ calendar[name] = function(args) {
+ var f = $parse(attrs[name]);
+ f(scope, {"args": args});
+ };
+ })(name);
+ }
+
+ }
+ }
+
+ var watch = scope["$watch"];
+ var config = attrs["config"] || attrs["daypilotConfig"];
+ var events = attrs["events"] || attrs["daypilotEvents"];
+
+ watch.call(scope, config, function (value, oldVal) {
+ for (var name in value) {
+ calendar[name] = value[name];
+ }
+ calendar.update();
+ }, true);
+
+ watch.call(scope, events, function(value) {
+ calendar.events.list = value;
+ calendar._loadEvents();
+ calendar._updateFreeBusy();
+ }, true);
+
+ };
+ }
+ };
+ }]);
+
+ })();
+
+ if (typeof Sys !== 'undefined' && Sys.Application && Sys.Application.notifyScriptLoaded) {
+ Sys.Application.notifyScriptLoaded();
+ }
+
+
+})();
diff --git a/views/company_view.xml b/views/company_view.xml
index be745af..aee751a 100755
--- a/views/company_view.xml
+++ b/views/company_view.xml
@@ -9,38 +9,33 @@
-
-
-
-
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
diff --git a/views/dss_ads.xml b/views/dss_ads.xml
index faf23b0..764be49 100755
--- a/views/dss_ads.xml
+++ b/views/dss_ads.xml
@@ -9,7 +9,9 @@