GridEditor.vue 31.8 KB
Newer Older
Anton's avatar
Intial  
Anton committed
1 2 3
<template lang="pug">

  div.cell-grid-container
4

5
    q-window-resize-observable(@resize="updateGridDimensions")
Anton's avatar
Intial  
Anton committed
6

christianrhansen's avatar
christianrhansen committed
7 8 9 10
    // ------------------------------------------------------------------------------------------ cached new cell helper
    // q-page-sticky.z-top(v-touch-pan="moveCachedCell", v-if="cachedNewCell", position="top-right")
    .fixed.z-top(v-touch-pan="moveCachedCell", v-if="cachedNewCell",
    :style="{top: touch.position.top - 16 - 16 + 'px', left: touch.position.left - 16 - 16 + 'px'}")
11
      .bg-frosted-glass.text-dark.q-px-md.q-py-sm.q-ma-md.shadow-4(:style="{width: gridDimensions.full.cell.width + 'px', height: gridDimensions.full.cell.height + 'px', borderRadius: 0}")
christianrhansen's avatar
christianrhansen committed
12
        | {{ cachedNewCell.type }} Cell:
13
        br
christianrhansen's avatar
christianrhansen committed
14
        | {{ cachedNewCell.value }}
15 16 17 18
        q-btn.bg-dark.text-white.on-right(@click="", round, flat, size="sm")
          q-icon(name="clear")

    // ------------------------------------------------------------------------------------------------------------ grid
Anton's avatar
Intial  
Anton committed
19
    div.cell-grid(
christianrhansen's avatar
christianrhansen committed
20
    @click="event => {test(event)}",
21 22 23 24 25 26
    @dragenter="handleGridDragOver",
    @dragover="handleGridDragOver",
    @dragleave="handleGridDragEnd",
    @drop="handleGridDrop",
    @contextmenu="handleGridContextMenu",
    :style="gridStyle")
Anton's avatar
Intial  
Anton committed
27 28 29 30

      q-context-menu(ref="gridmenu")
        q-list(link, separator, no-border, style="min-width: 150px; max-height: 300px;")
          q-item(
31 32 33 34 35
          v-for="action in gridContextMenuActions",
          :key="action.label",
          v-close-overlay,
          @click.native="event => {action.handler(event)}")
            q-item-main(:label="action.label")
Anton's avatar
Intial  
Anton committed
36 37 38

      template(v-if="!resizingGrid")

39
        template(v-for="(annotation, index) in annotations")
christianrhansen's avatar
christianrhansen committed
40 41

          //----- cell
Anton's avatar
Intial  
Anton committed
42
          .cell-item(
43 44 45 46 47 48 49 50 51 52 53 54 55
          v-if="!annotationUIStates[annotation._uuid] || !annotationUIStates[annotation._uuid].beingDragged",
          draggable="true",
          @click.prevent="event => {handleCellTouch(event, annotation)}",
          @dragstart="event => {handleCellDragStart(event, annotation)}",
          @dragend="event => {handleCellDragEnd(event, annotation)}",
          @contextmenu="handleCellContextMenu",
          :style="getAnnotationStyle(annotation)",
          :class="getAnnotationClasses(annotation._uuid, 'cell-item')",
          :key="`cell-${index}`")

            //----- edit-/close-button
            // TODO: find a more elegant solution
            .desktop-only
Mathias Bär's avatar
Mathias Bär committed
56
              q-btn.edit-button.absolute-top-right(
57 58 59 60 61 62 63 64 65
              @click.prevent="event => {handleCellEditClick(event, annotation)}",
              :class="getAnnotationClasses(annotation._uuid, 'editing')",
              style="top: 8px; right: 8px;",
              :icon="annotationUIStates[annotation._uuid].editing ? 'close' : 'edit'", flat, round, size="md"
              )
            .mobile-only
              q-btn.edit-button.absolute.fit.bg-transparent(
              @click.prevent="event => {touchMobileCell(event, annotation)}", flat)

66
            //----- selecting cells disabled because it has no use currently
67 68 69 70 71 72 73 74
            // switch with cell component below to re-enable it
            //cell(
              @click.native.prevent="event => {handleCellClick(event, annotation)}", :annotation="annotation", :preview="true")
            cell(
            :annotation="annotation",
            :preview="true"
            )

75
            //----- resize-handler
76
            .desktop-only.cell-item-resize-handle(
77 78 79 80 81 82
            draggable="true",
            @dragstart="event => {handleCellResizerDragStart(event, annotation)}",
            @dragend="event => {handleCellResizerDragEnd(event, annotation)}",
            @dragexit="event => {handleCellResizerDragEnd(event, annotation)}")
              q-icon(name="network cell")

83 84 85
            //----- context menu for cells (desktop only)
            // TODO: needs revision
            q-context-menu.desktop-only
86 87 88 89 90 91 92
              q-list(link, separator, no-border, style="min-width: 150px; max-height: 300px;")
                q-item(
                v-for="action in cellContextMenuActions",
                :key="action.label",
                v-close-overlay,
                @click.native="event => {action.handler(event, annotation)}")
                  q-item-main(:label="action.label")
Anton's avatar
Intial  
Anton committed
93

94
        //----- temporary cell when pulling a new cell into grid (desktop)
95
        template(v-for="(tmpCell, index) in tmpObjects")
96
          .cell-item.cell-item-tmp(:style="getAnnotationStyle(tmpCell)", :key="`cell-tmp-${index}`")
Anton's avatar
Intial  
Anton committed
97 98
            cell(:cell="tmpCell")

christianrhansen's avatar
christianrhansen committed
99
      // ---------------------------------------------------------------------------------------------------------------
Anton's avatar
Intial  
Anton committed
100
      template(v-else)
101
        .cell-item(:style="getAnnotationStyle({x:0,y:0,width:1,height:1})", key="cell-grid-resizer")
Anton's avatar
Intial  
Anton committed
102
          div.cell-item-resize-handle(
103 104 105 106 107
          draggable="true",
          @dragstart="event => {handleGridResizerDragStart(event)}",
          @dragend="event => {handleGridResizerDragEnd(event)}",
          @dragexit="event => {handleGridResizerDragEnd(event)}")
            q-icon(name="network cell")
Anton's avatar
Intial  
Anton committed
108

Mathias Bär's avatar
Mathias Bär committed
109
      //template(v-if="!isMobile")
110 111 112 113 114
        .fixed-top-right(style="right:18px; top:68px", v-if="!$store.state.mosys.showSources")
          q-btn(round, color="primary", small, @click="handleGridButtonClickEdit", style="margin-right: 0.5em")
            q-icon(name="add")
          q-btn(round, color="primary", small, @click="$router.push(`/mosys/grids/${$route.params.uuid}`)")
            q-icon(name="remove red eye")
Mathias Bär's avatar
Mathias Bär committed
115
      //template(v-if="isMobile")
Mathias Bär's avatar
Mathias Bär committed
116 117 118 119 120
        .fixed-top-right.q-mt-sm(v-if="!$store.state.mosys.showSources", style="z-index: 10000; padding-top: 3px;")
          q-btn.q-mr-sm(round, color="primary", size="sm", @click="handleGridButtonClickEdit")
            q-icon(name="add")
          q-btn.q-mr-md(round, color="primary", size="sm", @click="$router.push(`/mosys/grids/${$route.params.uuid}`)")
            q-icon(name="remove red eye")
Anton's avatar
Intial  
Anton committed
121

122 123 124 125 126
    // --------------------------------------------------------------------------------------------------------- buttons

    q-page-sticky.z-top.q-mb-md.backdrop-filter.shadow-1.q-mx-md.overflow-hidden(v-if="showEditingCells", position="bottom-right")
      div.row.text-center.text-dark.round-borders.full-width(style="border-bottom: 1px solid #bbb;")
        .col-3
127
          q-btn.full-width(@click="setEditMode('edit')", :class="{'bg-primary text-white' : editMode === 'edit'}", flat)
128 129
            q-icon(name="edit")
        .col-3
130
          q-btn.full-width(@click="setEditMode('resize')", :class="{'bg-primary text-white' : editMode === 'resize'}", flat)
131 132
            q-icon.flip-vertical(name="photo_size_select_small")
        .col-3
133
          q-btn.full-width(@click="setEditMode('move')", :class="{'bg-primary text-white' : editMode === 'move'}", flat)
134 135
            q-icon(name="open_with")
        .col-3
136
          q-btn.full-width(@click="setEditMode('delete')", :class="{'bg-primary text-white' : editMode === 'delete'}", flat)
137 138 139 140 141 142 143 144 145
            q-icon(name="delete")

      div.full-width

        //----- edit
        div#edit-content-mobile(v-if="editMode === 'edit'")
          grid-editor-editing-cells-mobile

        //----- move
christianrhansen's avatar
christianrhansen committed
146
        div(v-if="editMode === 'move'", :class="{'q-pa-sm' : editMode}")
147
          .text-center
148
            q-btn.bg-dark(@click="mobileCellMove(mobileSelectedCell, 0, -1)", round, flat, size="md")
149 150
              q-icon.rotate-90(name="keyboard_backspace")
          .text-center
151
            q-btn.bg-dark(@click="mobileCellMove(mobileSelectedCell, -1, 0)", round, flat, size="md")
152
              q-icon(name="keyboard_backspace")
153
            q-btn.invisible.text-dark(round, flat, size="md")
154
              q-icon.flip-vertical(name="photo_size_select_small")
155
            q-btn.bg-dark(@click="mobileCellMove(mobileSelectedCell, 1, 0)", round, flat, size="md")
156 157
              q-icon.rotate-180(name="keyboard_backspace")
          .text-center
158
            q-btn.bg-dark(@click="mobileCellMove(mobileSelectedCell, 0, 1)", round, flat, size="md")
159 160 161
              q-icon.rotate-270(name="keyboard_backspace")

        //----- resize
christianrhansen's avatar
christianrhansen committed
162
        div(v-if="editMode === 'resize'", :class="{'q-pa-sm' : editMode}")
163
          .text-center
164
            q-btn.bg-dark(@click="mobileCellResize(mobileSelectedCell, 0, -1)", round, flat, size="md")
165 166
              q-icon(name="remove")
          .text-center
167
            q-btn.bg-dark(@click="mobileCellResize(mobileSelectedCell, -1, 0)", round, flat, size="md")
168
              q-icon(name="remove")
169
            q-btn.invisible.text-dark(round, flat, size="md")
170
              q-icon.rotate-45(name="zoom_out_map")
171
            q-btn.bg-dark(@click="mobileCellResize(mobileSelectedCell, 1, 0)", round, flat, size="md")
172 173
              q-icon(name="add")
          .text-center
174
            q-btn.bg-dark(@click="mobileCellResize(mobileSelectedCell, 0, 1)", round, flat, size="md")
175 176 177 178 179 180 181 182
              q-icon(name="add")

        //----- delete
        div(v-if="editMode === 'delete'", :class="{'q-pa-md' : editMode}")
          .text-center
            .text-dark
              | {{ $t('labels.delete_cell') }}
            .q-mt-md
183 184
              q-btn.bg-dark.text-white.q-mx-sm(@click="event => {handleCellContextMenuDelete(event, mobileSelectedCell)}",
              flat, round, size="md")
185 186
                q-icon(name="check")

Anton's avatar
Intial  
Anton committed
187 188 189 190
</template>

<script>
  import Cell from './Cell'
191 192
  import { userHasFeature } from 'mbjs-quasar/src/lib'
  import { mapGetters } from 'vuex'
193
  import GridEditorEditingCellsMobile from './/GridEditorEditingCellsMobile'
Anton's avatar
Intial  
Anton committed
194 195 196 197 198 199

  const nullImage = new Image()
  nullImage.src = ''

  export default {
    components: {
200 201
      Cell,
      GridEditorEditingCellsMobile
Anton's avatar
Intial  
Anton committed
202
    },
203
    props: ['gridUuid', 'tabsAreOpen'],
Anton's avatar
Intial  
Anton committed
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
    data () {
      return {
        gridContextMenuActions: {
          insert_column_left: {
            label: 'Insert Column Left',
            handler: this.handleGridContextMenuInsertColumnLeft
          },
          delete_column: {
            label: 'Delete Column',
            handler: this.handleGridContextMenuDeleteColumn
          },
          insert_row_above: {
            label: 'Insert Row Above',
            handler: this.handleGridContextMenuInsertRowAbove
          },
          delete_row: {
            label: 'Delete Row',
            handler: this.handleGridContextMenuDeleteRow
          },
          edit_grid_dimensions: {
            label: 'Change Grid',
            handler: () => { this.resizingGrid = !this.resizingGrid }
          }
        },
        grid: undefined,
229
        annotations: undefined,
230
        tmpObjects: [],
231
        annotationUIStates: {},
Anton's avatar
Anton committed
232
        gridDimensions: { gridWidth: 0, gridHeight: 0, cellWidth: 0, cellHeight: 0 },
Anton's avatar
Intial  
Anton committed
233
        contextMenuClickPosition: {},
234
        resizingGrid: false,
christianrhansen's avatar
christianrhansen committed
235 236
        mobileSelectedCell: undefined,
        touch: {position: {top: undefined, left: undefined}}
Anton's avatar
Intial  
Anton committed
237 238
      }
    },
239 240
    computed: {
      ...mapGetters({
christianrhansen's avatar
christianrhansen committed
241
        cachedNewCell: 'mosys/getNewCell',
242
        user: 'auth/getUserState',
Mathias Bär's avatar
Mathias Bär committed
243 244
        isMobile: 'globalSettings/getIsMobile',
        // editingCells: 'mosys/getEditingCells'
245
        showEditingCells: 'mosys/getShowEditingCells',
246 247
        scrollPositionCache: 'mosys/getScrollPositionCache',
        editMode: 'mosys/getEditMode'
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270
      }),
      cellContextMenuActions () {
        const actions = {
          delete: {
            label: 'Delete',
            handler: this.handleCellContextMenuDelete
          },
          insert_column_left: {
            label: 'Insert Column Left',
            handler: this.handleGridContextMenuInsertColumnLeft
          },
          insert_row_above: {
            label: 'Insert Row Above',
            handler: this.handleGridContextMenuInsertRowAbove
          }
        }
        if (userHasFeature(this.user, 'cssediting')) {
          actions.edit_css_classname = {
            label: 'Edit CSS class name',
            handler: this.handleCellContextMenuEditCSS
          }
        }
        return actions
271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286
      },
      gridStyle () {
        if (!this.gridDimensions || !this.gridDimensions.full) return {}
        // TODO: fix mobile grid editor view
        const cell = this.gridDimensions.full.cell
        return {
          width: `${this.gridDimensions.full.width}px`,
          height: '100%',
          'grid-auto-columns': `${cell.width}px`,
          'grid-auto-rows': `${cell.height}px`,
          'background-image': `url("data:image/svg+xml;utf8,` +
            `<svg xmlns='http://www.w3.org/2000/svg' width='100%' height='100%'><defs>` +
            `<pattern id='smallGrid' width='${cell.width}' height='${cell.height}' patternUnits='userSpaceOnUse'>` +
            `<path d='M ${cell.width} 0 L 0 0 0 ${cell.height}' fill='none' stroke='gray' stroke-width='0.5'/>` +
            `</pattern></defs><rect width='100%' height='100%' fill='url(%23smallGrid)' /></svg>")`
        }
287 288
      }
    },
Anton's avatar
Intial  
Anton committed
289 290
    async mounted () {
      await this.fetchData()
291
      this.resetScrollPosition()
Anton's avatar
Intial  
Anton committed
292 293
    },
    watch: {
Anton's avatar
Anton committed
294
      annotations () {
295
        this.updateAnnotationUIStates()
Anton's avatar
Intial  
Anton committed
296 297 298 299 300 301
      },
      gridMetadata () {
        this.updateGridDimensions()
      },
      async gridUuid () {
        await this.fetchData()
Mathias Bär's avatar
Mathias Bär committed
302 303
      },
      showEditingCells (val) {
304
        // console.log('show editing cells', val)
Mathias Bär's avatar
Mathias Bär committed
305 306 307
        if (val === false) {
          this.updateAnnotationUIStates()
        }
308 309 310
      },
      tabsAreOpen () {
        this.resetScrollPosition()
Anton's avatar
Intial  
Anton committed
311 312 313
      }
    },
    methods: {
christianrhansen's avatar
christianrhansen committed
314 315 316 317 318 319 320 321 322 323 324 325 326 327
      moveCachedCell (obj) {
        this.touch.position = {top: obj.position.top, left: obj.position.left}
      },
      test (event) {
        console.log('--->>>', this.getGridPositionForEvent(event))
        this.addMobileCell(event)
      },
      addMobileCell (event) {
        if (this.cachedNewCell) {
          console.log('###--- ', this.getGridPositionForEvent(event))
          this.$store.commit('mosys/cacheNewCell', undefined)
          this.handleGridDrop(event)
        }
      },
328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371
      touchMobileCell (event, cell) {
        Object.keys(this.annotationUIStates).filter((k) => {
          if (k === cell._uuid && this.annotationUIStates[k].editing) console.log(k, cell._uuid)
          else this.annotationUIStates[k].editing = false
        })
        this.annotationUIStates[cell._uuid].editing = !this.annotationUIStates[cell._uuid].editing
        this.updateEditingCells()
        this.$root.$emit('mosys_saveScrollPosition')
        // this.handleCellEditClick(event, annotation)
      },
      async mobileCellMove (annotation, _x, _y) {
        let
          parsed = annotation.target.selector.parse(),
          sliced = parsed.xywh.slice(0, 2),
          x = sliced[0],
          y = sliced[1]
        x += _x
        y += _y
        let target = this.grid.get2DArea([x, y], parsed.xywh.slice(2))
        annotation.target.selector.value = target.selector.value
        await this.$store.dispatch('annotations/patch', [annotation.id, {
          target: {
            selector: { value: target.selector.value }
          }
        }])
      },
      handleCellTouch (event, annotation) {
        console.log(event, annotation)
        this.mobileSelectedCell = annotation
      },
      async mobileCellResize (annotation, width, height) {
        let
          parsed = annotation.target.selector.parse(),
          [x, y, w, h] = parsed.xywh
        w += width
        h += height
        const value = { xywh: [x, y, w, h] }
        annotation.target.selector.value = value

        await this.$store.dispatch('annotations/patch', [annotation.id, { target: { selector: { value } } }])
      },
      setEditMode (mode) {
        this.$store.commit('mosys/setEditMode', mode)
      },
372 373 374 375
      //
      // DATA
      //

Anton's avatar
Intial  
Anton committed
376 377 378
      async fetchData () {
        if (this.gridUuid) {
          this.grid = await this.$store.dispatch('maps/get', this.gridUuid)
379 380 381 382 383 384 385 386
          if (!Object.keys(this.grid.config).length) {
            this.grid.config = {
              columns: 10,
              rows: 6,
              ratio: 16 / 9.0
            }
            await this.updateGridMetadataStore()
          }
Anton's avatar
Intial  
Anton committed
387
          this.updateGridDimensions()
388 389 390 391 392 393
          const { items } = await this.$store.dispatch('annotations/find', {
            'target.id': this.grid.id,
            'body.purpose': 'linking',
            'body.type': 'Cell'
          })
          this.annotations = items
Anton's avatar
Intial  
Anton committed
394 395
        }
      },
396 397 398
      async updateGridMetadataStore () {
        await this.$store.dispatch('maps/patch', [this.grid.id, { config: this.grid.config }])
        this.updateGridDimensions()
Anton's avatar
Intial  
Anton committed
399
      },
400 401 402 403 404
      updateSelectedCells () {
        const _this = this
        let selectedCells = Object.keys(this.annotationUIStates).filter(k => {
          return _this.annotationUIStates[k].selected
        }).map(k => {
Christian Hansen's avatar
Christian Hansen committed
405 406
          // return _this.cells.find(c => c._uuid === k)
          return _this.annotations.find(c => c._uuid === k)
407 408
        })
        this.$store.commit('mosys/setSelectedCells', selectedCells)
Anton's avatar
Intial  
Anton committed
409
      },
Mathias Bär's avatar
Mathias Bär committed
410 411 412 413 414 415 416 417
      updateEditingCells () {
        const _this = this
        let editingCells = Object.keys(this.annotationUIStates).filter(k => {
          return _this.annotationUIStates[k].editing
        }).map(k => {
          // return _this.cells.find(c => c._uuid === k)
          return _this.annotations.find(c => c._uuid === k)
        })
418 419 420 421 422 423 424
        /*
        console.log('this.annotationUIStates: ', this.annotationUIStates)
        console.log('editingCells: ', editingCells)
        if (this.isMobile) {
          console.log('MOBILE')
        }
        */
Mathias Bär's avatar
Mathias Bär committed
425 426
        this.$store.commit('mosys/setEditingCells', editingCells)
      },
427 428 429 430 431
      updateAnnotationUIStates () {
        let newAnnotationUIStates = {}
        this.annotations.forEach(a => {
          newAnnotationUIStates[a._uuid] = {
            selected: false,
Mathias Bär's avatar
Mathias Bär committed
432
            editing: false,
433
            beingResized: false,
Mathias Bär's avatar
Mathias Bär committed
434
            beginDragged: false,
435
            annotation: a
Anton's avatar
Intial  
Anton committed
436
          }
437 438
        })
        this.annotationUIStates = newAnnotationUIStates
Anton's avatar
Intial  
Anton committed
439
        this.updateSelectedCells()
Mathias Bär's avatar
Mathias Bär committed
440
        this.updateEditingCells()
441
      },
442 443 444 445 446

      //
      // GRID DRAG & DROP HANDLERS
      //

447
      async handleGridDragOver (event) {
Anton's avatar
Intial  
Anton committed
448 449 450
        let _this = this
        if (this.resizingGrid) {
          const position = this.getGridPositionForEvent(event)
451 452
          this.grid.config.ratio = position.ox / (position.oy * 1.0)
          await this.updateGridMetadataStore()
Anton's avatar
Intial  
Anton committed
453 454
        }
        else {
455 456
          let annotation = this.annotations.filter(annotation => {
            if (!_this.annotationUIStates[annotation._uuid]) return false
Mathias Bär's avatar
Mathias Bär committed
457
            return _this.annotationUIStates[annotation._uuid].beingDragged ||
458
              _this.annotationUIStates[annotation._uuid].beingResized
Anton's avatar
Intial  
Anton committed
459 460
          }).shift()
          let offset, position
461 462 463 464
          if (!annotation) {
            annotation = {
              target: this.grid.get2DArea([1, 1], [1, 1])
            }
Anton's avatar
Intial  
Anton committed
465 466 467
            position = this.getGridPositionForEvent(event)
          }
          else {
468
            offset = this.annotationUIStates[annotation._uuid].draggingOffset
Anton's avatar
Intial  
Anton committed
469 470
            position = this.getGridPositionForEvent(event, offset)
          }
471 472
          if (!this.tmpObjects.length) this.tmpObjects.push(annotation)
          const parsed = annotation.target.selector.parse()
Anton's avatar
Intial  
Anton committed
473
          if (event.dataTransfer.types.includes('text/plain')) {
474 475
            parsed.xywh[0] = position.x
            parsed.xywh[1] = position.y
Anton's avatar
Intial  
Anton committed
476 477 478
            event.preventDefault()
          }
          else {
479 480
            parsed.xywh[2] = Math.max(1, 1 + position.x - parsed.xywh[0])
            parsed.xywh[3] = Math.max(1, 1 + position.y - parsed.xywh[1])
Anton's avatar
Intial  
Anton committed
481
          }
482
          annotation.target.selector.value = parsed
Anton's avatar
Intial  
Anton committed
483 484 485
        }
      },
      handleGridDragEnd () {
486
        this.tmpObjects = []
Anton's avatar
Intial  
Anton committed
487
      },
488 489
      async handleGridDrop (event) {
        let dropData = event.dataTransfer.getData('text/plain')
christianrhansen's avatar
christianrhansen committed
490 491
        console.log('dropData', dropData)
        // {"data":{"content":"vvvvvvsdscscdsc"},"config":{},"component":"CellText"}
492 493
        if (dropData) {
          dropData = JSON.parse(dropData)
494
          let annotation = this.annotations.find(a => a.id === dropData.id)
495 496 497 498 499
          const { x, y } = this.getGridPositionForEvent(
            event,
            annotation ? this.annotationUIStates[annotation._uuid].draggingOffset : undefined
          )
          if (annotation) {
500 501 502
            const
              parsed = annotation.target.selector.parse(),
              target = this.grid.get2DArea([x, y], parsed.xywh.slice(2))
503 504 505 506 507 508
            annotation.target.selector.value = target.selector.value
            await this.$store.dispatch('annotations/patch', [annotation.id, {
              target: {
                selector: { value: target.selector.value }
              }
            }])
Anton's avatar
Intial  
Anton committed
509 510
          }
          else {
511 512 513 514 515 516 517 518 519 520 521
            const
              { data, config, component } = dropData,
              cell = await this.$store.dispatch('cells/post', { data, config, component })
            annotation = await this.$store.dispatch('annotations/post', {
              body: {
                type: 'Cell',
                purpose: 'linking',
                source: {
                  id: cell.id
                }
              },
522
              target: this.grid.get2DArea([x, y], [1, 1])
523
            })
524
            this.annotations.push(annotation)
Anton's avatar
Anton committed
525
            this.updateAnnotationUIStates()
Anton's avatar
Intial  
Anton committed
526
          }
527

528
          this.tmpObjects = []
Anton's avatar
Intial  
Anton committed
529 530 531
          event.preventDefault()
        }
      },
532 533 534 535

      //
      // CELL DRAG & DROP HANDLERS
      //
Mathias Bär's avatar
Mathias Bär committed
536 537 538
      handleCellEditClick (event, cell) {
        this.annotationUIStates[cell._uuid].editing = !this.annotationUIStates[cell._uuid].editing
        this.updateEditingCells()
539
        this.$root.$emit('mosys_saveScrollPosition')
Mathias Bär's avatar
Mathias Bär committed
540
      },
541 542 543 544 545
      handleCellClick (event, cell) {
        this.annotationUIStates[cell._uuid].selected = !this.annotationUIStates[cell._uuid].selected
        this.updateSelectedCells()
      },
      handleCellDragStart (event, annotation) {
christianrhansen's avatar
christianrhansen committed
546
        console.log('§§§§§§§§', annotation)
547 548 549 550 551 552 553 554 555 556
        if (!this.annotationUIStates[annotation._uuid].beingResized) {
          event.dataTransfer.setData('text/plain', JSON.stringify(annotation))
          event.dataTransfer.setDragImage(nullImage, 0, 0)
          let elContainerBoundingBox = this.$el.getBoundingClientRect()
          let elBoundingBox = event.target.getBoundingClientRect()
          let offset = {
            x: (event.clientX - elContainerBoundingBox.x) - (elBoundingBox.x - elContainerBoundingBox.x),
            y: (event.clientY - elContainerBoundingBox.y) - (elBoundingBox.y - elContainerBoundingBox.y)
          }
          this.annotationUIStates[annotation._uuid].draggingOffset = offset
Mathias Bär's avatar
Mathias Bär committed
557
          this.annotationUIStates[annotation._uuid].beingDragged = true
558 559 560 561 562 563 564 565 566
        }
        this.tmpObjects.push(annotation)
      },
      handleCellDragEnd (event, annotation) {
        this.annotationUIStates[annotation._uuid].beingDragged = false
      },
      async handleCellContextMenuDelete (event, annotation) {
        this.annotationUIStates[annotation._uuid].selected = false
        this.updateSelectedCells()
Mathias Bär's avatar
Mathias Bär committed
567
        this.updateEditingCells()
568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610
        await this.$store.dispatch('cells/delete', annotation.body.source.id)
        await this.$store.dispatch('annotations/delete', annotation.id)
        this.annotations = this.annotations.filter(a => a.id !== annotation.id)
      },

      //
      // GRID RESIZE HANDLERS

      handleGridResizerDragStart (event) {
        event.dataTransfer.setDragImage(nullImage, 0, 0)
      },
      async handleGridResizerDragEnd () {
        await this.updateGridMetadataStore()
      },

      //
      // CELL RESIZE HANDLERS
      //

      handleCellResizerDragStart (event, annotation) {
        event.dataTransfer.setDragImage(nullImage, 0, 0)
        this.annotationUIStates[annotation._uuid].beingResized = true
        this.tmpObjects.push(annotation)
      },
      async handleCellResizerDragEnd (event, annotation) {
        let position = this.getGridPositionForEvent(event)
        let
          parsed = annotation.target.selector.parse(),
          [x, y, w, h] = parsed.xywh
        w = Math.max(1, 1 + position.x - x)
        h = Math.max(1, 1 + position.y - y)
        const value = { xywh: [x, y, w, h] }
        annotation.target.selector.value = value
        this.annotationUIStates[annotation._uuid].beingResized = false
        this.tmpObjects = []
        await this.$store.dispatch('annotations/patch', [annotation.id, { target: { selector: { value } } }])
      },

      //
      // CELL CONTEXT MENU
      //

      handleCellContextMenu (event) {
Anton's avatar
Intial  
Anton committed
611 612
        this.contextMenuClickPosition = this.getGridPositionForEvent(event)
      },
613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629
      async handleCellContextMenuEditCSS (event, annotation) {
        try {
          let styleClass = await this.$q.dialog({
            title: this.$t('forms.edit_css_class.title'),
            ok: this.$t('buttons.set_css_class'),
            cancel: this.$t('buttons.cancel'),
            prompt: {
              model: annotation.styleClass,
              type: 'text'
            }
          })
          if (styleClass) {
            if (!styleClass.length) styleClass = undefined
            else if (styleClass.indexOf('.') === 0) styleClass = styleClass.substr(1)
          }
          annotation.styleClass = styleClass
          this.$store.dispatch('annotations/patch', [annotation._uuid, { target: { styleClass: styleClass || null } }])
Anton's avatar
Intial  
Anton committed
630
        }
631 632 633 634 635 636 637 638 639
        catch (e) { /* dialog canceled */ }
      },

      //
      // GRID CONTEXT MENU
      //

      handleGridContextMenu (event) {
        this.contextMenuClickPosition = this.getGridPositionForEvent(event)
Anton's avatar
Intial  
Anton committed
640 641 642
      },
      async handleGridContextMenuInsertColumnLeft () {
        let position = this.contextMenuClickPosition
643 644 645 646 647 648 649 650
        for (let annotation of this.annotations) {
          const parsed = annotation.target.selector.parse()
          if (parsed.xywh[0] >= position.x) {
            parsed.xywh[0] += 1
            annotation.target.selector.value = parsed
            await this.$store.dispatch('annotations/patch', [annotation.id, {
              target: { selector: { value: parsed } }
            }])
Anton's avatar
Intial  
Anton committed
651 652
          }
        }
653
        this.grid.config.columns += 1
Anton's avatar
Intial  
Anton committed
654 655 656 657
        await this.updateGridMetadataStore()
      },
      async handleGridContextMenuDeleteColumn () {
        let position = this.contextMenuClickPosition
658 659 660 661 662 663 664 665
        for (let annotation of this.annotations) {
          const parsed = annotation.target.selector.parse()
          if (parsed.xywh[0] > position.x) {
            parsed.xywh[0] -= 1
            annotation.target.selector.value = parsed
            await this.$store.dispatch('annotations/patch', [annotation.id, {
              target: { selector: { value: parsed } }
            }])
Anton's avatar
Intial  
Anton committed
666 667
          }
        }
668
        this.grid.config.columns -= 1
Anton's avatar
Intial  
Anton committed
669 670 671 672
        await this.updateGridMetadataStore()
      },
      async handleGridContextMenuInsertRowAbove () {
        let position = this.contextMenuClickPosition
673 674 675 676 677 678 679 680
        for (let annotation of this.annotations) {
          const parsed = annotation.target.selector.parse()
          if (parsed.xywh[1] >= position.y) {
            parsed.xywh[1] += 1
            annotation.target.selector.value = parsed
            await this.$store.dispatch('annotations/patch', [annotation.id, {
              target: { selector: { value: parsed } }
            }])
Anton's avatar
Intial  
Anton committed
681 682
          }
        }
683
        this.grid.config.rows += 1
Anton's avatar
Intial  
Anton committed
684 685 686 687
        await this.updateGridMetadataStore()
      },
      async handleGridContextMenuDeleteRow () {
        let position = this.contextMenuClickPosition
688 689 690 691 692 693 694 695
        for (let annotation of this.annotations) {
          const parsed = annotation.target.selector.parse()
          if (parsed.xywh[1] > position.y) {
            parsed.xywh[1] -= 1
            annotation.target.selector.value = parsed
            await this.$store.dispatch('annotations/patch', [annotation.id, {
              target: { selector: { value: parsed } }
            }])
Anton's avatar
Intial  
Anton committed
696 697
          }
        }
698
        this.grid.config.rows -= 1
Anton's avatar
Intial  
Anton committed
699 700
        await this.updateGridMetadataStore()
      },
701 702 703 704 705

      //
      // NAVIGATION
      //

Anton's avatar
Intial  
Anton committed
706
      handleGridButtonClickEdit () {
707
        this.$store.commit('mosys/toggleSources')
Anton's avatar
Intial  
Anton committed
708
      },
709 710 711 712 713

      //
      // GRID HELPERS
      //

Anton's avatar
Anton committed
714 715
      getGridPositionForEvent (event, offset = { x: 0, y: 0 }) {
        offset = { x: 0, y: 0 } // TODO: remove quick fix
Anton's avatar
Intial  
Anton committed
716 717 718 719 720
        const elContainerBoundingBox = this.$el.getBoundingClientRect()
        const ox = event.clientX + this.$el.scrollLeft - elContainerBoundingBox.x - offset.x
        const oy = event.clientY + this.$el.scrollTop - elContainerBoundingBox.y - offset.y
        const x = Math.ceil(ox / this.gridDimensions.full.cell.width)
        const y = Math.ceil(oy / this.gridDimensions.full.cell.height)
Anton's avatar
Anton committed
721
        return { x: x, y: y, ox: ox, oy: oy }
Anton's avatar
Intial  
Anton committed
722
      },
723 724 725 726
      updateGridDimensions (size) {
        if (!this.grid || !this.grid.config) return

        let elWidth = size ? size.width : this.$el.offsetWidth
727
        // let elHeight = size ? size.height : this.$el.offsetHeight
728
        let cellSizeRatio = this.grid.config.ratio
729 730
        // let gridHeight = elHeight
        let gridHeight = this.$el.offsetHeight
731
        let cellHeight = gridHeight / this.grid.config.rows
Anton's avatar
Intial  
Anton committed
732
        let cellWidth = elWidth / Math.round(elWidth / (cellHeight * cellSizeRatio))
733
        let gridWidth = cellWidth * this.grid.config.columns
Anton's avatar
Intial  
Anton committed
734
        let cellsPerWidth = elWidth / cellWidth
735
        let cellWidthMini = elWidth / this.grid.config.columns
Anton's avatar
Intial  
Anton committed
736 737 738 739 740 741 742 743 744 745 746 747 748
        let gridHeightMini = cellWidthMini / cellSizeRatio
        this.gridDimensions = {
          full: {
            width: gridWidth,
            height: gridHeight,
            cell: {
              width: cellWidth,
              height: cellHeight
            },
            cells_per_width: cellsPerWidth
          },
          mini: {
            width: elWidth,
749
            height: gridHeightMini * this.grid.config.rows,
Anton's avatar
Intial  
Anton committed
750 751 752 753 754 755 756
            cell: {
              width: cellWidthMini,
              height: gridHeightMini
            }
          }
        }
      },
757 758
      getAnnotationStyle (annotation) {
        const parsed = annotation.target.selector.parse()
Anton's avatar
Intial  
Anton committed
759
        return {
760 761 762 763
          'grid-column-start': parsed.xywh[0],
          'grid-column-end': `span ${parsed.xywh[2]}`,
          'grid-row-start': parsed.xywh[1],
          'grid-row-end': `span ${parsed.xywh[3]}`
Anton's avatar
Intial  
Anton committed
764
        }
Mathias Bär's avatar
Mathias Bär committed
765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781
      },
      getAnnotationClasses (uuid, which) {
        if (which === 'editing') {
          if (this.annotationUIStates[uuid].editing) {
            return 'bg-primary text-white'
          }
          else {
            return 'bg-grey'
          }
        }
        if (which === 'cell-item') {
          return {
            selected: this.annotationUIStates[uuid] ? this.annotationUIStates[uuid].selected : false,
            editing: this.annotationUIStates[uuid] ? this.annotationUIStates[uuid].editing : false
          }
        }
        else return {}
782 783 784 785
      },
      resetScrollPosition () {
        console.log('reset', this.scrollPositionCache, this.$el.scrollLeft)
        this.$el.scrollLeft = this.scrollPositionCache
Anton's avatar
Intial  
Anton committed
786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812
      }
    }
  }
</script>

<style scoped lang="stylus">
  .cell-grid
    display grid
    background-color #eee

  .cell-item

    &
      background-color rgba(0,0,0,0.2)
      border 1px solid #777
      margin 1px
      box-sizing border-box
      position relative
      overflow: hidden
      grid-column-start: 1
      grid-column-end: span 1
      grid-row-start: 1
      grid-row-end: span 1

    &:hover
      background-color lightblue

Mathias Bär's avatar
Mathias Bär committed
813 814 815 816
      .edit-button
        display: block

    &.editing
817
      background-color lightpink
Mathias Bär's avatar
Mathias Bär committed
818 819 820 821

      .edit-button
        display: block

Anton's avatar
Intial  
Anton committed
822 823 824
    &.selected
      background-color lightpink

Mathias Bär's avatar
Mathias Bär committed
825 826 827
    .edit-button
      display: none

Anton's avatar
Intial  
Anton committed
828 829 830 831 832 833 834 835 836 837 838 839 840 841 842
    .cell-item-inner
      width 100%
      height 100%

    .cell-item-resize-handle
      color rgba(0,0,0,0.2)
      position absolute
      right 0
      bottom 0
      width 18px
      height 20px

      &:hover
        color black

843 844 845 846 847 848 849 850 851
  .backdrop-filter
    backdrop-filter blur(6px)
    background-color rgba(255, 255, 255, .3)
    border-radius .5rem

  #edit-content-mobile
    width calc(100vw - 32px)
    max-height calc(calc(100vh - 60px - 16px - 16px) / 2)
    overflow-y scroll
Anton's avatar
Intial  
Anton committed
852
</style>