prsm.js

/********************************************************************************************* 

PRSM Participatory System Mapper 

Copyright (c) [2022] Nigel Gilbert email: prsm@prsm.uk

This software is licenced under the PolyForm Noncommercial License 1.0.0

<https://polyformproject.org/licenses/noncommercial/1.0.0>

See the file LICENSE.md for details.


This is the main entry point for PRSM.  
********************************************************************************************/

import * as Y from 'yjs'
import { WebsocketProvider } from 'y-websocket'
import { Network } from 'vis-network/peer'
import { DataSet } from 'vis-data/peer'
import diff from 'microdiff'
import localForage from 'localforage'
import {
  listen,
  elem,
  getScaleFreeNetwork,
  uuidv4,
  isEmpty,
  deepMerge,
  deepCopy,
  splitText,
  dragElement,
  standardizeColor,
  setNodeHidden,
  setEdgeHidden,
  factorSizeToPercent,
  setFactorSizeFromPercent,
  convertDashes,
  getDashes,
  objectEquals,
  generateName,
  statusMsg,
  alertMsg,
  cancelAlertMsg,
  clearStatusBar,
  shorten,
  initials,
  CP,
  timeAndDate,
  setEndOfContenteditable,
  exactTime,
  humanSize,
  isQuillEmpty,
  displayHelp,
} from './utils.js'
import {
  openFile,
  savePRSMfile,
  exportPNGfile,
  setFileName,
  exportExcel,
  exportDOT,
  exportGML,
  exportGraphML,
  exportGEXF,
  exportNotes,
  readSingleFile,
} from './files.js'
import Tutorial from './tutorial.js'
import { styles } from './samples.js'
import { trophic } from './trophic.js'
import { cluster, openCluster } from './cluster.js'
import { mergeRoom, diffRoom } from './merge.js'
import Quill from 'quill'
import {
  setUpSamples,
  reApplySampleToNodes,
  refreshSampleNode,
  reApplySampleToLinks,
  refreshSampleLink,
  legend,
  clearLegend,
} from './styles.js'
import {
  canvas,
  nChanges,
  setUpBackground,
  updateFromRemote,
  redraw,
  resizeCanvas,
  zoomCanvas,
  panCanvas,
  deselectTool,
  copyBackgroundToClipboard,
  pasteBackgroundFromClipboard,
  upgradeFromV1,
  updateFromDrawingMap,
} from './background.js'
import { getAIresponse } from './ai.js'
import { version } from '../package.json'
import { compressToUTF16, decompressFromUTF16 } from 'lz-string'

const appName = 'Participatory System Mapper'
const shortAppName = 'PRSM'
const GRIDSPACING = 50 // for snap to grid
const NODEWIDTH = 10 // chars for label splitting
const TIMETOSLEEP = 15 * 60 * 1000 // if no mouse movement for this time, user is assumed to have left or is sleeping
const TIMETOEDIT = 5 * 60 * 1000 // if node/edge edit dialog is not saved after this time, the edit is cancelled
const magnification = 3 // magnification of the loupe (magnifier 'glass')
export const NLEVELS = 20 // max. number of levels for trophic layout
const ROLLBACKS = 10 // max. number of versions stored for rollback

export let network
export let room
/* debug options (add to the URL thus: &debug=yjs,gui)
 * yjs - display yjs observe events on console
 * changes - show details of changes to yjs types
 * trans - all transactions
 * gui - show all mouse events
 * plain - save PRSM file as plain text, not compressed
 * cluster - show creation of clusters
 * aware - show awareness traffic
 * round - round trip timing
 * back - drawing on background
 * prompt - show AI prompts
 */
export let debug = ''
const allowAI = false // set to false to hide 'sparkle' buttons and so make AI features inaccessible to users
let viewOnly // when true, user can only view, not modify, the network
let showCopyMapButton = false // show the Copy Map button on the navbar in viewOnly mode
let nodes // a dataset of nodes
let edges // a dataset of edges
export let data // an object with the nodes and edges datasets as properties
export const doc = new Y.Doc()
export let websocket = 'wss://www.prsm.uk/wss' // web socket server URL
let wsProvider // web socket provider
export let clientID // unique ID for this browser
let yNodesMap // shared map of nodes
let yEdgesMap // shared map of edges
export let ySamplesMap // shared map of styles
export let yNetMap // shared map of global network settings
export let yPointsArray // shared array of the background drawing commands
export let yDrawingMap // shared map of background objects
export let yUndoManager // shared list of commands for undo
let dontUndo // when non-null, don't add an item to the undo stack
let yAwareness // awareness channel
export let yHistory // log of actions
export let container //the DOM body element
export let netPane // the DOM pane showing the network
let panel // the DOM right side panel element
let myNameRec // the user's name record {actual name, type, etc.}
export let lastNodeSample = 'group0' // the last used node style
export let lastLinkSample = 'edge0' // the last used edge style
/** @type {(string|boolean)} */
let inAddMode = false // true when adding a new Factor to the network; used to choose cursor pointer
let inEditMode = false //true when node or edge is being edited (dialog is open)
let snapToGridToggle = false // true when snapping nodes to the (unseen) grid
export let drawingSwitch = false // true when the drawing layer is uppermost
let showNotesToggle = true // show notes when factors and links are selected
let showVotingToggle = false // whether to show the voting thumb up/down buttons under factors
// if set, there are  nodes that need to be hidden when the map is drawn for the first time
const hiddenNodes = {
  radiusSetting: null,
  streamSetting: null,
  pathsSetting: null,
  selected: [],
}
const tutorial = new Tutorial() // object driving the tutorial
export const cp = new CP()
// color picker
let checkMapSaved = false // if the map is new (no 'room' in URL), or has been imported from a file, and changes have been made, warn user before quitting
let dirty = false // map has been changed by user and may need saving
let followme // clientId of user's cursor to follow
let editor = null // Quill editor
let popupWindow = null // window for editing Notes
let popupEditor = null // Quill editor in popup window
let sideDrawEditor = null // Quill editor in side drawer
let loadingDelayTimer // timer to delay the start of the loading animation for few moments
let netLoaded = false // becomes true when map is fully displayed
let savedState = '' // the current state of the map (nodes, edges, network settings) before current user action
let unknownRoomTimeout = null // timer to check if the room exists
const setupStartTime = Date.now() // time when setup started
/**
 * top level function to initialise everything
 */
window.addEventListener('load', () => {
  loadingDelayTimer = setTimeout(() => {
    elem('loading').style.display = 'block'
  }, 200)
  addEventListeners()
  setUpPage()
  setUpBackground()
  startY()
  setUpUserName()
  setUpAwareness()
  setUpShareDialog()
  draw()
})
/**
 * Clean up before user departs
 */

window.onbeforeunload = function (event) {
  unlockAll()
  yAwareness.setLocalStateField('addingFactor', { state: 'done' })
  yAwareness.setLocalState(null)
  // get confirmation from user before exiting if there are unsaved changes
  if (checkMapSaved && dirty) {
    event.preventDefault()
    event.returnValue = 'You have unsaved unchanges.  Are you sure you want to leave?'
  }
}

/**
 * Set up all the permanent event listeners
 */
function addEventListeners() {
  listen('maptitle', 'keydown', (e) => {
    // disallow Enter key
    if (e.key === 'Enter') {
      e.preventDefault()
    }
  })
  listen('net-pane', 'keydown', (e) => {
    if (e.which === 8 || e.which === 46) deleteNode()
  })
  listen('recent-rooms-caret', 'click', createTitleDropDown)
  listen('maptitle', 'keydown', (e) => {
    if (e.target.innerText === 'Untitled map') window.getSelection().selectAllChildren(e.target)
  })
  listen('maptitle', 'keyup', mapTitle)
  listen('maptitle', 'paste', pasteMapTitle)
  listen('maptitle', 'click', (e) => {
    if (e.target.innerText === 'Untitled map') window.getSelection().selectAllChildren(e.target)
  })
  listen('body', 'keydown', (e) => {
    if ((e.ctrlKey && e.key === 's') || (e.metaKey && e.key === 's')) {
      savePRSMfile()
      e.preventDefault()
    }
  })
  listen('body', 'keydown', (e) => {
    if ((e.ctrlKey && e.key === 'o') || (e.metaKey && e.key === 'o')) {
      openFile()
      e.preventDefault()
    }
  })
  listen('body', 'keydown', (e) => {
    if ((e.ctrlKey && e.key === 'z') || (e.metaKey && e.key === 'z')) {
      undo()
      e.preventDefault()
    }
  })
  listen('body', 'keydown', (e) => {
    if ((e.ctrlKey && e.key === 'y') || (e.metaKey && e.key === 'y')) {
      redo()
      e.preventDefault()
    }
  })

  listen('addNode', 'click', plusNode)
  listen('net-pane', 'contextmenu', contextMenu)
  listen('net-pane', 'click', unFollow)
  listen('net-pane', 'click', removeTitleDropDown)
  listen('drawer-handle', 'click', () => {
    elem('drawer-wrapper').classList.toggle('hide-drawer')
  })
  listen('addLink', 'click', plusLink)
  listen('deleteNode', 'click', deleteNode)
  listen('undo', 'click', undo)
  listen('redo', 'click', redo)
  listen('fileInput', 'change', readSingleFile)
  listen('openFile', 'click', openFile)
  listen('replaceMap', 'click', openFile)
  listen('mergeMap', 'click', mergeMap)
  listen('merge', 'click', doMerge)
  listen('mergeClose', 'click', () => elem('mergeDialog').close())
  listen('saveFile', 'click', savePRSMfile)
  listen('exportPRSM', 'click', savePRSMfile)
  listen('exportImage', 'click', exportPNGfile)
  listen('exportExcel', 'click', exportExcel)
  listen('exportGML', 'click', exportGML)
  listen('exportDOT', 'click', exportDOT)
  listen('exportGraphML', 'click', exportGraphML)
  listen('exportGEXF', 'click', exportGEXF)
  listen('exportNotes', 'click', exportNotes)
  listen('copy-map', 'click', () => doClone(false))
  listen('search', 'click', search)
  listen('help', 'click', displayHelp)
  listen('panelToggle', 'click', togglePanel)
  listen('zoom', 'change', zoomnet)
  listen('navbar', 'dblclick', fit)
  listen('zoomminus', 'click', () => {
    zoomincr(-0.1)
  })
  listen('zoomplus', 'click', () => {
    zoomincr(0.1)
  })
  listen('nodesButton', 'click', (e) => {
    openTab('nodesTab', e)
  })
  listen('linksButton', 'click', (e) => {
    openTab('linksTab', e)
  })
  listen('networkButton', 'click', (e) => {
    openTab('networkTab', e)
  })
  listen('analysisButton', 'click', (e) => {
    openTab('analysisTab', e)
  })
  listen('layoutSelect', 'change', autoLayout)
  listen('snaptogridswitch', 'click', snapToGridSwitch)
  listen('curveSelect', 'change', selectCurve)
  listen('drawing', 'click', toggleDrawingLayer)
  listen('allFactors', 'click', selectAllFactors)
  listen('allLinks', 'click', selectAllLinks)
  listen('showLegendSwitch', 'click', legendSwitch)
  listen('showVotingSwitch', 'click', votingSwitch)
  listen('showUsersSwitch', 'click', showUsersSwitch)
  listen('showHistorySwitch', 'click', showHistorySwitch)
  listen('showNotesSwitch', 'click', showNotesSwitch)
  listen('clustering', 'change', selectClustering)
  listen('lock', 'click', setFixed)
  listen('newNodeWindow', 'click', openNotesWindow)
  listen('newEdgeWindow', 'click', openNotesWindow)
  listen('sparklesNode', 'click', genAINode)
  listen('sparklesEdge', 'click', genAIEdge)
  listen('sparklesSideNote', 'click', genAISideNote)

  Array.from(document.getElementsByName('radius')).forEach((elem) => {
    elem.addEventListener('change', analyse)
  })
  Array.from(document.getElementsByName('stream')).forEach((elem) => {
    elem.addEventListener('change', analyse)
  })
  Array.from(document.getElementsByName('paths')).forEach((elem) => {
    elem.addEventListener('change', analyse)
  })
  listen('sizing', 'change', sizingSwitch)
  Array.from(document.getElementsByClassName('sampleNode')).forEach((elem) =>
    elem.addEventListener('click', (event) => {
      applySampleToNode(event)
    })
  )
  Array.from(document.getElementsByClassName('sampleLink')).forEach((elem) =>
    elem.addEventListener('click', (event) => {
      applySampleToLink(event)
    })
  )
  listen('nodeStyleEditFactorSize', 'input', (event) => progressBar(event.target))

  listen('history-copy', 'click', copyHistoryToClipboard)

  listen('body', 'copy', copyToClipboard)
  listen('body', 'paste', pasteFromClipboard)
  // change pointer when entering drag handles
  Array.from(document.getElementsByClassName('drag-handle')).forEach((el) => {
    el.addEventListener('pointerenter', () => (el.style.cursor = 'move'))
    el.addEventListener('pointerout', () => (el.style.cursor = 'auto'))
  })
  // if user has changed to this  tab, ensure that the network has been drawn
  document.addEventListener('visibilitychange', () => {
    network.redraw()
  })
}

/**
 * create all the DOM elements on the web page
 */
function setUpPage() {
  elem('version').innerHTML = version
  container = elem('container')
  netPane = elem('net-pane')
  panel = elem('panel')
  // check debug options set on URL: ?debug=yjs|gui|cluster|viewing|start|copyButton
  // each of these generates trace output on the console
  const searchParams = new URL(document.location).searchParams
  if (searchParams.has('debug')) debug = searchParams.get('debug')
  // don't allow user to change anything if URL includes ?viewing
  // this is now obsolete, but retained for backwards compatibility
  viewOnly = searchParams.has('viewing')
  if (viewOnly) hideNavButtons()
  if (searchParams.has('copyButton')) showCopyMapButton = true
  // treat user as first time user if URL includes ?start=true
  if (searchParams.has('start')) localStorage.setItem('doneIntro', 'false')
  panel.classList.add('hide')
  container.panelHidden = true
  cp.createColorPicker('netBackColorWell', updateNetBack)
  setUpPinchZoom()
  setUpSamples()
  updateLastSamples(lastNodeSample, lastLinkSample)
  makeNotesPanelResizeable(elem('nodeNotePanel'))
  makeNotesPanelResizeable(elem('edgeNotePanel'))
  dragElement(elem('nodeNotePanel'), elem('nodeNoteHeader'))
  dragElement(elem('edgeNotePanel'), elem('edgeNoteHeader'))
  hideNotes()
  setUpSideDrawer()
  // remove AI sparkle buttons if not providing AI features
  if (!allowAI) {
    Array.from(document.getElementsByClassName('sparkle')).forEach((elem) => {
      elem.style.display = 'none'
    })
  }
  displayWhatsNew()
}
const sliderColor = getComputedStyle(document.documentElement).getPropertyValue('--slider')
/**
 * draw the solid bar to the left of the thumb on a slider
 * @param {HTMLElement} sliderEl input[type=range] element
 */
export function progressBar(sliderEl) {
  const sliderValue = sliderEl.value
  sliderEl.style.background = `linear-gradient(to right, ${sliderColor} ${sliderValue}%, #ccc ${sliderValue}%)`
}
/**
 * show the What's New modal dialog unless this is a new user or user has already seen this dialog
 * for this (Major.Minor) version
 */
function displayWhatsNew() {
  // new user - don't tell them what is new
  if (!localStorage.getItem('doneIntro')) return
  const versionDecoded = version.match(/^(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)/)
  const seen = localStorage.getItem('seenWN')
  if (seen) {
    const seenDecoded = seen.match(/^(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)/)
    // if this is a new minor version, show the What's New dialog
    if (
      seenDecoded &&
      versionDecoded[1] === seenDecoded[1] &&
      versionDecoded[2] === seenDecoded[2]
    ) {
      return
    }
  }
  elem('whatsnewversion').innerHTML = `Version ${version}`
  elem('whatsnew').style.display = 'flex'
  elem('net-pane').addEventListener('click', hideWhatsNew, { once: true })
}
/**
 * hide the What's New dialog when the user has clicked Continue, and note tha the user has seen it
 */
function hideWhatsNew() {
  localStorage.setItem('seenWN', version)
  elem('whatsnew').style.display = 'none'
}
/**
 * create a new shared document and start the WebSocket provider
 */
function startY(newRoom) {
  const url = new URL(document.location)
  if (newRoom) room = newRoom
  else {
    // get the room number from the URL, or if none, generate a new one
    room = url.searchParams.get('room')
  }
  if (room == null || room === '') {
    room = generateRoom()
    checkMapSaved = true
  } else room = room.toUpperCase()
  // if debug flag includes 'local' or using a non-standard port (i.e neither 80 nor 443)
  // assume that the websocket port is 1234 in the same domain as the url
  if (/local/.test(debug) || (url.port && url.port !== 80 && url.port !== 443)) {
    websocket = `ws://${url.hostname}:1234`
  }
  wsProvider = new WebsocketProvider(websocket, `prsm${room}`, doc)
  wsProvider.on('synced', () => {
    // if this is a clone, load the cloned data
    initiateClone()
    // (if the room already exists, wait until the map data is loaded before displaying it)
    if (url.searchParams.get('room') !== null) {
      observed('synced')
      if (/load/.test(debug)) {
        console.log(
          `Nodes: ${yNodesMap.size} Edges: ${yEdgesMap.size} Samples: ${ySamplesMap.size} Network settings: ${yNetMap.size}	Points: ${yPointsArray.length} Drawing objects: ${yDrawingMap.size} History entries: ${yHistory.length}	`
        )
      }
      unknownRoomTimeout = setTimeout(() => {
        if (!netLoaded) {
          displayNetPane(
            `${exactTime()} Timed out waiting for ${room} to load. Found only ${Array.from(foundMaps).join(', ')} maps.`
          )
        }
      }, 6000)
    } else {
      // if this is a new map, display it
      displayNetPane(`${exactTime()} no remote content loaded from ${websocket}`)
    }
  })
  wsProvider.disconnectBc()
  wsProvider.on('status', (event) => {
    console.log(
      `${exactTime()}${event.status}${event.status === 'connected' ? ' to' : ' from'} room ${room} using ${websocket}`
    )
  })

  /* 
	create a yMap for the nodes and one for the edges (we need two because there is no 
	guarantee that the the ids of nodes will differ from the ids of edges) 
	 */
  yNodesMap = doc.getMap('nodes')
  yEdgesMap = doc.getMap('edges')
  ySamplesMap = doc.getMap('samples')
  yNetMap = doc.getMap('network')
  yPointsArray = doc.getArray('points')
  yDrawingMap = doc.getMap('drawing')
  yHistory = doc.getArray('history')
  yAwareness = wsProvider.awareness

  /* create a dummy item in yNodesMap and yEdgesMap to stop having to wait for the these maps 
	if there are no nodes or edges (thus allowing to distinguish between zero nodes/edges and 
	no node/edge map yet loaded) */
  yNodesMap.set('_dummy_', { dummy: true })
  yEdgesMap.set('_dummy_', { dummy: true })

  /* set up observers to listen for changes in the yMaps */

  doc.on('afterTransaction', () => {
    if (!netLoaded) {
      fit()
    }
  })
  if (/trans/.test(debug)) {
    doc.on('afterTransaction', (tr) => {
      console.log(
        `${exactTime()} transaction (${JSON.stringify(tr)})  (${tr.local ? 'local' : 'remote'})`
      )
      console.log('netLoaded', netLoaded)
      const nodesEvent = tr.changed.get(yNodesMap)
      if (nodesEvent) console.log(nodesEvent)
      const edgesEvent = tr.changed.get(yEdgesMap)
      if (edgesEvent) console.log(edgesEvent)
      const sampleEvent = tr.changed.get(ySamplesMap)
      if (sampleEvent) console.log(sampleEvent)
      const netEvent = tr.changed.get(yNetMap)
      if (netEvent) console.log(netEvent)
    })
  }

  clientID = doc.clientID
  console.log(`My client ID: ${clientID}`)

  /* set up the undo managers */
  yUndoManager = new Y.UndoManager([yNodesMap, yEdgesMap, yNetMap], {
    trackedOrigins: new Set([null]), // add undo items to the stack by default
  })
  dontUndo = null

  nodes = new DataSet()
  edges = new DataSet()
  data = {
    nodes,
    edges,
  }

  /* 
	for convenience when debugging
	 */
  window.debug = debug
  window.data = data
  window.clientID = clientID
  window.yNodesMap = yNodesMap
  window.yEdgesMap = yEdgesMap
  window.ySamplesMap = ySamplesMap
  window.yNetMap = yNetMap
  window.yUndoManager = yUndoManager
  window.yHistory = yHistory
  window.yPointsArray = yPointsArray
  window.yDrawingMap = yDrawingMap
  window.styles = styles
  window.yAwareness = yAwareness
  window.mergeRoom = mergeRoom
  window.diffRoom = diffRoom
  window.wsProvider = wsProvider

  const foundMaps = new Set()
  /**
   * note that one of the required yMaps has been loaded; if all have been found, display the map
   * @param {string} what name of the yMap that has just been loaded
   */
  function observed(what) {
    // do nothing if the map is already displayed
    if (netLoaded) return
    if (/load/.test(debug)) {
      console.log(`${exactTime()} Observed: ${what}`)
    }
    foundMaps.add(what)
    if (
      foundMaps.has('nodes') &&
      foundMaps.has('edges') &&
      foundMaps.has('network') &&
      foundMaps.has('synced')
    ) {
      displayNetPane(`${exactTime()} all content loaded from ${websocket}`)
      if (/load/.test(debug)) {
        console.log(
          `Nodes: ${yNodesMap.size} Edges: ${yEdgesMap.size} Samples: ${ySamplesMap.size} Network settings: ${yNetMap.size} Points: ${yPointsArray.length} Drawing objects: ${yDrawingMap.size} History entries: ${yHistory.length}	`
        )
      }
    }
  }

  /* 
	nodes.on listens for when local nodes or edges are changed (added, updated or removed).
	If a local node is removed, the yMap is updated to broadcast to other clients that the node 
	has been deleted. If a local node is added or updated, that is also broadcast.
	 */
  nodes.on('*', (evt, properties, origin) => {
    yjsTrace(
      'nodes.on',
      `${evt}  ${JSON.stringify(properties.items)} origin: ${origin} dontUndo: ${dontUndo}`
    )
    clearTimeout(unknownRoomTimeout)
    if (!viewOnly) {
      doc.transact(() => {
        properties.items.forEach((id) => {
          if (origin === null) {
            // this is a local change
            if (evt === 'remove') {
              yNodesMap.delete(id.toString())
            } else {
              yNodesMap.set(id.toString(), deepCopy(nodes.get(id)))
            }
          }
        })
      }, dontUndo)
    }
    dontUndo = null
  })
  /* 
	yNodesMap.observe listens for changes in the yMap, receiving a set of the keys that have
	had changed values.  If the change was to delete an entry, the corresponding node and all links to/from it are
	removed from the local nodes dataSet. Otherwise, if the received node differs from the local one, 
	the local node dataSet is updated (which includes adding a new node if it does not already exist locally).
	 */
  yNodesMap.observe((evt) => {
    yjsTrace('yNodesMap.observe', evt)
    const nodesToUpdate = []
    const nodesToRemove = []
    for (const key of evt.keysChanged) {
      if (yNodesMap.has(key)) {
        const obj = yNodesMap.get(key)
        if (objectEquals(obj, { dummy: true })) continue // skip dummy entry
        if (!objectEquals(obj, data.nodes.get(key))) {
          // fix nodes if this is a view only copy
          if (viewOnly) obj.fixed = true
          nodesToUpdate.push(deepCopy(obj))
          // if a note on a node is being remotely edited and is on display here, update the local note and the padlock
          if (editor && editor.id === key && evt.transaction.local === false) {
            editor.setContents(obj.note)
            elem('fixed').style.display = obj.fixed ? 'inline' : 'none'
            elem('unfixed').style.display = obj.fixed ? 'none' : 'inline'
          }
        }
      } else {
        hideNotes()
        if (data.nodes.get(key)) {
          network.getConnectedEdges(key).forEach((edge) => nodesToRemove.push(edge))
        }
        nodesToRemove.push(key)
      }
    }
    if (nodesToUpdate.length > 0) nodes.update(nodesToUpdate, 'remote')
    if (nodesToRemove.length > 0) nodes.remove(nodesToRemove, 'remote')
    if (/changes/.test(debug) && (nodesToUpdate.length > 0 || nodesToRemove.length > 0)) {
      showChange(evt, yNodesMap)
    }
    observed('nodes')
  })
  /* 
	See comments above about nodes
	 */
  edges.on('*', (evt, properties, origin) => {
    yjsTrace(
      'edges.on',
      `${evt}  ${JSON.stringify(properties.items)} origin: ${origin} dontUndo: ${dontUndo}`
    )
    if (!viewOnly) {
      doc.transact(() => {
        properties.items.forEach((id) => {
          if (origin === null) {
            if (evt === 'remove') yEdgesMap.delete(id.toString())
            else {
              yEdgesMap.set(id.toString(), deepCopy(edges.get(id)))
            }
          }
        })
      }, dontUndo)
    }
    dontUndo = null
  })
  yEdgesMap.observe((evt) => {
    yjsTrace('yEdgesMap.observe', evt)
    const edgesToUpdate = []
    const edgesToRemove = []
    for (const key of evt.keysChanged) {
      if (yEdgesMap.has(key)) {
        const obj = yEdgesMap.get(key)
        if (objectEquals(obj, { dummy: true })) continue // skip dummy entry
        if (!objectEquals(obj, data.edges.get(key))) {
          edgesToUpdate.push(deepCopy(obj))
          if (editor && editor.id === key && evt.transaction.local === false) {
            editor.setContents(obj.note)
          }
        }
      } else {
        hideNotes()
        edgesToRemove.push(key)
      }
    }
    if (edgesToUpdate.length > 0) edges.update(edgesToUpdate, 'remote')
    if (edgesToRemove.length > 0) edges.remove(edgesToRemove, 'remote')
    if (edgesToUpdate.length > 0 || edgesToRemove.length > 0) {
      // if user is in mid-flight adding a Link, and someone else has just added a link,
      // vis-network will cancel the edit mode for this user.  Re-instate it.
      if (inAddMode === 'addLink') network.addEdgeMode()
    }
    if (/changes/.test(debug) && (edgesToUpdate.length > 0 || edgesToRemove.length > 0)) {
      showChange(evt, yEdgesMap)
    }
    observed('edges')
  })
  /**
   * utility trace function that prints the change in the value of a YMap property to the console
   * @param {YEvent} evt
   * @param {MapType} ymap
   */
  function showChange(evt, ymap) {
    evt.changes.keys.forEach((change, key) => {
      if (change.action === 'add') {
        console.log(
          `Property "${key}" was added. 
				Initial value: `,
          ymap.get(key)
        )
      } else if (change.action === 'update') {
        console.log(
          `Property "${key}" was updated. 
					New value: "`,
          ymap.get(key),
          `"
					Previous value: "`,
          change.oldValue,
          `" 
					Difference: "`,
          typeof change.oldValue === 'object' && typeof ymap.get(key) === 'object'
            ? diff(change.oldValue, ymap.get(key))
            : `${change.oldValue} ${ymap.get(key)}`,
          `"`
        )
      } else if (change.action === 'delete') {
        console.log(
          `Property "${key}" was deleted. 
				Previous value: `,
          change.oldValue
        )
      }
    })
  }
  ySamplesMap.observe((evt) => {
    yjsTrace('ySamplesMap.observe', evt)
    const nodesToUpdate = []
    const edgesToUpdate = []
    for (const key of evt.keysChanged) {
      const sample = ySamplesMap.get(key)
      if (sample.node !== undefined) {
        if (!objectEquals(styles.nodes[key], sample.node)) {
          styles.nodes[key] = sample.node
          refreshSampleNode(key)
          nodesToUpdate.push(key)
        }
      } else {
        if (!objectEquals(styles.edges[key], sample.edge)) {
          styles.edges[key] = sample.edge
          refreshSampleLink(key)
          edgesToUpdate.push(key)
        }
      }
    }
    if (nodesToUpdate) {
      reApplySampleToNodes(nodesToUpdate)
    }
    if (edgesToUpdate) {
      reApplySampleToLinks(edgesToUpdate)
    }
    observed('samples')
  })
  /*
	Map controls (those on the Network tab) are of three kinds:
	1. Those that affect only the local map and are not promulgated to other users
	e.g zoom, show drawing layer, show history
	2. Those where the control status (e.g. whether a switch is on or off) is promulgated,
	but the effect of the switch is handled by yNodesMap and yEdgesMap (e.g. Show Factors
		x links away; Size Factors to)
	3. Those whose effects are promulgated and switches controlled here by yNetMap (e.g
		Background)
	For cases 2 and 3, the functions called here must not invoke yNetMap.set() to avoid loops
	*/
  yNetMap.observe((evt) => {
    yjsTrace('YNetMap.observe', evt)

    if (evt.transaction.origin) // evt is not local
    {
      for (const key of evt.keysChanged) {
        const obj = yNetMap.get(key)
        switch (key) {
          case 'viewOnly': {
            viewOnly = viewOnly || obj
            if (viewOnly) {
              hideNavButtons()
              disableSideDrawerEditing()
            }
            break
          }
          case 'mapTitle':
          case 'maptitle': {
            setMapTitle(obj)
            break
          }
          case 'snapToGrid': {
            doSnapToGrid(obj)
            break
          }
          case 'curve': {
            setCurve(obj)
            break
          }
          case 'background': {
            setBackground(obj)
            break
          }
          case 'legend': {
            setLegend(obj, false)
            break
          }
          case 'voting': {
            setVoting(obj)
            break
          }
          case 'showNotes': {
            doShowNotes(obj)
            break
          }
          case 'radius': {
            hiddenNodes.radiusSetting = obj.radiusSetting
            hiddenNodes.selected = obj.selected
            setRadioVal('radius', hiddenNodes.radiusSetting)
            break
          }
          case 'stream': {
            hiddenNodes.streamSetting = obj.streamSetting
            hiddenNodes.selected = obj.selected
            setRadioVal('stream', hiddenNodes.streamSetting)
            break
          }
          case 'paths': {
            hiddenNodes.pathsSetting = obj.pathsSetting
            hiddenNodes.selected = obj.selected
            setRadioVal('paths', hiddenNodes.pathsSetting)
            break
          }
          case 'sizing': {
            sizing(obj)
            break
          }
          case 'hideAndStream':
          case 'linkRadius':
            // old settings (before v1.6) - ignore
            break
          case 'factorsHiddenByStyle': {
            updateFactorsOrLinksHiddenByStyle(obj)
            break
          }
          case 'linksHiddenByStyle': {
            updateFactorsOrLinksHiddenByStyle(obj)
            break
          }
          case 'attributeTitles': {
            recreateClusteringMenu(obj)
            break
          }
          case 'cluster': {
            setCluster(obj)
            break
          }
          case 'mapDescription': {
            setSideDrawer(obj)
            break
          }
          case 'lastLoaded':
          case 'version': {
            // ignore these  - for info only
            break
          }
          default:
            console.log('Bad key in yMapNet.observe: ', key)
        }
      }
    }
    observed('network')
  })
  yPointsArray.observe((evt) => {
    yjsTrace('yPointsArray.observe', yPointsArray.get(yPointsArray.length - 1))
    if (evt.transaction.local === false) upgradeFromV1(yPointsArray.toArray())
  })
  yDrawingMap.observe((evt) => {
    yjsTrace('yDrawingMap.observe', evt)
    updateFromRemote(evt)
    observed('drawing')
  })
  yHistory.observe(() => {
    yjsTrace('yHistory.observe', yHistory.get(yHistory.length - 1))
    if (elem('showHistorySwitch').checked) showHistory()
    observed('history')
  })
  yUndoManager.on('stack-item-added', (evt) => {
    yjsTrace('yUndoManager.on stack-item-added', evt)
    if (/changes/.test(debug)) {
      evt.changedParentTypes.forEach((v) => {
        showChange(v[0], v[0].target)
      })
    }
    undoRedoButtonStatus()
  })
  yUndoManager.on('stack-item-popped', (evt) => {
    yjsTrace('yUndoManager.on stack-item-popped', evt)
    if (/changes/.test(debug)) {
      evt.changedParentTypes.forEach((v) => {
        showChange(v[0], v[0].target)
      })
    }
    pruneDanglingEdges()
    undoRedoButtonStatus()
  })
  /**
   * In some slightly obscure circumstances, (specifically, client A undoes the creation of a factor that
   * client B has subsequently linked to another factor), the undo operation can result in a link that
   * has no source or destination factor.  Tracking such a situation is rather complex, so this cleans
   * up the mess without bothering about its cause.
   */
  function pruneDanglingEdges() {
    data.edges.forEach((edge) => {
      if (data.nodes.get(edge.from) === null) {
        dontUndo = 'danglingEdge'
        data.edges.remove(edge.id)
      }
      if (data.nodes.get(edge.to) == null) {
        dontUndo = 'danglingEdge'
        data.edges.remove(edge.id)
      }
    })
  }
} // end startY()

/**
 * load cloned data from localStorage
 * if there is no clone, returns without doing anything
 */
function initiateClone() {
  localForage
    .getItem('clone')
    .then((clone) => {
      localForage
        .removeItem('clone')
        .then(() => {
          // if there is no clone, clone will be null
          if (clone) {
            const state = JSON.parse(decompressFromUTF16(clone))
            data.nodes.update(state.nodes)
            data.edges.update(state.edges)
            doc.transact(() => {
              for (const k in state.net) {
                yNetMap.set(k, state.net[k])
              }
              viewOnly = state.options.viewOnly
              yNetMap.set('viewOnly', viewOnly)
              data.nodes.get().forEach((obj) => (obj.fixed = viewOnly))
              if (viewOnly) hideNavButtons()
              for (const k in state.samples) {
                ySamplesMap.set(k, state.samples[k])
              }
              if (state.paint) {
                yPointsArray.delete(0, yPointsArray.length)
                yPointsArray.insert(0, state.paint)
              }
              if (state.drawing) {
                for (const k in state.drawing) {
                  yDrawingMap.set(k, state.drawing[k])
                }
                updateFromDrawingMap()
              }
              logHistory(state.options.created.action, state.options.created.actor)
            }, 'clone')
            unSelect()
            fit()
          }
        })
        .catch((err) => {
          console.log('Cant delete localForage clone key: ', err)
        })
    })
    .catch((err) => {
      console.log('Cant get localForage clone key: ', err)
    })
}
/**
 * Display observed yjs event
 * @param {string} where
 * @param {object} what
 */
function yjsTrace(where, what) {
  if (/yjs/.test(debug)) {
    console.log(exactTime(), where, what)
  }
}
/**
 * create a random string of the form AAA-BBB-CCC-DDD
 */
function generateRoom() {
  let room = ''
  for (let i = 0; i < 4; i++) {
    for (let j = 0; j < 3; j++) {
      room += String.fromCharCode(65 + Math.floor(Math.random() * 26))
    }
    if (i < 3) room += '-'
  }
  return room
}
/**
 * randomly create some nodes and edges as a binary tree, mainly used for testing
 * @param {number} nNodes
 */
function getRandomData(nNodes) {
  const SFNdata = getScaleFreeNetwork(nNodes)
  nodes.add(SFNdata.nodes)
  edges.add(SFNdata.edges)
  reApplySampleToNodes(['group0'])
  reApplySampleToLinks(['edge0'])
  recalculateStats()
}
/**
 * Once any existing map has been loaded, fit it to the pane and reveal it
 * @param {string} msg message for console
 */
function displayNetPane(msg) {
  console.log(msg)
  if (!netLoaded) {
    elem('loading').style.display = 'none'
    fit()
    setMapTitle(yNetMap.get('mapTitle'))
    netPane.style.visibility = 'visible'
    clearTimeout(loadingDelayTimer)
    yUndoManager.clear()
    undoRedoButtonStatus()
    network.unselectAll()
    setUpTutorial()
    netLoaded = true
    drawMinimap()
    savedState = saveState()
    setAnalysisButtonsFromRemote()
    toggleDeleteButton()
    setLegend(yNetMap.get('legend'), false)
    console.log(
      exactTime(),
      `Doc size: ${humanSize(Y.encodeStateAsUpdate(doc).length)}, Load time: ${((Date.now() - setupStartTime) / 1000).toFixed(1)}s`
    )
    yNetMap.set('lastLoaded', Date.now())
    yNetMap.set('version', version)
  }
}
// to handle iPad viewport sizing problem when tab bar appears and to keep panels on screen
setvh()

window.onresize = function () {
  setvh()
  keepPaneInWindow(panel)
  resizeCanvas()
}
/**
 * Hack to get window size when orientation changes.  Should use screen.orientation, but this is not
 * implemented by Safari
 */
const portrait = window.matchMedia('(orientation: portrait)')
portrait.addEventListener('change', () => {
  setvh()
})
/**
 * in View Only mode, hide all the Nav Bar buttons except the search button
 * and make the map title not editable
 */
function hideNavButtons() {
  elem('buttons').style.visibility = 'hidden'
  elem('search').parentElement.style.visibility = 'visible'
  elem('search').parentElement.style.borderLeft = 'none'
  if (showCopyMapButton) {
    elem('copy-map-button').style.display = 'block'
    elem('copy-map-button').style.visibility = 'visible'
  }
  elem('maptitle').contentEditable = 'false'
  if (!container.panelHidden) {
    panel.classList.add('hide')
    container.panelHidden = true
  }
}
/** restore all the Nav Bar buttons when leaving view only mode (e.g. when
 * going back online)
 */
function showNavButtons() {
  elem('buttons').style.visibility = 'visible'
  elem('search').parentElement.style.visibility = 'visible'
  elem('search').parentElement.style.borderLeft = '1px solid rgb(255, 255, 255)'
  elem('copy-map-button').style.display = 'none'
  elem('maptitle').contentEditable = 'true'
}
/**
 * cancel View Only mode (only available via the console)
 */
function cancelViewOnly() {
  viewOnly = false
  yNetMap.set('viewOnly', false)
  showNavButtons()
  data.nodes.get().forEach((obj) => (obj.fixed = false))
  network.setOptions({ interaction: { dragNodes: true, hover: true } })
}
window.cancelViewOnly = cancelViewOnly
/**
 * to handle iOS weirdness in fixing the vh unit (see https://css-tricks.com/the-trick-to-viewport-units-on-mobile/)
 */
function setvh() {
  document.body.height = window.innerHeight
  // First we get the viewport height and we multiple it by 1% to get a value for a vh unit
  const vh = window.innerHeight * 0.01
  // Then we set the value in the --vh custom property to the root of the document
  document.documentElement.style.setProperty('--vh', `${vh}px`)
}

/**
 * retrieve or generate user's name
 */
function setUpUserName() {
  try {
    myNameRec = JSON.parse(localStorage.getItem('myName'))
  } catch {
    myNameRec = null
  }
  saveUserName(myNameRec?.name ? myNameRec.name : '')
  console.log(`My name: ${myNameRec.name}`)
}
/**
 * Save a new user name into local storage
 * @param {String} name
 */
function saveUserName(name) {
  if (name.length > 0) {
    myNameRec.name = name
    myNameRec.anon = false
  } else {
    myNameRec = generateName()
  }
  myNameRec.id = clientID
  localStorage.setItem('myName', JSON.stringify(myNameRec))
  yAwareness.setLocalState({ user: myNameRec })
  showAvatars()
}

/**
 * if this is the user's first time, show them how the user interface works
 */
function setUpTutorial() {
  if (localStorage.getItem('doneIntro') !== 'done' && viewOnly === false) {
    tutorial.onexit(function () {
      localStorage.setItem('doneIntro', 'done')
    })
    tutorial.onstep(0, () => {
      const splashNameBox = elem('splashNameBox')
      const anonName = myNameRec.name || generateName().name
      splashNameBox.placeholder = anonName
      splashNameBox.focus()
      splashNameBox.addEventListener('blur', () => {
        saveUserName(splashNameBox.value || anonName)
      })
      splashNameBox.addEventListener('keyup', (e) => {
        if (e.key === 'Enter') splashNameBox.blur()
      })
    })
    tutorial.start()
  }
}

/**
 * draw the network, after setting the vis-network options
 */
function draw() {
  // for testing, you can append ?t=XXX to the URL of the page, where XXX is the number
  // of factors to include in a random network
  const url = new URL(document.location.href.toLowerCase())
  const nNodes = parseInt(url.searchParams.get('t'))
  if (nNodes) getRandomData(nNodes)
  // create a network
  const options = {
    nodes: {
      chosen: {
        node: function (values, id, selected) {
          values.shadow = selected
        },
      },
    },
    edges: {
      chosen: {
        edge: function (values, id, selected) {
          values.shadow = selected
        },
      },
      smooth: {
        type: 'cubicBezier',
      },
    },
    physics: {
      enabled: false,
      stabilization: false,
    },
    interaction: {
      multiselect: true,
      selectConnectedEdges: false,
      hover: false,
      hoverConnectedEdges: false,
      zoomView: false,
      tooltipDelay: 0,
    },
    manipulation: {
      enabled: false,
      addNode: function (item, callback) {
        item.label = ''
        item = deepMerge(item, styles.nodes[lastNodeSample])
        item.grp = lastNodeSample
        item.created = timestamp()
        addLabel(item, cancelAdd, callback)
        showPressed('addNode', 'remove')
      },
      editNode: function (item, callback) {
        // for some weird reason, vis-network copies the group properties into the
        // node properties before calling this fn, which we don't want.  So we
        // revert to using the original node properties before continuing.
        item = data.nodes.get(item.id)
        item.modified = timestamp()
        const point = network.canvasToDOM({ x: item.x, y: item.y })
        editNode(item, point, cancelEdit, callback)
      },
      addEdge: function (item, callback) {
        inAddMode = false
        network.setOptions({
          interaction: { dragView: true, selectable: true },
        })
        showPressed('addLink', 'remove')
        if (item.from === item.to) {
          callback(null)
          stopEdit()
          return
        }
        if (duplEdge(item.from, item.to).length > 0) {
          alertMsg('There is already a link from this Factor to the other.', 'error')
          callback(null)
          stopEdit()
          return
        }
        if (data.nodes.get(item.from).isCluster || data.nodes.get(item.to).isCluster) {
          alertMsg('Links cannot be made to or from a cluster', 'error')
          callback(null)
          stopEdit()
          return
        }
        item = deepMerge(item, styles.edges[lastLinkSample])
        item.grp = lastLinkSample
        item.created = timestamp()
        clearStatusBar()
        callback(item)
        logHistory(
          `added link from '${data.nodes.get(item.from).label}' to '${data.nodes.get(item.to).label}'`
        )
      },
      editEdge: {
        editWithoutDrag: function (item, callback) {
          item = data.edges.get(item.id)
          item.modified = timestamp()
          // find midpoint of edge
          const point = network.canvasToDOM({
            x: (network.getPosition(item.from).x + network.getPosition(item.to).x) / 2,
            y: (network.getPosition(item.from).y + network.getPosition(item.to).y) / 2,
          })
          editEdge(item, point, cancelEdit, callback)
        },
      },
      deleteNode: function (item, callback) {
        let locked = false
        item.nodes.forEach((nId) => {
          const n = data.nodes.get(nId)
          if (n.locked) {
            locked = true
            alertMsg(
              `Factor '${shorten(n.oldLabel)}' can't be deleted because it is locked`,
              'warn'
            )
            callback(null)
          }
        })
        if (locked) return
        clearStatusBar()
        hideNotes()
        // delete also all the edges that link to the nodes being deleted
        item.nodes.forEach((nId) => {
          network.getConnectedEdges(nId).forEach((eId) => {
            if (item.edges.indexOf(eId) === -1) item.edges.push(eId)
          })
        })
        item.edges.forEach((edgeId) => {
          logHistory(
            `deleted link from '${data.nodes.get(data.edges.get(edgeId).from).label}' to '${
              data.nodes.get(data.edges.get(edgeId).to).label
            }'`
          )
        })
        network.unselectAll()
        item.nodes.forEach((nodeId) => {
          logHistory(`deleted factor: '${data.nodes.get(nodeId).label}'`)
        })
        callback(item)
      },
      deleteEdge: function (item, callback) {
        item.edges.forEach((edgeId) => {
          logHistory(
            `deleted link from '${data.nodes.get(data.edges.get(edgeId).from).label}' to '${
              data.nodes.get(data.edges.get(edgeId).to).label
            }'`
          )
        })
        callback(item)
      },
      controlNodeStyle: {
        shape: 'dot',
        color: 'red',
        size: 5,
        group: undefined,
      },
    },
  }
  if (viewOnly) {
    options.interaction = {
      dragNodes: false,
      hover: false,
    }
  }
  network = new Network(netPane, data, options)
  window.network = network
  elem('zoom').value = network.getScale()

  // start with factor tab open, but hidden
  elem('nodesButton').click()

  // listen for click events on the network pane
  let doubleClickTimer = null
  network.on('click', (params) => {
    if (/gui/.test(debug)) console.log('**click**', params)
    // if user is doing an analysis, and has clicked on a node, show the node notes
    if (
      getRadioVal('radius') !== 'All' ||
      getRadioVal('stream') !== 'All' ||
      getRadioVal('paths') !== 'All'
    ) {
      if (!showNotesToggle) return
      hideNotes()
      const clickedNodeId = network.getNodeAt(params.pointer.DOM)
      if (clickedNodeId) showNodeData(clickedNodeId)
      else {
        const clickedEdgeId = network.getEdgeAt(params.pointer.DOM)
        if (clickedEdgeId) showEdgeData(clickedEdgeId)
      }
      return
    }
    // if user has clicked on a portal node, open the map in another tab and go to it
    if (params.nodes.length === 1) {
      const node = data.nodes.get(params.nodes[0])
      // tricky stuff to distinguish a single click (move to map) from a double click (edit node)
      if (node.portal && doubleClickTimer === null) {
        doubleClickTimer = setTimeout(() => {
          window.open(`${window.location.pathname}?room=${node.portal}`, node.portal)
          doubleClickTimer = null
        }, 500)
      }
    }
    const keys = params.event.pointers[0]
    if (!keys) return
    if (keys.metaKey) {
      // if the Command key (on a Mac) is down, and the click is on a node/edge, log it to the console
      if (params.nodes.length === 1) {
        const node = data.nodes.get(params.nodes[0])
        console.log('node = ', node)
        window.node = node
      }
      if (params.edges.length === 1) {
        const edge = data.edges.get(params.edges[0])
        console.log('edge = ', edge)
        window.edge = edge
      }
      return
    }
    if (keys.altKey) {
      // if the Option/ALT key is down, add a node if on the background
      if (params.nodes.length === 0 && params.edges.length === 0) {
        const pos = params.pointer.canvas
        let item = { id: uuidv4(), label: '', x: pos.x, y: pos.y }
        item = deepMerge(item, styles.nodes[lastNodeSample])
        item.grp = lastNodeSample
        addLabel(item, clearPopUp, function (newItem) {
          if (newItem !== null) data.nodes.add(newItem)
        })
      }
      return
    }
    if (keys.shiftKey) {
      if (!inEditMode) showMagnifier(keys)
      return
    }
    // Might be a click on a thumb up/down
    if (showVotingToggle) {
      for (const node of data.nodes.get()) {
        const bBox = network.getBoundingBox(node.id)
        const clickPos = params.pointer.canvas
        if (
          clickPos.x > bBox.left &&
          clickPos.x < bBox.right &&
          clickPos.y > bBox.bottom &&
          clickPos.y < bBox.bottom + 20
        ) {
          if (clickPos.x < bBox.left + (bBox.right - bBox.left) / 2) {
            // if user has not already voted for this, add their vote, i.e. add their clientID
            // or if they have voted, remove it
            if (node.thumbUp?.includes(clientID)) {
              node.thumbUp = node.thumbUp.filter((c) => c !== clientID)
            } else if (node.thumbUp) node.thumbUp.push(clientID)
            else node.thumbUp = [clientID]
          } else {
            if (node.thumbDown?.includes(clientID)) {
              node.thumbDown = node.thumbDown.filter((c) => c !== clientID)
            } else if (node.thumbDown) node.thumbDown.push(clientID)
            else node.thumbDown = [clientID]
          }
          data.nodes.update(node)
          return
        }
      }
    }
  })

  // despatch to edit a node or an edge or to fit the network on the pane
  network.on('doubleClick', function (params) {
    if (/gui/.test(debug)) console.log('**doubleClick**')
    clearTimeout(doubleClickTimer)
    doubleClickTimer = null
    if (params.nodes.length === 1) {
      if (!(viewOnly || inEditMode)) network.editNode()
    } else if (params.edges.length === 1) {
      if (!(viewOnly || inEditMode)) network.editEdgeMode()
    } else {
      fit()
    }
  })
  network.on('selectNode', function (params) {
    if (/gui/.test(debug)) console.log('selectNode', params)
    // if user is doing an analysis, do nothing
    if (
      getRadioVal('radius') !== 'All' ||
      getRadioVal('stream') !== 'All' ||
      getRadioVal('paths') !== 'All'
    ) {
      return
    }
    // if a 'hidden' node is clicked, it is selected, but we don't want this
    // reset the selected nodes to all except the hidden one
    network.setSelection({
      nodes: params.nodes.filter((id) => !data.nodes.get(id).nodeHidden),
      edges: params.edges.filter((id) => !data.edges.get(id).edgeHidden),
    })
    showSelected()
    showNodeOrEdgeData()
    toggleDeleteButton()
    if (getRadioVal('radius') !== 'All') analyse()
    if (getRadioVal('stream') !== 'All') analyse()
    if (getRadioVal('paths') !== 'All') analyse()
  })
  network.on('deselectNode', function (params) {
    if (/gui/.test(debug)) console.log('deselectNode', params)
    // if user is doing an analysis, do nothing, but first reselect the unselected nodes
    if (
      getRadioVal('radius') !== 'All' ||
      getRadioVal('stream') !== 'All' ||
      getRadioVal('paths') !== 'All'
    ) {
      network.setSelection({
        nodes: params.previousSelection.nodes.map((node) => node.id),
        edges: params.previousSelection.edges.map((edge) => edge.id),
      })
      return
    }
    // if some other node(s) are already selected, and the user has
    // clicked on one of the selected nodes, do nothing,
    // i.e reselect all the nodes previously selected
    // similarly, if the user has clicked on a 'hidden' node,
    // reselect the previous nodes and do nothing
    if (params.nodes) {
      // clicked on a node
      const prevSelIds = params.previousSelection.nodes.map((node) => node.id)
      let hiddenEdge
      if (params.edges.length) hiddenEdge = data.edges.get(params.edges[0]).edgeHidden
      if (
        prevSelIds.includes(params.nodes[0]) ||
        data.nodes.get(params.nodes[0]).nodeHidden ||
        hiddenEdge
      ) {
        // reselect the previously selected nodes
        network.selectNodes(
          params.previousSelection.nodes.map((node) => node.id),
          false
        )
        return
      }
    }
    showSelected()
    showNodeOrEdgeData()
    toggleDeleteButton()
  })
  network.on('hoverNode', function () {
    changeCursor('grab')
  })
  network.on('blurNode', function () {
    changeCursor('default')
  })
  network.on('selectEdge', function (params) {
    if (/gui/.test(debug)) console.log('selectEdge')
    // if user is doing an analysis, do nothing
    if (
      getRadioVal('radius') !== 'All' ||
      getRadioVal('stream') !== 'All' ||
      getRadioVal('paths') !== 'All'
    ) {
      return
    }
    network.setSelection({
      nodes: params.nodes.filter((id) => !data.nodes.get(id).nodeHidden),
      edges: params.edges.filter((id) => !data.edges.get(id).edgeHidden),
    })
    showSelected()
    showNodeOrEdgeData()
    toggleDeleteButton()
  })
  network.on('deselectEdge', function (params) {
    if (/gui/.test(debug)) console.log('deselectEdge')
    // if user is doing an analysis, do nothing, but first reselect the unselected nodes
    if (
      getRadioVal('radius') !== 'All' ||
      getRadioVal('stream') !== 'All' ||
      getRadioVal('paths') !== 'All'
    ) {
      network.setSelection({
        nodes: params.previousSelection.nodes.map((node) => node.id),
        edges: params.previousSelection.edges.map((edge) => edge.id),
      })
      return
    }
    if (params.edges) {
      // clicked on an edge(see selectNode for comments)
      const prevSelIds = params.previousSelection.edges.map((edge) => edge.id)
      if (prevSelIds.includes(params.edges[0]) || data.edges.get(params.edges[0]).edgeHidden) {
        // reselect the previously selected edges
        network.selectEdges(
          params.previousSelection.edges.map((edge) => edge.id),
          false
        )
        return
      }
    }
    hideNotes()
    showSelected()
    toggleDeleteButton()
  })
  network.on('oncontext', function (e) {
    const nodeId = network.getNodeAt(e.pointer.DOM)
    if (nodeId) openCluster(nodeId)
  })

  let viewPosition
  let selectionCanvasStart = {}
  let selectionStart = {}
  const selectionArea = document.createElement('div')
  selectionArea.className = 'selectionBox'
  selectionArea.style.display = 'none'
  elem('main').appendChild(selectionArea)

  network.on('dragStart', function (params) {
    if (/gui/.test(debug)) console.log('dragStart')
    viewPosition = network.getViewPosition()
    const e = params.event.pointers[0]
    // start drawing a selection rectangle if the CTRL key is down and click is on the background
    if (e.ctrlKey && params.nodes.length === 0 && params.edges.length === 0) {
      network.setOptions({ interaction: { dragView: false } })
      listen('net-pane', 'mousemove', showAreaSelection)
      selectionStart = { x: e.offsetX, y: e.offsetY }
      selectionCanvasStart = params.pointer.canvas
      selectionArea.style.left = `${e.offsetX}px`
      selectionArea.style.top = `${e.offsetY}px`
      selectionArea.style.width = '0px'
      selectionArea.style.height = '0px'
      selectionArea.style.display = 'block'
      return
    }
    if (e.altKey) {
      if (!inAddMode) {
        removeFactorCursor()
        changeCursor('crosshair')
        inAddMode = 'addLink'
        showPressed('addLink', 'add')
        statusMsg('Now drag to the middle of the Destination factor')
        network.setOptions({
          interaction: { dragView: false, selectable: false },
        })
        network.addEdgeMode()
        return
      }
    }
    changeCursor('grabbing')
  })
  /**
   * update the selection rectangle as the mouse moves
   * @param {Event} event
   */
  function showAreaSelection(event) {
    selectionArea.style.left = `${Math.min(selectionStart.x, event.offsetX)}px`
    selectionArea.style.top = `${Math.min(selectionStart.y, event.offsetY)}px`
    selectionArea.style.width = `${Math.abs(event.offsetX - selectionStart.x)}px`
    selectionArea.style.height = `${Math.abs(event.offsetY - selectionStart.y)}px`
  }
  network.on('dragging', function () {
    if (/gui/.test(debug)) console.log('dragging')
    const endViewPosition = network.getViewPosition()
    panCanvas(viewPosition.x - endViewPosition.x, viewPosition.y - endViewPosition.y)
    viewPosition = endViewPosition
  })
  network.on('dragEnd', function (params) {
    if (/gui/.test(debug)) console.log('dragEnd')
    const endViewPosition = network.getViewPosition()
    panCanvas(viewPosition.x - endViewPosition.x, viewPosition.y - endViewPosition.y)
    if (selectionArea.style.display === 'block') {
      selectionArea.style.display = 'none'
      network.setOptions({ interaction: { dragView: true } })
      elem('net-pane').removeEventListener('mousemove', showAreaSelection)
    }
    const e = params.event.pointers[0]
    if (e.ctrlKey && params.nodes.length === 0 && params.edges.length === 0) {
      network.storePositions()
      const selectionCanvasEnd = params.pointer.canvas
      if (selectionCanvasStart.x > selectionCanvasEnd.x) {
        ;[selectionCanvasStart.x, selectionCanvasEnd.x] = [
          selectionCanvasEnd.x,
          selectionCanvasStart.x,
        ]
      }
      if (selectionCanvasStart.y > selectionCanvasEnd.y) {
        ;[selectionCanvasStart.y, selectionCanvasEnd.y] = [
          selectionCanvasEnd.y,
          selectionCanvasStart.y,
        ]
      }
      const selectedNodes = data.nodes.get({
        filter: function (node) {
          return (
            !node.nodeHidden &&
            node.x >= selectionCanvasStart.x &&
            node.x <= selectionCanvasEnd.x &&
            node.y >= selectionCanvasStart.y &&
            node.y <= selectionCanvasEnd.y
          )
        },
      })
      network.setSelection({
        nodes: selectedNodes.map((n) => n.id).concat(network.getSelectedNodes()),
      })
      showSelected()
      showNodeOrEdgeData()
      return
    }
    const newPositions = network.getPositions(params.nodes)
    data.nodes.update(
      data.nodes.get(params.nodes).map((n) => {
        n.x = newPositions[n.id].x
        n.y = newPositions[n.id].y
        if (snapToGridToggle) snapToGrid(n)
        return n
      })
    )
    changeCursor('default')
  })
  network.on('controlNodeDragging', function () {
    if (/gui/.test(debug)) console.log('controlNodeDragging')
    changeCursor('crosshair')
  })
  network.on('controlNodeDragEnd', function (event) {
    if (/gui/.test(debug)) console.log('controlNodeDragEnd')
    if (event.controlEdge.from !== event.controlEdge.to) changeCursor('default')
  })
  network.on('beforeDrawing', (ctx) => redraw(ctx))
  network.on('afterDrawing', (ctx) => drawBadges(ctx))

  // listen for changes to the network structure
  // and recalculate the network statistics when there is one
  data.nodes.on('add', recalculateStats)
  data.nodes.on('remove', recalculateStats)
  data.edges.on('add', recalculateStats)
  data.edges.on('remove', recalculateStats)

  /* --------------------------------------------set up the magnifier --------------------------------------------*/
  const magSize = 300 // diameter of loupe
  const halfMagSize = magSize / 2.0
  const netPaneCanvas = netPane.firstElementChild.firstElementChild
  const magnifier = document.createElement('canvas')
  magnifier.width = magSize
  magnifier.height = magSize
  magnifier.className = 'magnifier'
  const magnifierCtx = magnifier.getContext('2d')
  magnifierCtx.fillStyle = 'white'
  netPane.appendChild(magnifier)
  let bigNetPane = null
  let bigNetwork = null
  let bigNetCanvas = null
  let netPaneRect = null
  let magnifying = false

  netPane.addEventListener('keydown', (e) => {
    if (!inEditMode && e.shiftKey && !magnifying) createMagnifier(e)
  })
  netPane.addEventListener('mousemove', (e) => {
    if (magnifying && !inEditMode && e.shiftKey) showMagnifier(e)
  })
  netPane.addEventListener('keyup', (e) => {
    if (e.key === 'Shift') closeMagnifier()
  })
  // ensure magnifier shows even if mouse is over the panel (e.g. when doing analysis)
  panel.addEventListener('keydown', (e) => {
    if (!inEditMode && e.shiftKey && !magnifying) createMagnifier(e)
  })
  panel.addEventListener('mousemove', (e) => {
    if (magnifying && !inEditMode && e.shiftKey) showMagnifier(e)
  })
  panel.addEventListener('keyup', (e) => {
    if (e.key === 'Shift') closeMagnifier()
  })

  /**
   * create a copy of the network, but magnified and off screen
   */
  function createMagnifier(e) {
    if (bigNetPane) {
      bigNetwork.destroy()
      bigNetPane.remove()
    }
    if (drawingSwitch) return
    magnifying = true
    netPaneRect = netPane.getBoundingClientRect()
    network.storePositions()
    bigNetPane = document.createElement('div')
    bigNetPane.id = 'big-net-pane'
    bigNetPane.style.position = 'absolute'
    bigNetPane.style.top = '-9999px'
    bigNetPane.style.left = '-9999px'
    bigNetPane.style.width = `${netPane.offsetWidth * magnification}px`
    bigNetPane.style.height = `${netPane.offsetHeight * magnification}px`
    netPane.appendChild(bigNetPane)
    const bigNetData = {
      nodes: new DataSet(),
      edges: new DataSet(),
    }
    bigNetData.nodes.add(data.nodes.get())
    bigNetData.edges.add(data.edges.get())
    bigNetwork = new Network(bigNetPane, bigNetData, {
      physics: { enabled: false },
    })
    /* // unhide any hidden nodes and edges
		let changedNodes = []
		bigNetData.nodes.forEach((n) => {
			if (n.nodeHidden) {
				changedNodes.push(setNodeHidden(n, false))
			}
		})
		let changedEdges = []
		bigNetData.edges.forEach((e) => {
			if (e.edgeHidden) {
				changedEdges.push(setEdgeHidden(e, false))
			}
		})
		bigNetData.nodes.update(changedNodes)
		bigNetData.edges.update(changedEdges) */
    bigNetCanvas = bigNetPane.firstElementChild.firstElementChild
    bigNetwork.on('afterDrawing', () => {
      setCanvasBackground(bigNetCanvas)
    })
    bigNetwork.moveTo({
      position: network.getViewPosition(),
      scale: magnification * network.getScale(),
    })
    netPane.style.cursor = 'none'
    magnifier.style.display = 'none'
    showMagnifier(e)
  }
  /**
   * display the loupe, centred on the mouse pointer, and fill it with
   * an image copied from the magnified network
   */
  function showMagnifier(e) {
    e.preventDefault()
    if (drawingSwitch) return
    if (bigNetCanvas == null) createMagnifier()
    magnifierCtx.fillRect(0, 0, magSize, magSize)
    magnifierCtx.drawImage(
      bigNetCanvas,
      ((e.clientX - netPaneRect.x) * bigNetCanvas.width) / netPaneCanvas.clientWidth - halfMagSize,
      ((e.clientY - netPaneRect.y) * bigNetCanvas.height) / netPaneCanvas.clientHeight -
        halfMagSize,
      magSize,
      magSize,
      0,
      0,
      magSize,
      magSize
    )
    magnifier.style.top = `${e.clientY - netPaneRect.y - halfMagSize}px`
    magnifier.style.left = `${e.clientX - netPaneRect.x - halfMagSize}px`
    magnifier.style.display = 'block'
  }
  /**
   * destroy the magnified network copy
   */
  function closeMagnifier() {
    if (bigNetPane) {
      bigNetwork.destroy()
      bigNetPane.remove()
    }
    netPane.style.cursor = 'default'
    magnifier.style.display = 'none'
    magnifying = false
  }
} // end draw()

/**
 * draw the background on the given canvas (which will be a magnified version of the net pane)
 * @param {HTMLElement} canvas
 * @returns canvas
 */
export function setCanvasBackground(canvas) {
  const context = canvas.getContext('2d')
  context.setTransform()
  context.globalCompositeOperation = 'destination-over'
  // apply the background objects
  const backgroundCanvas = document.getElementById('underlay').firstElementChild.firstElementChild
  context.drawImage(backgroundCanvas, 0, 0, canvas.width, canvas.height)
  // apply the background colour, if any, or white
  context.fillStyle = elem('underlay').style.backgroundColor || 'rgb(255, 255, 255)'
  context.fillRect(0, 0, canvas.width, canvas.height)
  return canvas
}

/* --------------------------------------------draw and update the minimap --------------------------------------------*/
/**
 * Draw the minimap, which is a scaled down version of the network
 * with a 'radar' overlay showing the current view
 *
 * @param {number} [ratio=5] - the ratio of the size of the minimap to the network
 */
export function drawMinimap(ratio = 5) {
  let fullNetPane, fullNetwork, initialScale, initialPosition, minimapWidth, minimapHeight
  const minimapWrapper = document.getElementById('minimapWrapper') // a div to contain the minimap
  const minimapImage = document.getElementById('minimapImage') // an img, child of minimapWrapper
  const minimapRadar = document.getElementById('minimapRadar') // a div, child of minimapWrapper
  // size the minimap
  minimapSetup()
  // set up dragging of the radar overlay
  let dragging = false // if true, ignore clicks when user is dragging radar overlay
  dragRadar()
  /**
   * Set the size of the minimap and its components
   */
  function minimapSetup() {
    const { clientWidth, clientHeight } = network.body.container
    minimapWidth = clientWidth / ratio
    minimapHeight = clientHeight / ratio
    minimapWrapper.style.width = `${minimapWidth}px`
    minimapWrapper.style.height = `${minimapHeight}px`
    minimapRadar.style.width = `${minimapWidth}px`
    minimapRadar.style.height = `${minimapHeight}px`
    drawMinimapImage()
    drawRadar()
  }
  /**
   * Draw a copy of the full network offscreen, then create an image of it
   * The visible network can't be used, because it may be scaled and panned, but the minimap image needs to
   * show the full network
   */
  function drawMinimapImage() {
    if (!elem('fullnetPane')) {
      // if the full network does not exist, create it
      fullNetPane = document.createElement('div')
      fullNetPane.style.position = 'absolute'
      fullNetPane.style.top = '-9999px'
      fullNetPane.style.left = '-9999px'
      fullNetPane.style.width = `${netPane.offsetWidth}px`
      fullNetPane.style.height = `${netPane.offsetHeight}px`
      fullNetPane.id = 'fullNetPane'
      netPane.appendChild(fullNetPane)
      fullNetwork = new Network(fullNetPane, data, {
        physics: { enabled: false },
      })
    }
    fullNetwork.setOptions({ edges: { smooth: elem('curveSelect').value === 'cubicBezier' } })
    fullNetwork.fit()
    initialScale = fullNetwork.getScale()
    initialPosition = fullNetwork.getViewPosition()

    const fullNetworklCanvas = fullNetPane.firstElementChild.firstElementChild
    fullNetwork.on('afterDrawing', () => {
      // make the image as a reduced version of the fullNetwork
      const tempCanvas = document.createElement('canvas')
      const tempContext = tempCanvas.getContext('2d')
      tempCanvas.width = minimapWidth
      tempCanvas.height = minimapHeight
      tempContext.drawImage(fullNetworklCanvas, 0, 0, minimapWidth, minimapHeight)
      minimapImage.src = tempCanvas.toDataURL()
      minimapImage.width = minimapWidth
      minimapImage.height = minimapHeight
    })
  }
  /**
   * Move a radar overlay on the minimap to show the current view of the network
   */
  function drawRadar() {
    const scale = initialScale / network.getScale()
    // fade out the whole minimap if the network is all visible in the viewport
    // (there is no value in having a minimap in this case)
    if (scale >= 1 && networkInPane()) {
      minimapWrapper.style.display = 'none'
      return
    } else minimapWrapper.style.display = 'block'
    const currentDOMPosition = network.canvasToDOM(network.getViewPosition())
    const initialDOMPosition = network.canvasToDOM(initialPosition)

    minimapRadar.style.left = `${Math.round(
      ((currentDOMPosition.x - initialDOMPosition.x) * scale) / ratio +
        (minimapWidth * (1 - scale)) / 2
    )}px`
    minimapRadar.style.top = `${Math.round(
      ((currentDOMPosition.y - initialDOMPosition.y) * scale) / ratio +
        (minimapHeight * (1 - scale)) / 2
    )}px`
    minimapRadar.style.width = `${minimapWidth * scale}px`
    minimapRadar.style.height = `${minimapHeight * scale}px`
  }
  /**
   *
   * @returns {boolean} - true if the network is entirely within the viewport
   */
  function networkInPane() {
    const netPaneTopLeft = network.DOMtoCanvas({ x: 0, y: 0 })
    const netPaneBottomRight = network.DOMtoCanvas({
      x: netPane.clientWidth,
      y: netPane.clientHeight,
    })
    for (const nodeId of data.nodes.getIds()) {
      const boundingBox = network.getBoundingBox(nodeId)
      if (boundingBox.left < netPaneTopLeft.x) return false
      if (boundingBox.right > netPaneBottomRight.x) return false
      if (boundingBox.top < netPaneTopLeft.y) return false
      if (boundingBox.bottom > netPaneBottomRight.y) return false
    }
    return true
  }
  /**
   * Whenever the network is resized, the minimap needs to be resized and the radar overlay moved
   */
  network.on('resize', () => {
    minimapSetup()
  })
  /**
   * Whenever the network is changed, panned or zoomed, the radar overlay needs to be moved
   */
  network.on('afterDrawing', () => {
    drawRadar()
  })
  /**
   * Set up dragging of the radar overlay
   */
  function dragRadar() {
    let x, y, radarStart
    minimapRadar.addEventListener('pointerdown', dragMouseDown)
    minimapWrapper.addEventListener(
      'wheel',
      (e) => {
        e.preventDefault()
        // reject all but vertical touch movements
        if (Math.abs(e.deltaX) <= 1) zoomscroll(e)
      },
      { passive: false }
    )
    /**
     * note that the mouse is down on the radar overlay and start dragging
     * @param {event} e
     */
    function dragMouseDown(e) {
      e.preventDefault()
      x = e.clientX
      y = e.clientY
      radarStart = { x: minimapRadar.offsetLeft, y: minimapRadar.offsetTop }
      minimapRadar.addEventListener('pointermove', drag)
      minimapRadar.addEventListener('pointerup', dragMouseUp)
    }
    /**
     * move the radar overlay as the mouse moves
     * @param {event} e
     */
    function drag(e) {
      e.preventDefault()
      dragging = true
      const dx = e.clientX - x
      const dy = e.clientY - y
      let left = radarStart.x + dx
      let top = radarStart.y + dy
      if (left < 0) left = 0
      if (left + minimapRadar.offsetWidth >= minimapWidth) {
        left = minimapWidth - minimapRadar.offsetWidth
      }
      if (top < 0) top = 0
      if (top + minimapRadar.offsetHeight >= minimapHeight) {
        top = minimapHeight - minimapRadar.offsetHeight
      }
      minimapRadar.style.left = `${Math.round(left)}px`
      minimapRadar.style.top = `${Math.round(top)}px`
      const initialDOMPosition = network.canvasToDOM(initialPosition)
      const scale = initialScale / network.getScale()
      const radarRect = minimapRadar.getBoundingClientRect()
      const wrapperRect = minimapWrapper.getBoundingClientRect()
      network.moveTo({
        position: network.DOMtoCanvas({
          x:
            ((radarRect.left - wrapperRect.left + (radarRect.width - wrapperRect.width) / 2) *
              ratio) /
              scale +
            initialDOMPosition.x,
          y:
            ((radarRect.top - wrapperRect.top + (radarRect.height - wrapperRect.height) / 2) *
              ratio) /
              scale +
            initialDOMPosition.y,
        }),
      })
    }
    /**
     * note that the mouse is up and stop dragging
     * @param {event} e
     */
    function dragMouseUp(e) {
      e.preventDefault()
      if (dragging) {
        minimapRadar.removeEventListener('pointermove', drag)
        minimapRadar.removeEventListener('pointerup', dragMouseUp)
      }
    }
  }
}
/* -------------------------------------------- network map utilities --------------------------------------------*/
/**
 * clear the map by destroying all nodes and edges and background objects
 */
export function clearMap() {
  doc.transact(() => {
    unSelect()
    ensureNotDrawing()
    network.destroy()
    checkMapSaved = true
    data.nodes.clear()
    data.edges.clear()
    yDrawingMap.clear()
    canvas.clear()
    draw()
  })
}

/**
 * note that the map has been saved to file and so user does not need to be warned
 * about quitting without saving
 */
export function markMapSaved() {
  checkMapSaved = false
  dirty = false
}
/**
 * un fade the delete button to show that it can be used when something is selected
 */
export function toggleDeleteButton() {
  if (network.getSelectedNodes().length > 0 || network.getSelectedEdges().length > 0) {
    elem('deleteNode').classList.remove('disabled')
  } else elem('deleteNode').classList.add('disabled')
}

function contextMenu(event) {
  event.preventDefault()
}
/****************************************************** update history for history log **************************/
/**
 * return an object with the current time as an integer date and the current user's name
 */
export function timestamp() {
  return { time: Date.now(), user: myNameRec.name }
}
window.timestamp = timestamp
/**
 * Generate a key for a time slot in the history log
 *
 * @param {integer} time
 * @returns {string} key
 */
function timekey(time) {
  return room + time
}
/**
 * push a record that action has been taken on to the end of the history log
 *  also record current state of the map for possible roll back
 *  and note changes have been made to the map
 * @param {String} action
 * @param {String} actor - the user who took the action
 * @param {boolean} dontSaveState - if defined, don't save the current state of the map
 */
export async function logHistory(action, actor, dontSaveState = null) {
  const now = Date.now()
  yHistory.push([
    {
      action,
      time: now,
      user: actor || myNameRec.name,
    },
  ])
  // store the current state of the map for possible rollback
  if (!dontSaveState) {
    await localForage.setItem(timekey(now), savedState).then(() => {
      savedState = saveState()

      // delete all but the last ROLLBACKS saved states
      for (let i = 0; i < yHistory.length - ROLLBACKS; i++) {
        const obj = yHistory.get(i)
        if (obj.time) localForage.removeItem(timekey(obj.time))
      }
    })
  }
  if (elem('history-window').style.display === 'block') showHistory()
  dirty = true
}
/**
 * Generate a compressed dump of the current state of the map, sufficient to reproduce it
 * @returns binary string
 */
export function saveState(options) {
  return compressToUTF16(
    JSON.stringify({
      nodes: data.nodes.get(),
      edges: data.edges.get(),
      net: yNetMap.toJSON(),
      samples: ySamplesMap.toJSON(),
      paint: yPointsArray.toArray(),
      drawing: yDrawingMap.toJSON(),
      options,
    })
  )
}

/******************************************************** map notes side drawer *********************************************************/
/**
 * set up the side drawer for notes
 */
function setUpSideDrawer() {
  sideDrawEditor = new Quill(elem('drawer-editor'), {
    modules: {
      //we need to have this in HTML, to add the AI sparkle icon to the tools
      toolbar: viewOnly ? null : '#sideNotesToolbar',
    },
    placeholder: 'Notes about the map',
    theme: 'snow',
    readOnly: viewOnly,
  })

  sideDrawEditor.on('text-change', (delta, oldDelta, source) => {
    if (source === 'user') {
      yNetMap.set('mapDescription', {
        text: isQuillEmpty(sideDrawEditor) ? '' : sideDrawEditor.getContents(),
      })
    }
  })
}
export function setSideDrawer(contents) {
  sideDrawEditor.setContents(contents.text)
}
export function disableSideDrawerEditing() {
  sideDrawEditor.disable()
  elem('drawer').firstElementChild.style.display = 'none'
}
async function genAISideNote() {
  if (data.nodes.length === 0) {
    alertMsg('Add some factors and causal links to the map first', 'error')
    return
  }
  alertMsg('Processing...', 'info', true)
  sideDrawEditor.setText('Processing...\n')
  const sparklesElem = elem('sparklesSideNote')
  sparklesElem.classList.add('rotating')
  const causes = data.edges
    .get()
    .map(
      (e) =>
        data.nodes.get(e.from).label.replaceAll('\n', ' ') +
        ' causes ' +
        data.nodes.get(e.to).label.replaceAll('\n', ' ')
    )
    .join('; ')
  let title = elem('maptitle').innerText
  if (title === 'Untitled map') title = 'System Map'
  /* 	let aiResponse = await getAIresponse(`A system map includes the following causal relationships. Write a description of the system map that will help a non-expert understand it. Use no more than 300 words. >>>${causes}<<<`) */
  const aiResponse =
    await getAIresponse(`I want you to generate a compact, readable narrative description of a system map.

I will provide:
• A title for the system map.
• A list of causal links, each given as a pair of factors in the form “A causes B”, where A is the cause and B is the effect.

Your task is to:
	1.	Produce a short, coherent, well-structured markdown description explaining the main dynamics of the system.
	2.	Cluster related factors, highlight feedback loops if present, and describe the overall behaviour of the system.
	3.	Avoid restating every causal link individually. Instead, synthesise them into a readable explanation.
	4.	Use bullet points where appropriate but do *not* include a title or any section headings.
	5.	Keep the description concise and accessible, suitable for a briefing note.  Use no more than 300 words.
  
Output only the final description in markdown.

Here is the title and list of causal links:
Title: >>>${title}<<<
Causal Links:
 >>>${causes}<<<`)

  sideDrawEditor.setContents(aiResponse)
  yNetMap.set('mapDescription', { text: sideDrawEditor.getContents() })
  sparklesElem.classList.remove('rotating')
  cancelAlertMsg()
}

/************************************************************* badges around the factors ******************************************/
const noteImage = new Image()
noteImage.src =
  ''
const lockImage = new Image()
lockImage.src =
  ''
const thumbUpImage = new Image()
thumbUpImage.src =
  ''
const thumbUpFilledImage = new Image()
thumbUpFilledImage.src =
  ''
const thumbDownImage = new Image()
thumbDownImage.src =
  ''
const thumbDownFilledImage = new Image()
thumbDownFilledImage.src =
  ''
/**
 * draw badges (icons) around Factors and Links
 * @param {CanvasRenderingContext2D} ctx NetPane canvas context
 */
function drawBadges(ctx) {
  // padlock for locked factors
  if (!viewOnly) {
    // for a view only map, factors are always locked, so don't bother with padlock
    data.nodes
      .get()
      .filter((node) => !node.nodeHidden && node.fixed && !node.clusteredIn)
      .forEach((node) => {
        const box = network.getBoundingBox(node.id)
        drawTheBadge(lockImage, ctx, box.left - 10, box.top)
      })
  }
  if (showNotesToggle) {
    // note card for Factors and Links with Notes
    data.nodes
      .get()
      .filter(
        (node) =>
          !node.hidden &&
          !node.nodeHidden &&
          node.note &&
          node.note !== 'Notes' &&
          !node.clusteredIn
      )
      .forEach((node) => {
        const box = network.getBoundingBox(node.id)
        drawTheBadge(noteImage, ctx, box.right, box.top)
      })
    // an edge note badge is placed where a middle arrow would be
    const changedEdges = []
    data.edges.get().forEach((edge) => {
      if (
        !edge.edgeHidden &&
        edge.note &&
        edge.note !== 'Notes' &&
        edge.arrows &&
        edge.arrows.middle &&
        !edge.arrows.middle.enabled
      ) {
        // there is a note, but the badge is not shown, so show it
        changedEdges.push(edge)
        edge.arrows.middle.enabled = true
        edge.arrows.middle.type = 'image'
        edge.arrows.middle.src = noteImage.src
      } else if (
        (!edge.note || (edge.note && edge.note === 'Notes') || edge.edgeHidden) &&
        edge.arrows &&
        edge.arrows.middle &&
        edge.arrows.middle.enabled
      ) {
        // there is not a note, but the badge is shown, so remove it
        changedEdges.push(edge)
        edge.arrows.middle.enabled = false
      }
    })
    data.edges.update(changedEdges)
  }
  // draw the voting thumbs up/down (but not for nodes inside a cluster, or for cluster nodes)
  if (showVotingToggle) {
    data.nodes
      .get()
      .filter((node) => !node.hidden && !node.nodeHidden && !node.clusteredIn && !node.isCluster)
      .forEach((node) => {
        const box = network.getBoundingBox(node.id)
        drawTheBadge(
          node.thumbUp?.includes(clientID) ? thumbUpFilledImage : thumbUpImage,
          ctx,
          box.left + 20,
          box.bottom
        )
        drawThumbCount(ctx, node.thumbUp, box.left + 36, box.bottom + 10)
        drawTheBadge(
          node.thumbDown?.includes(clientID) ? thumbDownFilledImage : thumbDownImage,
          ctx,
          box.right - 36,
          box.bottom
        )
        drawThumbCount(ctx, node.thumbDown, box.right - 20, box.bottom + 10)
      })
  }

  /**
   *
   * @param {image} badgeImage
   * @param {context} ctx
   * @param {number} x
   * @param {number} y
   */
  function drawTheBadge(badgeImage, ctx, x, y) {
    ctx.beginPath()
    ctx.drawImage(badgeImage, Math.floor(x), Math.floor(y))
  }
  /**
   * draw the length of the voters array, i.e. the count of those who have voted
   * @param {context} ctx
   * @param {array} voters
   * @param {number} x
   * @param {number} y
   */
  function drawThumbCount(ctx, voters, x, y) {
    if (voters) {
      ctx.beginPath()
      ctx.fillStyle = 'black'
      ctx.fillText(voters.length.toString(), x, y)
    }
  }
}

/**
 * Move the node to the nearest spot that it on the grid
 * @param {object} node
 */
function snapToGrid(node) {
  node.x = GRIDSPACING * Math.round(node.x / GRIDSPACING)
  node.y = GRIDSPACING * Math.round(node.y / GRIDSPACING)
}

/*************************************************************** clipboard ************************************** */
/**
 * Copy the selected nodes and links to the clipboard
 * NB this doesn't yet work in Firefox, as they haven't implemented the Clipboard API and Permissions yet.
 * @param {Event} event
 */
function copyToClipboard(event) {
  if (document.getSelection().toString()) return // only copy factors if there is no text selected (e.g. in Notes)
  event.preventDefault()
  if (drawingSwitch) {
    copyBackgroundToClipboard(event)
    return
  }
  const nIds = network.getSelectedNodes()
  const eIds = network.getSelectedEdges()
  if (nIds.length + eIds.length === 0) {
    alertMsg('Nothing selected to copy', 'warn')
    return
  }
  const nodes = []
  const edges = []
  nIds.forEach((nId) => {
    nodes.push(data.nodes.get(nId))
    const edgesFromNode = network.getConnectedEdges(nId)
    edgesFromNode.forEach((eId) => {
      const edge = data.edges.get(eId)
      if (nIds.includes(edge.to) && nIds.includes(edge.from) && !edges.find((e) => e.id === eId)) {
        edges.push(edge)
      }
    })
  })
  eIds.forEach((eId) => {
    const edge = data.edges.get(eId)
    if (!nodes.find((n) => n.id === edge.from)) nodes.push(data.nodes.get(edge.from))
    if (!nodes.find((n) => n.id === edge.to)) nodes.push(data.nodes.get(edge.to))
    if (!edges.find((e) => e.id === eId)) edges.push(data.edges.get(eId))
  })
  copyText(JSON.stringify({ nodes, edges }))
}
/**
 * copy the contents of the history log to the clipboard
 * @param {object} event
 */
function copyHistoryToClipboard(event) {
  event.preventDefault()
  const history = yHistory
    .toArray()
    .map(
      (rec) =>
        `${timeAndDate(rec.time, true)}\t${rec.user}\t${rec.action.replace(/\s+/g, ' ').trim()}\n`
    )
    .join('')
  copyText(history)
}
async function copyText(text) {
  try {
    if (typeof navigator.clipboard.writeText !== 'function') {
      throw new Error('navigator.clipboard.writeText not a function')
    }
  } catch {
    alertMsg('Copying not implemented in this browser', 'error')
    return false
  }
  try {
    await navigator.clipboard.writeText(text)
    alertMsg('Copied to clipboard', 'info')
    return true
  } catch (err) {
    console.error('Failed to copy: ', err)
    alertMsg('Copy failed', 'error')
    return false
  }
}

async function pasteFromClipboard() {
  if (drawingSwitch) {
    pasteBackgroundFromClipboard()
    return
  }
  const clip = await getClipboardContents()
  let nodes
  let edges
  try {
    ;({ nodes, edges } = JSON.parse(clip))
  } catch {
    // silently return (i.e. use system paste) if there is nothing relevant on the clipboard
    return
  }
  unSelect()
  nodes.forEach((node) => {
    const oldId = node.id
    node.id = uuidv4()
    node.x += 40
    node.y += 40
    edges.forEach((edge) => {
      if (edge.from === oldId) edge.from = node.id
      if (edge.to === oldId) edge.to = node.id
    })
  })
  edges.forEach((edge) => {
    edge.id = uuidv4()
  })
  data.nodes.add(nodes)
  data.edges.add(edges)
  network.setSelection({
    nodes: nodes.map((n) => n.id),
    edges: edges.map((e) => e.id),
  })
  showSelected()
  alertMsg('Pasted', 'info')
  logHistory('pasted factors and/or links from clipboard')
}

async function getClipboardContents() {
  try {
    if (typeof navigator.clipboard.readText !== 'function') {
      throw new Error('navigator.clipboard.readText not a function')
    }
  } catch {
    alertMsg('Pasting not implemented in this browser', 'error')
    return null
  }
  try {
    return await navigator.clipboard.readText()
  } catch (err) {
    console.error('Failed to read clipboard contents: ', err)
    alertMsg('Failed to paste', 'error')
    return null
  }
}

/* ----------------- dialogs for creating and editing nodes and links ----------------*/

/**
 * Initialise the dialog for creating nodes/edges
 * @param {string} popUpTitle
 * @param {number} height
 * @param {object} item
 * @param {function} cancelAction
 * @param {function} saveAction
 * @param {function} callback
 */
function initPopUp(popUpTitle, height, item, cancelAction, saveAction, callback) {
  inAddMode = false
  inEditMode = true
  changeCursor('default')
  elem('popup').style.height = `${height}px`
  elem('popup').style.borderColor = item.color.background
  elem('popup-operation').innerHTML = popUpTitle
  elem('popup-saveButton').onclick = saveAction.bind(this, item, callback)
  elem('popup-cancelButton').onclick = cancelAction.bind(this, item, callback)
  const popupLabel = elem('popup-label')
  popupLabel.style.fontSize = '14px'
  popupLabel.innerText = item.label === undefined ? '' : item.label //.replace(/\n/g, ' ')
  popupLabel.focus()
  // Set the cursor to the end
  setEndOfContenteditable(popupLabel)
  listen('popup', 'keydown', captureReturn)
  function captureReturn(e) {
    if (e.key === 'Enter' && !e.shiftKey) {
      elem('popup').removeEventListener('keydown', captureReturn)
      saveAction(item, callback)
    } else if (e.key === 'Escape') {
      elem('popup').removeEventListener('keydown', captureReturn)
      cancelAction(item, callback)
    }
  }
}
/**
 * Position the editing dialog box so that it is to the left of the item being edited,
 * but not outside the window
 * @param {Object} point
 */
function positionPopUp(point) {
  const popUp = elem('popup')
  popUp.style.display = 'block'
  // popup appears to the left of the given point
  popUp.style.top = `${point.y - popUp.offsetHeight / 2}px`
  const left = point.x - popUp.offsetWidth / 2 - 3
  popUp.style.left = `${left < 0 ? 0 : left}px`
  dragElement(popUp, elem('popup-top'))
}
/**
 * Hide the editing dialog box
 */
function clearPopUp() {
  elem('popup-saveButton').onclick = null
  elem('popup-cancelButton').onclick = null
  elem('popup-label').onkeyup = null
  elem('popup').style.display = 'none'
  if (elem('popup-node-editor')) elem('popup-node-editor').remove()
  if (elem('popup-link-editor')) elem('popup-link-editor').remove()
  if (elem('popup').timer) {
    clearTimeout(elem('popup').timer)
    elem('popup').timer = undefined
  }
  yAwareness.setLocalStateField('addingFactor', { state: 'done' })
  inEditMode = false
}
/**
 * User has pressed 'cancel' - abandon adding a node and hide the dialog
 * @param {Function} callback
 */
function cancelAdd(item, callback) {
  clearPopUp()
  callback(null)
  stopEdit()
}
/**
 * User has pressed 'cancel' - abandon the edit and hide the dialog
 * @param {object} item
 * @param {function} [callback]
 */
function cancelEdit(item, callback) {
  clearPopUp()
  item.label = item.oldLabel
  item.font.color = item.oldFontColor
  if (item.shape === 'portal') item.shape = 'image'
  if (item.from) {
    unlockEdge(item)
  } else {
    unlockNode(item)
  }
  if (callback) callback(null)
  stopEdit()
}
/**
 * A factor is being created:  get its label from the user
 * @param {Object} item - the node
 * @param {Function} cancelAction
 * @param {Function} callback
 */
function addLabel(item, cancelAction, callback) {
  if (elem('popup').style.display === 'block') return // can't add factor when factor is already being added
  initPopUp('Add Factor', 60, item, cancelAction, saveLabel, callback)
  const pos = network.canvasToDOM({ x: item.x, y: item.y })
  positionPopUp(pos)
  removeFactorCursor()
  ghostFactor(pos)
  elem('popup-label').focus()
}
/**
 * broadcast to other users that a new factor is being added here
 * @param {Object} pos offset coordinates of Add Factor dialog
 */
function ghostFactor(pos) {
  yAwareness.setLocalStateField('addingFactor', {
    state: 'adding',
    pos: network.DOMtoCanvas(pos),
    name: myNameRec.name,
  })
  elem('popup').timer = setTimeout(() => {
    // close it after a time if the user has gone away
    yAwareness.setLocalStateField('addingFactor', { state: 'done' })
  }, TIMETOEDIT)
}
/**
 * called when a node has been added.  Save the label provided
 * @param {Object} node the item that has been added
 * @param {Function} callback
 */
function saveLabel(node, callback) {
  node.label = splitText(elem('popup-label').innerText, NODEWIDTH)
  clearPopUp()
  if (node.label === '') {
    alertMsg('No label: cancelled', 'error')
    callback(null)
    return
  }
  network.manipulation.inMode = 'addNode' // ensure still in Add mode, in case others have done something meanwhile
  callback(node)
  logHistory(`added factor '${node.label}'`)
}
/**
 * Draw a dialog box for user to edit a node
 * @param {Object} item the node
 * @param {Object} point the centre of the node
 * @param {Function} cancelAction what to do if the edit is cancelled
 * @param {Function} callback what to do if the edit is saved
 */
function editNode(item, point, cancelAction, callback) {
  if (item.locked) return
  initPopUp('Edit Factor', 180, item, cancelAction, saveNode, callback)
  elem('popup').insertAdjacentHTML(
    'beforeend',
    `
		<div class="popup-node-editor" id="popup-node-editor">	
			<div>fill</div>
			<div>border</div>
			<div>font</div>
			<div class="input-color-container">
				<div class="color-well" id="node-backgroundColor"></div>
			</div>
			<div class="input-color-container">
				<div class="color-well" id="node-borderColor"></div>
			</div>
			<div class="input-color-container">
				<div class="color-well" id="node-fontColor"></div>
			</div>
			<div>
				<select name="nodeEditShape" id="nodeEditShape">
					<option value="box">Shape...</option>
					<option value="ellipse">Ellipse</option>
					<option value="circle">Circle</option>
					<option value="dot">Dot</option>
					<option value="box">Rect</option>
					<option value="diamond">Diamond</option>
					<option value="star">Star</option>
					<option value="triangle">Triangle</option>
					<option value="hexagon">Hexagon</option>
					<option value="text">Text</option>
					<option value="portal">Portal</option>
				</select>
			</div>
			<div>
				<select name="nodeEditBorder" id="node-borderType">
					<option value="solid" selected>Solid</option>
					<option value="dashed">Dashed</option>
					<option value="dots">Dotted</option>
					<option value="none">No border</option>
				</select>
			</div>
			<div>
				<select name="nodeEditFontSize" id="nodeEditFontSize">
					<option value="14">Size...</option>
					<option value="24">Large</option>
					<option value="14">Normal</option>
					<option value="10">Small</option>
				</select>
			</div>
			<div id="popup-sizer">
				<label
					>&nbsp;Size:
					<input type="range" class="xrange" id="nodeEditSizer" />
				</label>
			</div>
		</div>
		`
  )
  cp.createColorPicker('node-backgroundColor')
  elem('node-backgroundColor').style.backgroundColor = standardizeColor(item.color.background)
  if (item.shape === 'image' && !item.isCluster) {
    item.shape = 'portal'
    if (elem('popup-portal-room')) elem('popup-portal-room').value = item.portal
    else {
      makePortalInput(item.portal)
    }
  }
  elem('nodeEditShape').value = item.shape
  cp.createColorPicker('node-borderColor')
  elem('node-borderColor').style.backgroundColor = standardizeColor(item.color.border)
  cp.createColorPicker('node-fontColor')
  elem('node-fontColor').style.backgroundColor = standardizeColor(item.font.color)
  elem('node-borderType').value = getDashes(item.shapeProperties.borderDashes, item.borderWidth)
  elem('nodeEditFontSize').value = item.font.size

  elem('nodeEditSizer').value = factorSizeToPercent(item.size)
  progressBar(elem('nodeEditSizer'))
  listen('nodeEditSizer', 'input', (event) => progressBar(event.target))
  listen('nodeEditShape', 'change', (event) => {
    if (event.target.value === 'portal') {
      makePortalInput(item.portal)
    } else {
      elem('popup-portal-link')?.remove()
      item.portal = undefined
    }
  })
  positionPopUp(point)
  elem('popup-label').focus()
  elem('popup').timer = setTimeout(() => {
    //ensure that the node cannot be locked out for ever
    cancelEdit(item, callback)
    alertMsg('Edit timed out', 'warn')
  }, TIMETOEDIT)
  lockNode(item)
  /**
   * Generate HTML for the textarea to obtain the room name of the portal
   * @param {string} portal room name to go to
   */
  function makePortalInput(portal) {
    portal = portal || ''
    // expand the dialog to accommodate the textarea
    elem('popup').style.height = `${230}px`
    elem('popup-node-editor').insertAdjacentHTML(
      'beforeend',
      `<div id="popup-portal-link">
      				<label for="popup-portal-room">Map:</label>
      				<textarea id="popup-portal-room" rows="1" placeholder="ABC-DEF-GHI-JKL">${portal}</textarea>
    			</div>`
    )
  }
}
// fancy portal image icon
const portalSvg = `<svg viewBox="0 0 512 512" xmlns="http://www.w3.org/2000/svg" fill="#000000"><g id="SVGRepo_bgCarrier" stroke-width="0"></g><g id="SVGRepo_tracerCarrier" stroke-linecap="round" stroke-linejoin="round"></g><g id="SVGRepo_iconCarrier"><path fill="#ff0000" d="M298.736 21.016c-99.298 0-195.928 104.647-215.83 233.736-7.074 45.887-3.493 88.68 8.512 124.787-4.082-6.407-7.92-13.09-11.467-20.034-16.516-32.335-24.627-65.378-25-96.272-11.74 36.254-8.083 82.47 14.482 126.643 27.7 54.227 81.563 91.94 139.87 97.502 5.658.725 11.447 1.108 17.364 1.108 99.298 0 195.93-104.647 215.83-233.736 9.28-60.196.23-115.072-22.133-156.506 21.625 21.867 36.56 45.786 44.617 69.496.623-30.408-14.064-65.766-44.21-95.806-33.718-33.598-77.227-50.91-114.995-50.723-2.328-.118-4.67-.197-7.04-.197zm-5.6 36.357c40.223 0 73.65 20.342 95.702 53.533 15.915 42.888 12.51 108.315.98 147.858-16.02 54.944-40.598 96.035-79.77 126.107-41.79 32.084-98.447 24.39-115.874-5.798-1.365-2.363-2.487-4.832-3.38-7.385 11.724 14.06 38.188 14.944 61.817 1.3 25.48-14.71 38.003-40.727 27.968-58.108-10.036-17.384-38.826-19.548-64.307-4.837-9.83 5.676-17.72 13.037-23.14 20.934.507-1.295 1.043-2.59 1.626-3.88-18.687 24.49-24.562 52.126-12.848 72.417 38.702 45.923 98.07 25.503 140.746-6.426 37.95-28.392 72.32-73.55 89.356-131.988 1.265-4.34 2.416-8.677 3.467-13.008-.286 2.218-.59 4.442-.934 6.678-16.807 109.02-98.412 197.396-182.272 197.396-35.644 0-65.954-15.975-87.74-42.71-26.492-48.396-15.988-142.083 4.675-185.15 26.745-55.742 66.133-122.77 134.324-116.804 46.03 4.027 63.098 58.637 39.128 116.22-8.61 20.685-21.192 39.314-36.21 54.313 24.91-16.6 46.72-42.13 59.572-73 23.97-57.583 6.94-113.422-39.13-116.805-85.737-6.296-137.638 58.55-177.542 128.485-9.21 19.9-16.182 40.35-20.977 60.707.494-7.435 1.312-14.99 2.493-22.652C127.67 145.75 209.275 57.373 293.135 57.373z"></path></g></svg>`
/**
 * save the node format details that have been edited
 * @param {Object} item the node that has been edited
 * @param {Function} callback
 */
function saveNode(item, callback) {
  unlockNode(item)
  item.label = splitText(elem('popup-label').innerText, NODEWIDTH)
  if (item.label === '') {
    // if there is no label, cancel (nodes must have a label)
    alertMsg('No label: cancelled', 'error')
    callback(null)
  }
  let color = elem('node-backgroundColor').style.backgroundColor
  item.color.background = color
  item.color.highlight.background = color
  item.color.hover.background = color
  color = elem('node-borderColor').style.backgroundColor
  item.color.border = color
  item.color.highlight.border = color
  item.color.hover.border = color
  item.font.color = elem('node-fontColor').style.backgroundColor
  const borderType = elem('node-borderType').value
  item.borderWidth = borderType === 'none' ? 0 : borderType === 'solid' ? 1 : 4
  item.shapeProperties.borderDashes = convertDashes(borderType)
  item.shape = elem('nodeEditShape').value
  if (item.shape === 'portal') {
    item.portal = elem('popup-portal-room')?.value
    if (!item.portal) {
      alertMsg('No map room provided', 'error')
      callback(null)
      return
    }
    item.portal = item.portal.match(/[a-zA-Z]{3}-[a-zA-Z]{3}-[a-zA-Z]{3}-[a-zA-Z]{3}/)
    if (!item.portal) {
      alertMsg('Ill-formed map room provided', 'error')
      callback(null)
      return
    }
    item.portal = item.portal[0]
    item.shape = 'image'
    item.image = 'data:image/svg+xml;charset=utf-8,' + encodeURIComponent(portalSvg)
  }
  if (item.isCluster) {
    item.shape = 'image'
  }
  item.font.size = parseInt(elem('nodeEditFontSize').value)
  setFactorSizeFromPercent(item, elem('nodeEditSizer').value)
  network.manipulation.inMode = 'editNode' // ensure still in Add mode, in case others have done something meanwhile
  if (item.label.replace(/\s+|\n/g, '') === item.oldLabel.replace(/\s+|\n/g, '')) {
    logHistory(`edited factor: '${item.label}'`)
  } else logHistory(`edited factor, changing label from '${item.oldLabel}' to '${item.label}'`)
  clearPopUp()
  callback(item)
}
/**
 * User is about to edit the node.  Make sure that no one else can edit it simultaneously
 * @param {Node} item
 */
function lockNode(item) {
  item.locked = true
  item.opacity = 0.3
  item.oldLabel = item.label
  item.oldFontColor = item.font.color
  item.label = `${item.label}\n\n[Being edited by ${myNameRec.name}]`
  item.wasFixed = Boolean(item.fixed)
  item.fixed = true
  dontUndo = 'locked'
  data.nodes.update(item)
}
/**
 * User has finished editing the node.  Unlock it.
 * @param {Node} item
 */
function unlockNode(item) {
  item.locked = false
  item.opacity = 1
  item.fixed = item.wasFixed
  item.label = item.oldLabel
  dontUndo = 'unlocked'
  data.nodes.update(item)
  showNodeOrEdgeData()
}
/**
 * ensure that all factors and links are unlocked (called only when user leaves the page, to clear up for others)
 */
function unlockAll() {
  data.nodes.forEach((node) => {
    if (node.locked) cancelEdit(deepCopy(node))
  })
  data.edges.forEach((edge) => {
    if (edge.locked) cancelEdit(deepCopy(edge))
  })
}
/**
 * Draw a dialog box for user to edit an edge
 * @param {Object} item the edge
 * @param {Object} point the centre of the edge
 * @param {Function} cancelAction what to do if the edit is cancelled
 * @param {Function} callback what to do if the edit is saved
 */
function editEdge(item, point, cancelAction, callback) {
  if (item.locked) return
  initPopUp('Edit Link', 170, item, cancelAction, saveEdge, callback)
  elem('popup').insertAdjacentHTML(
    'beforeend',
    `<div class="popup-link-editor" id="popup-link-editor">
		<div>colour</div>
		<div></div>
		<div></div>
		<div class="input-color-container">
			<div class="color-well" id="linkEditLineColor"></div>
		</div>
		<div>
			<select name="linkEditWidth" id="linkEditWidth">
				<option value="1">Width: 1</option>
				<option value="4">Width: 4</option>
				<option value="8">Width: 8</option>
			</select>
		</div>
		<div>
			<select name="linkEditArrows" id="linkEditArrows">
				<option value="vee">Arrows...</option>
				<option value="vee">Sharp</option>
				<option value="arrow">Triangle</option>
				<option value="bar">Bar</option>
				<option value="circle">Circle</option>
				<option value="box">Box</option>
				<option value="diamond">Diamond</option>
				<option value="none">None</option>
			</select>
		</div>
		<div>
			<select name="linkEditDashes" id="linkEditDashes">
				<option value="solid" selected>Solid</option>
				<option value="dashedLinks">Dashed</option>
				<option value="dots">Dotted</option>
			</select>
		</div>
		<div>
			<i>Font size:</i>	
		</div>
		<div>
			<select id="linkEditFontSize">
				<option value="24">Large</option>
				<option value="14">Normal</option>
				<option value="10">Small</option>
			</select>
		</div>
	</div>
`
  )
  elem('popup').style.borderColor = item.color.color
  elem('linkEditWidth').value = parseInt(item.width)
  cp.createColorPicker('linkEditLineColor')
  elem('linkEditLineColor').style.backgroundColor = standardizeColor(item.color.color)
  elem('linkEditDashes').value = getDashes(item.dashes, null)
  elem('linkEditArrows').value = item.arrows.to.enabled ? item.arrows.to.type : 'none'
  elem('linkEditFontSize').value = parseInt(item.font.size)
  positionPopUp(point)
  elem('popup-label').focus()
  elem('popup').timer = setTimeout(() => {
    //ensure that the edge cannot be locked out for ever
    cancelEdit(item, callback)
    alertMsg('Edit timed out', 'warn')
  }, TIMETOEDIT)
  lockEdge(item)
}
/**
 * save the edge format details that have been edited
 * @param {Object} item the edge that has been edited
 * @param {Function} callback
 */
function saveEdge(item, callback) {
  unlockEdge(item)
  item.label = splitText(elem('popup-label').innerText, NODEWIDTH)
  if (item.label === '') item.label = ' '
  const color = elem('linkEditLineColor').style.backgroundColor
  item.color.color = color
  item.color.hover = color
  item.color.highlight = color
  item.width = parseInt(elem('linkEditWidth').value)
  if (!item.width) item.width = 1
  item.dashes = convertDashes(elem('linkEditDashes').value)
  item.arrows.to = {
    enabled: elem('linkEditArrows').value !== 'none',
    type: elem('linkEditArrows').value,
  }
  item.font.size = parseInt(elem('linkEditFontSize').value)
  network.manipulation.inMode = 'editEdge' // ensure still in edit mode, in case others have done something meanwhile
  // vis-network silently deselects all edges in the callback (why?).  So we have to mark this edge as unselected in preparation
  clearStatusBar()
  logHistory(
    `edited link from '${data.nodes.get(item.from).label}' to '${data.nodes.get(item.to).label}'`
  )
  clearPopUp()
  callback(item)
}
function lockEdge(item) {
  item.locked = true
  item.font.color = 'rgba(0,0,0,0.5)'
  item.opacity = 0.1
  item.oldLabel = item.label || ' '
  item.label = `Being edited by ${myNameRec.name}`
  dontUndo = 'locked'
  data.edges.update(item)
}
/**
 * User has finished editing the edge.  Unlock it.
 * @param {object} item
 */
function unlockEdge(item) {
  item.locked = false
  item.font.color = 'rgba(0,0,0,1)'
  item.opacity = 1
  item.label = item.oldLabel
  item.oldLabel = undefined
  dontUndo = 'unlocked'
  data.edges.update(item)
  showNodeOrEdgeData()
}
/* ----------------- end of node and edge creation and editing dialog -----------------*/

/**
 * if there is already a link from the 'from' node to the 'to' node, return it
 * @param {Object} from A node
 * @param {Object} to Another node
 */
function duplEdge(from, to) {
  return data.edges.get({
    filter: function (item) {
      return item.from === from && item.to === to
    },
  })
}

/**
 * Change the cursor style for the net pane and nav bar
 * @param {object} newCursorStyle
 */
function changeCursor(newCursorStyle) {
  if (inAddMode) return
  netPane.style.cursor = newCursorStyle
  elem('navbar').style.cursor = newCursorStyle
}
/**
 * User has set or changed the map title: update the UI and broadcast the new title
 * @param {event} e
 */
function mapTitle(e) {
  let title = e.target.innerText.trim()
  title = setMapTitle(title)
  yNetMap.set('mapTitle', title)
}
function pasteMapTitle(e) {
  e.preventDefault()
  let paste = (e.clipboardData || window.clipboardData).getData('text/plain')
  if (paste instanceof HTMLElement) paste = paste.textContent
  const selection = window.getSelection()
  if (!selection.rangeCount) return false
  selection.deleteFromDocument()
  selection.getRangeAt(0).insertNode(document.createTextNode(paste))
  setMapTitle(elem('maptitle').innerText)
}
/**
 * Format the map title
 * @param {string} title
 */
export function setMapTitle(title) {
  const div = elem('maptitle')
  clearStatusBar()
  if (!title) {
    title = 'Untitled map'
  }
  if (title === 'Untitled map') {
    div.classList.add('unsetmaptitle')
    document.title = appName
  } else {
    if (title.length > 50) {
      title = title.slice(0, 50)
      alertMsg('Map title is too long: truncated', 'warn')
    }
    div.classList.remove('unsetmaptitle')
    document.title = `${title}: ${shortAppName} map`
  }
  if (title !== div.innerText.trim()) div.innerText = title
  if (title.length >= 50) setEndOfContenteditable(div)
  setFileName()
  titleDropDown(title)
  return title
}
/**
 * Add this title to the local record of maps used
 * The list is stored as an object so that it is easy to add [room, title] pairs
 * and easy to modify the title of an existing room
 * @param {String} title
 */

const TITLELISTLEN = 500
function titleDropDown(title) {
  let recentMaps = localStorage.getItem('recents')
  if (recentMaps) recentMaps = JSON.parse(recentMaps)
  else recentMaps = {}
  //TODO this should be Map, not an object, to guarantee preservation of the insertion order
  if (title !== 'Untitled map') {
    recentMaps[room] = title
    // save only the most recent entries
    recentMaps = Object.fromEntries(Object.entries(recentMaps).slice(-TITLELISTLEN))
    localStorage.setItem('recents', JSON.stringify(recentMaps))
  }
  // if there is more than 1, append a down arrow after the map title as a cue to there being a list
  if (Object.keys(recentMaps).length > 1) elem('recent-rooms-caret').classList.remove('hidden')
}
/**
 * Create a drop down list of previous maps used for user selection
 */
function createTitleDropDown() {
  removeTitleDropDown()
  const selectList = document.createElement('ul')
  selectList.id = 'recent-rooms-select'
  selectList.classList.add('room-titles')
  elem('recent-rooms').appendChild(selectList)
  const recentMaps = JSON.parse(localStorage.getItem('recents'))
  // list is with New Map and then the most recent at the top
  if (recentMaps) {
    makeTitleDropDownEntry('*New map*', '*new*', false)
    const props = Object.keys(recentMaps).reverse()
    props.forEach((prop) => {
      makeTitleDropDownEntry(recentMaps[prop], prop)
    })
  }
  /**
   * create a previous map menu item
   * @param {string} name Title of map
   * @param {string} room
   */
  function makeTitleDropDownEntry(name, room) {
    const li = document.createElement('li')
    li.classList.add('room-title')
    li.textContent = name
    li.dataset.room = room
    li.addEventListener('click', (event) => changeRoom(event))
    selectList.appendChild(li)
  }
}
/**
 * User has clicked one of the previous map titles - confirm and change to the web page for that room
 * @param {Event} event
 */
function changeRoom(event) {
  if (data.nodes.length > 0) {
    if (!confirm('Are you sure you want to move to a different map?')) return
  }
  const newRoom = event.target.dataset.room
  removeTitleDropDown()
  const url = new URL(document.location)
  url.search = newRoom !== '*new*' ? `?room=${newRoom}` : ''
  window.location.replace(url)
}
/**
 * Remove the drop down list of previous maps if user clicks on the net-pane or on a map title.
 */
function removeTitleDropDown() {
  const oldSelect = elem('recent-rooms-select')
  if (oldSelect) oldSelect.remove()
}
/**
 * unselect all nodes and edges
 */
export function unSelect() {
  hideNotes()
  network.unselectAll()
  clearStatusBar()
}
/* 
  ----------- Calculate statistics in the background -------------
*/
// set  up a web worker to calculate network statistics in parallel with whatever
// the user is doing
const worker = new Worker(new URL('./betweenness.js', import.meta.url), { type: 'module' })
/**
 * Ask the web worker to recalculate network statistics
 */
export function recalculateStats() {
  // wait 200 mSecs for things to settle down before recalculating
  setTimeout(() => {
    worker.postMessage([nodes.get(), edges.get()])
  }, 200)
}
worker.onmessage = function (e) {
  if (typeof e.data === 'string') alertMsg(e.data, 'error')
  else {
    const nodesToUpdate = []
    data.nodes.get().forEach((n) => {
      if (n.bc !== e.data[n.id]) {
        n.bc = e.data[n.id]
        nodesToUpdate.push(n)
      }
    })
    if (nodesToUpdate) {
      data.nodes.update(nodesToUpdate)
    }
  }
}
/* 
  ----------- Status messages ---------------------------------------
*/
/**
 * return a string listing the labels of the given nodes, with nice connecting words
 * @param {Array} factors List of node Ids
 * @param {Boolean} suppressType If true, don't start string with 'Factors'
 */
function listFactors(factors, suppressType) {
  if (factors.length > 5) return `${factors.length} factors`
  let str = ''
  if (!suppressType) {
    str = 'Factor'
    if (factors.length > 1) str = `${str}s`
    str = `${str}: `
  }
  return str + lf(factors)

  function lf(factors) {
    // recursive fn to return a string of the node labels, separated by commas and 'and'
    const n = factors.length
    const label = `'${shorten(data.nodes.get(factors[0]).label)}'`
    if (n === 1) return label
    factors.shift()
    if (n === 2) return label.concat(` and ${lf(factors)}`)
    return label.concat(`, ${lf(factors)}`)
  }
}

/**
 * return a string listing the number of Links, or if just one, the starting and ending factors
 * @param {Array} links
 */
function listLinks(links) {
  if (links.length > 1) return `${links.length} links`
  const link = data.edges.get(links[0])
  return `Link from "${shorten(data.nodes.get(link.from).label)}" to "${shorten(data.nodes.get(link.to).label)}"`
}
/**
 * returns string of currently selected labels of links and factors, nicely formatted
 * @returns {String} string of labels of links and factors, nicely formatted
 */
function selectedLabels() {
  const selectedNodes = network.getSelectedNodes()
  const selectedEdges = network.getSelectedEdges()
  let msg = ''
  if (selectedNodes.length > 0) msg = listFactors(selectedNodes)
  if (selectedNodes.length > 0 && selectedEdges.length > 0) msg += ' and '
  if (selectedEdges.length > 0) msg += listLinks(selectedEdges)
  return msg
}
/**
 * show the nodes and links selected in the status bar
 */
function showSelected() {
  const msg = selectedLabels()
  if (msg.length > 0) statusMsg(`${msg} selected`)
  else clearStatusBar()
  toggleDeleteButton()
}
/* ----------------------------------------zoom slider -------------------------------------------- */
Network.prototype.zoom = function (scale) {
  const newScale = scale === undefined ? 1 : scale < 0.001 ? 0.001 : scale
  const animationOptions = {
    scale: newScale,
    animation: {
      duration: 0,
    },
  }
  this.view.moveTo(animationOptions)
  zoomCanvas(newScale)
}

/**
 * rescale and redraw the network so that it fits the pane
 */
export function fit() {
  const prevPos = network.getViewPosition()
  network.fit({
    position: { x: 0, y: 0 }, // fit to centre of canvas
  })
  const newPos = network.getViewPosition()
  const newScale = network.getScale()
  zoomCanvas(1.0)
  panCanvas(prevPos.x - newPos.x, prevPos.y - newPos.y, 1.0)
  zoomCanvas(newScale)
  elem('zoom').value = newScale
  network.storePositions()
}
/**
 * expand/reduce the network view using the value in the zoom slider
 */
function zoomnet() {
  network.zoom(Number(elem('zoom').value))
}
/**
 * zoom by the given amount (+ve or -ve);
 * used by the + and - buttons at the ends of the zoom slider
 * and by trackpad zoom/pinch.
 * If the new zoom level becomes below zero, do nothing
 * @param {Number} incr
 */
function zoomincr(incr) {
  let newScale = network.getScale() * (1 + incr)
  if (newScale <= 0) newScale = 0.015
  if (newScale <= 4 && newScale >= 0) {
    elem('zoom').value = newScale
  }
  network.zoom(newScale)
}
/**
 * Set up pinch-to-zoom using native touch events
 */
function setUpPinchZoom() {
  let initialDistance = null
  let initialScale = 1

  function getTouchDistance(touch1, touch2) {
    const dx = touch1.clientX - touch2.clientX
    const dy = touch1.clientY - touch2.clientY
    return Math.sqrt(dx * dx + dy * dy)
  }

  netPane.addEventListener(
    'touchstart',
    (e) => {
      if (e.touches.length === 2) {
        e.preventDefault()
        initialDistance = getTouchDistance(e.touches[0], e.touches[1])
        initialScale = Number(elem('zoom').value)
      }
    },
    { passive: false }
  )

  netPane.addEventListener(
    'touchmove',
    (e) => {
      if (e.touches.length === 2 && initialDistance) {
        e.preventDefault()
        const currentDistance = getTouchDistance(e.touches[0], e.touches[1])
        const scale = currentDistance / initialDistance
        let newZoom = initialScale * scale
        if (newZoom > 4) newZoom = 4
        if (newZoom <= 0.015) newZoom = 0.015
        elem('zoom').value = newZoom
        network.zoom(newZoom)
      }
    },
    { passive: false }
  )

  netPane.addEventListener('touchend', (e) => {
    if (e.touches.length < 2) {
      initialDistance = null
    }
  })

  netPane.addEventListener('touchcancel', () => {
    initialDistance = null
  })
}

let clicks = 0 // accumulate 'mousewheel' clicks sent while display is updating
let ticking = false // if true, we are waiting for an AnimationFrame */
// see https://www.html5rocks.com/en/tutorials/speed/animations/

// listen for zoom/pinch (confusingly, referred to as mousewheel events)
listen(
  'net-pane',
  'wheel',
  (e) => {
    e.preventDefault()
    // reject all but vertical touch movements
    if (Math.abs(e.deltaX) <= 1) zoomscroll(e)
  },
  // must be passive, else pinch/zoom is intercepted by the browser itself
  { passive: false }
)
/**
 * Zoom using a trackpad (with a mousewheel or two fingers)
 * @param {Event} event
 */
function zoomscroll(event) {
  clicks += event.deltaY
  requestZoom()
}
function requestZoom() {
  if (!ticking) requestAnimationFrame(zoomUpdate)
  ticking = true
}
const MOUSEWHEELZOOMRATE = 0.01 // how many 'clicks' of the mouse wheel/finger track correspond to 1 zoom increment
function zoomUpdate() {
  zoomincr(-clicks * MOUSEWHEELZOOMRATE)
  ticking = false
  clicks = 0
}

/* -----------Operations related to the top button bar (not the side panel)------------- */

/**
 * react to the user pressing the Add node button
 * handles cases when the button is disabled; has previously been pressed; and the Add link
 * button is active, as well as the normal case
 *
 */
function plusNode() {
  switch (inAddMode) {
    case 'disabled':
      return
    case 'addNode': {
      removeFactorCursor()
      showPressed('addNode', 'remove')
      stopEdit()
      break
    }
    case 'addLink': {
      showPressed('addLink', 'remove')
      stopEdit()
    } // falls through
    default:
      // false
      // don't allow user to add a factor while editing another one
      if (elem('popup').style.display === 'block') break
      network.unselectAll()
      changeCursor('cell')
      ghostCursor()
      inAddMode = 'addNode'
      showPressed('addNode', 'add')
      unSelect()
      statusMsg('Click on the map to add a factor')
      network.addNodeMode()
  }
}
/**
 * show a box attached to the cursor to guide where the Factor will be placed when the user clicks.
 */
function ghostCursor() {
  // no ghost cursor if the hardware only supports touch
  if (!window.matchMedia('(any-hover: hover)').matches) return
  const box = document.createElement('div')
  box.classList.add('ghost-factor', 'factor-cursor')
  box.innerText = 'Click on the map to add a factor'
  box.id = 'factor-cursor'
  document.body.appendChild(box)
  const netPaneRect = netPane.getBoundingClientRect()
  keepInWindow(box, netPaneRect)
  document.addEventListener('pointermove', () => {
    keepInWindow(box, netPaneRect)
  })
  function keepInWindow(box, netPaneRect) {
    const boxHalfWidth = box.offsetWidth / 2
    const boxHalfHeight = box.offsetHeight / 2
    const left = window.event.pageX - boxHalfWidth
    box.style.left = `${
      left <= netPaneRect.left
        ? netPaneRect.left
        : left >= netPaneRect.right - box.offsetWidth
          ? netPaneRect.right - box.offsetWidth
          : left
    }px`
    const top = window.event.pageY - boxHalfHeight
    box.style.top = `${
      top <= netPaneRect.top
        ? netPaneRect.top
        : top >= netPaneRect.bottom - box.offsetHeight
          ? netPaneRect.bottom - box.offsetHeight
          : top
    }px`
  }
}
/**
 * remove the factor cursor if it exists
 */
function removeFactorCursor() {
  const factorCursor = elem('factor-cursor')
  if (factorCursor) {
    factorCursor.remove()
  }
  clearStatusBar()
}
/**
 * react to the user pressing the Add Link button
 * handles cases when the button is disabled; has previously been pressed; and the Add Node
 * button is active, as well as the normal case
 */
function plusLink() {
  switch (inAddMode) {
    case 'disabled':
      return
    case 'addLink': {
      showPressed('addLink', 'remove')
      stopEdit()
      break
    }
    case 'addNode': {
      showPressed('addNode', 'remove')
      stopEdit() // falls through
    } // falls through
    default:
      // false
      // don't allow user to add a factor while editing another one
      if (elem('popup').style.display === 'block') break
      removeFactorCursor()
      if (data.nodes.length < 2) {
        alertMsg('Two Factors needed to link', 'error')
        break
      }
      changeCursor('crosshair')
      inAddMode = 'addLink'
      showPressed('addLink', 'add')
      unSelect()
      statusMsg(
        'Now drag from the middle of the Source factor to the middle of the Destination factor'
      )
      network.setOptions({
        interaction: { dragView: false, selectable: false },
      })
      network.addEdgeMode()
  }
}
/**
 * cancel adding node and links
 */
function stopEdit() {
  inAddMode = false
  network.disableEditMode()
  network.setOptions({
    interaction: { dragView: true, selectable: true },
  })
  clearStatusBar()
  changeCursor('default')
}
/**
 * Add or remove the CSS style showing that the button has been pressed
 * @param {string} el the Id of the button
 * @param {*} action whether to add or remove the style
 *
 */
function showPressed(el, action) {
  elem(el).children.item(0).classList[action]('pressed')
}

function undo() {
  if (buttonIsDisabled('undo')) return
  unSelect()
  yUndoManager.undo()
  logHistory('undid last action')
  undoRedoButtonStatus()
}

function redo() {
  if (buttonIsDisabled('redo')) return
  unSelect()
  yUndoManager.redo()
  logHistory('redid last action')
  undoRedoButtonStatus()
}

export function undoRedoButtonStatus() {
  setButtonDisabledStatus('undo', yUndoManager.undoStack.length === 0)
  setButtonDisabledStatus('redo', yUndoManager.redoStack.length === 0)
}
/**
 * Returns true if the button is not disabled
 * @param {String} id
 * @returns Boolean
 */
function buttonIsDisabled(id) {
  return elem(id).classList.contains('disabled')
}
/**
 * Change the visible state of a button
 * @param {String} id
 * @param {Boolean} state - true to make the button disabled
 */
function setButtonDisabledStatus(id, state) {
  if (state) elem(id).classList.add('disabled')
  else elem(id).classList.remove('disabled')
}
/**
 * Delete the selected node, plus all the edges that connect to it (so no edge is left dangling)
 */
function deleteNode() {
  network.deleteSelected()
  clearStatusBar()
  toggleDeleteButton()
}

/**
 * set up the modal dialog that opens when the user clicks the Share icon in the nav bar
 */
function setUpShareDialog() {
  const modal = elem('shareModal')
  const inputElem = elem('text-to-copy')
  const copiedText = elem('copied-text')

  // When the user clicks the button, open the modal
  listen('share', 'click', () => {
    const path = `${window.location.pathname}?room=${room}`
    const linkToShare = window.location.origin + path
    copiedText.style.display = 'none'
    modal.style.display = 'block'
    inputElem.cols = linkToShare.length.toString()
    inputElem.value = linkToShare
    inputElem.style.height = `${inputElem.scrollHeight - 3}px`
    inputElem.select()
    network.storePositions()
  })
  listen('clone-button', 'click', () => openWindow('clone'))
  listen('view-button', 'click', () => openWindow('view'))
  listen('table-button', 'click', () => openWindow('table'))
  // When the user clicks on <span> (x), close the modal
  listen('modal-close', 'click', (event) => closeShareDialog(event))
  // When the user clicks anywhere on the background, close it
  listen('shareModal', 'click', (event) => closeShareDialog(event))

  listen('copy-text', 'click', (e) => {
    e.preventDefault()
    // Select the text
    inputElem.select()
    if (copyText(inputElem.value)) // Display the copied text message
    {
      copiedText.style.display = 'inline-block'
    }
  })

  function openWindow(type) {
    let path = ''
    switch (type) {
      case 'clone': {
        doClone(false)
        break
      }
      case 'view': {
        doClone(true)
        break
      }
      case 'table': {
        path = `${window.location.pathname.replace('prsm.html', 'table.html')}?room=${room}`
        window.open(path, '_blank')
        break
      }
      default:
        console.log('Bad case in openWindow()')
        break
    }
    modal.style.display = 'none'
  }

  function closeShareDialog(event) {
    if (event.target === modal || event.target === elem('modal-close')) {
      modal.style.display = 'none'
    }
  }
}
function doClone(onlyView) {
  // undo any ongoing analysis and unselect all nodes and edges
  setRadioVal('radius', 'All')
  setRadioVal('stream', 'All')
  setRadioVal('paths', 'All')
  analyse()
  unSelect()

  const options = {
    created: {
      action: `cloned this map from room: ${room + (onlyView ? ' (Read Only)' : '')}`,
      actor: myNameRec.name,
    },
    viewOnly: onlyView,
  }
  // save state as a UTF16 string
  const state = saveState(options)
  // save it in local storage
  localForage
    .setItem('clone', state)
    .then(() => {
      // make a room id
      const clonedRoom = generateRoom()
      // open a new map
      let path = `${window.location.pathname}?room=${clonedRoom}`
      const debugType = new URL(window.location.href).searchParams.get('debug')
      if (onlyView && elem('addCopyButton').checked) path += '&copyButton'
      if (debugType) path += `&debug=${debugType}`
      window.open(path, '_blank')
      logHistory(
        `made a ${onlyView ? 'read-only copy' : 'clone'} of the map into room: ${clonedRoom}`
      )
    })
    .catch(function (err) {
      console.log('Error saving clone to local storage:', err)
    })
}
function mergeMap() {
  elem('mergedRoom').value = ''
  elem('mergeDialog').showModal()
}
function doMerge() {
  const path = elem('mergedRoom').value
  if (!path) {
    alertMsg('No map given to merge', 'error')
    return
  }
  try {
    const url = new URL(path)
    const roomToMerge = url.searchParams.get('room')
    console.log('merging ', roomToMerge)
    mergeRoom(roomToMerge)
    logHistory(`merged map from room: ${roomToMerge}`)
  } catch {
    alertMsg('Invalid map URL', 'error')
    return
  }
  elem('mergeDialog').close()
}
/* ----------------------------------------------------------- Search ------------------------------------------------------*/
/**
 * Open an input for user to type label of node to search for and generate suggestions when user starts typing
 */
function search() {
  const searchBar = elem('search-bar')
  if (searchBar.style.display === 'block') hideSearchBar()
  else {
    searchBar.style.display = 'block'
    elem('search-icon').style.display = 'block'
    searchBar.focus()
    listen('search-bar', 'keyup', searchTargets)
  }
}
/**
 * generate and display a set of suggestions - nodes with labels that include the substring that the user has typed
 */
function searchTargets() {
  let str = elem('search-bar').value
  if (!str || str === ' ') {
    if (elem('targets')) elem('targets').remove()
    return
  }
  let targets = elem('targets')
  if (targets) targets.remove()
  targets = document.createElement('ul')
  targets.id = 'targets'
  targets.classList.add('search-ul')
  str = str.toLowerCase()
  const suggestions = data.nodes.get().filter((n) => n.label.toLowerCase().includes(str))
  suggestions.forEach((n) => {
    const li = document.createElement('li')
    li.classList.add('search-suggestion')
    const div = document.createElement('div')
    div.classList.add('search-suggestion-text')
    div.innerText = n.label.replace(/\n/g, ' ')
    div.dataset.id = n.id
    div.addEventListener('click', (event) => doSearch(event))
    li.appendChild(div)
    targets.appendChild(li)
  })
  elem('suggestion-list').appendChild(targets)
}
/**
 * do the search using the string in the search bar and, when found, focus on that node
 */
function doSearch(event) {
  const nodeId = event.target.dataset.id
  if (nodeId) {
    const prevPos = network.getViewPosition()
    network.focus(nodeId, { scale: 1.5 })
    const newPos = network.getViewPosition()
    const newScale = network.getScale()
    zoomCanvas(1.0)
    panCanvas(prevPos.x - newPos.x, prevPos.y - newPos.y, 1.0)
    zoomCanvas(newScale)
    elem('zoom').value = newScale
    network.storePositions()
    hideSearchBar()
  }
}
function hideSearchBar() {
  const searchBar = elem('search-bar')
  if (elem('targets')) elem('targets').remove()
  searchBar.value = ''
  searchBar.style.display = 'none'
  elem('search-icon').style.display = 'none'
}

/**
 * show or hide the side panel
 */
function togglePanel() {
  if (container.panelHidden) {
    panel.classList.remove('hide')
    positionNotes()
  } else {
    panel.classList.add('hide')
  }
  container.panelHidden = !container.panelHidden
}
dragElement(elem('panel'), elem('panelHeader'))

/* ------------------------------------------------operations related to the side panel -------------------------------------*/

/**
 * when the window is resized, make sure that the pane is still visible
 * @param {HTMLelement} pane
 */
function keepPaneInWindow(pane) {
  if (pane.offsetLeft + pane.offsetWidth > container.offsetLeft + container.offsetWidth) {
    pane.style.left = `${container.offsetLeft + container.offsetWidth - pane.offsetWidth}px`
  }
  if (pane.offsetTop + pane.offsetHeight > container.offsetTop + container.offsetHeight) {
    pane.style.top = `${
      container.offsetTop +
      container.offsetHeight -
      pane.offsetHeight -
      document.querySelector('footer').offsetHeight
    }px`
  }
}

function openTab(tabId, evt) {
  const tabcontent = document.getElementsByClassName('tabcontent')
  // Get all elements with class="tabcontent" and hide them by moving them off screen
  for (let i = 0; i < tabcontent.length; i++) {
    tabcontent[i].classList.add('hide')
  }
  // Get all elements with class="tablinks" and remove the class "active"
  const tablinks = document.getElementsByClassName('tablinks')
  for (let i = 0; i < tablinks.length; i++) {
    tablinks[i].className = tablinks[i].className.replace(' active', '')
  }
  // Show the current tab, and add an "active" class to the button that opened the tab
  elem(tabId).classList.remove('hide')
  evt.currentTarget.className += ' active'
  clearStatusBar()
  // if a Notes panel is in the way, move it
  positionNotes()
}

// Factors and Links Tabs
function applySampleToNode(event) {
  if (event.detail !== 1) return // only process single clicks here
  const selectedNodeIds = network.getSelectedNodes()
  if (selectedNodeIds.length === 0) return
  const nodesToUpdate = []
  const sample = event.currentTarget.groupNode
  for (let node of data.nodes.get(selectedNodeIds)) {
    if (node.isCluster) break
    if (sample !== node.grp) {
      node = deepMerge(node, styles.nodes[sample])
      node.grp = sample
      node.modified = timestamp()
      nodesToUpdate.push(node)
    }
  }
  data.nodes.update(nodesToUpdate)
  const nNodes = nodesToUpdate.length
  if (nNodes) {
    logHistory(
      `applied ${styles.nodes[sample].groupLabel} style to ${
        nNodes === 1 ? nodesToUpdate[0].label : nNodes + ' factors'
      }`
    )
  }
  lastNodeSample = sample
}
/**
 * Apply the sample's format to the selected links
 * @param {event} event
 */
function applySampleToLink(event) {
  if (event.detail !== 1) return // only process single clicks here
  const sample = event.currentTarget.groupLink
  const selectedEdges = network.getSelectedEdges()
  if (selectedEdges.length === 0) return
  const edgesToUpdate = []
  for (let edge of data.edges.get(selectedEdges)) {
    if (edge.isClusterEdge) break
    if (sample !== edge.grp) {
      edge = deepMerge(edge, styles.edges[sample])
      edge.grp = sample
      edge.modified = timestamp()
      edgesToUpdate.push(edge)
    }
  }
  data.edges.update(edgesToUpdate)
  const nEdges = edgesToUpdate.length
  if (nEdges) {
    logHistory(
      `applied ${styles.edges[sample].groupLabel} style to ${nEdges} link${nEdges === 1 ? '' : 's'} `
    )
  }
  lastLinkSample = sample
}
/**
 * Remember the last style sample that the user clicked and use this for future factors/links
 * Mark the sample with a light blue border
 * @param {number} nodeId
 * @param {number} linkId
 */
export function updateLastSamples(nodeId, linkId) {
  if (nodeId) {
    lastNodeSample = nodeId
    const sampleNodes = Array.from(document.getElementsByClassName('sampleNode'))
    const node = sampleNodes.filter((e) => e.groupNode === nodeId)[0]
    sampleNodes.forEach((n) => n.classList.remove('sampleSelected'))
    node.classList.add('sampleSelected')
  }
  if (linkId) {
    lastLinkSample = linkId
    const sampleLinks = Array.from(document.getElementsByClassName('sampleLink'))
    const link = sampleLinks.filter((e) => e.groupLink === linkId)[0]
    sampleLinks.forEach((n) => n.classList.remove('sampleSelected'))
    link.classList.add('sampleSelected')
  }
}

/**
 * Hide or reveal all the Factors or Links with the given style
 * @param {Object} obj {sample: state}
 */
function updateFactorsOrLinksHiddenByStyle(obj) {
  for (const sampleElementId in obj) {
    const sampleElement = elem(sampleElementId)
    const state = obj[sampleElementId]
    sampleElement.dataset.hide = state ? 'hidden' : 'visible'
    sampleElement.style.opacity = state ? 0.6 : 1.0
  }
}

/********************************************************Notes********************************************** */
/**
 * Globally either display or don't display notes when a factor or link is selected
 * @param {Event} e
 */
function showNotesSwitch(e) {
  showNotesToggle = e.target.checked
  doShowNotes(showNotesToggle)
  yNetMap.set('showNotes', showNotesToggle)
}
function doShowNotes(toggle) {
  elem('showNotesSwitch').checked = toggle
  showNotesToggle = toggle
  network.redraw()
  showNodeOrEdgeData()
}
/**
 * User has clicked the padlock.  Toggle padlock state and fix the location of the node
 */
function setFixed() {
  if (viewOnly) return
  const locked = elem('fixed').style.display === 'none'
  const node = data.nodes.get(editor.id)
  node.fixed = locked
  elem('fixed').style.display = node.fixed ? 'inline' : 'none'
  elem('unfixed').style.display = node.fixed ? 'none' : 'inline'
  data.nodes.update(node)
}
/**
 * Display a panel to show info about the selected edge or node
 */
function showNodeOrEdgeData() {
  hideNotes()
  if (!showNotesToggle) return
  if (network.getSelectedNodes().length === 1) showNodeData()
  else if (network.getSelectedEdges().length === 1) showEdgeData()
}
/**
 * open another window (popupWindow) in which Notes can be edited
 */
function openNotesWindow() {
  popupWindow = window.open(
    './dist/NoteWindow.html',
    'popupWindowName',
    'toolbar=no,width=600,height=600'
  )
}
/**
 * Hide the Node or Edge Data panel
 */
function hideNotes() {
  if (editor == null) return
  let notesPanel = document.getElementById('nodeNotePanel')
  if (notesPanel.classList.contains('hide')) notesPanel = document.getElementById('edgeNotePanel')
  if (notesPanel.classList.contains('hide')) return
  notesPanel.classList.add('hide')
  document.getSelection().removeAllRanges()
  notesPanel.querySelector('.ql-toolbar').remove()
  editor = null
  if (popupWindow) popupWindow.close()
}
/**
 * Show the notes box and the fixed node check box
 * @param {integer} nodeId
 */
function showNodeData(nodeId) {
  const panel = elem('nodeNotePanel')
  nodeId = nodeId || network.getSelectedNodes()[0]
  const node = data.nodes.get(nodeId)
  elem('fixed').style.display = node.fixed && !viewOnly ? 'inline' : 'none'
  elem('unfixed').style.display = node.fixed || viewOnly ? 'none' : 'inline'
  elem('nodeLabel').innerHTML = node.label ? shorten(node.label) : ''
  if (node.created) {
    elem('nodeCreated').innerHTML = `${timeAndDate(node.created.time)} by ${node.created.user}`
    elem('nodeCreation').style.display = 'flex'
  } else elem('nodeCreation').style.display = 'none'
  if (node.modified) {
    elem('nodeModified').innerHTML = `${timeAndDate(node.modified.time)} by ${node.modified.user}`
    elem('nodeModification').style.display = 'flex'
  } else elem('nodeModification').style.display = 'none'
  editor = new Quill('#node-notes', {
    modules: {
      toolbar: viewOnly
        ? null
        : [
            'bold',
            'italic',
            'underline',
            'link',
            { list: 'ordered' },
            { list: 'bullet' },
            { indent: '-1' },
            { indent: '+1' },
          ],
    },
    placeholder: 'Notes',
    theme: 'snow',
    readOnly: viewOnly,
  })
  window.editor = editor // used by popupEditor to access this editor
  editor.id = node.id
  if (node.note) {
    if (node.note instanceof Object) editor.setContents(node.note)
    else editor.setText(node.note)
  } else editor.setText('')
  editor.on('text-change', (delta, oldDelta, source) => {
    if (source === 'user') {
      data.nodes.update({
        id: nodeId,
        note: isQuillEmpty(editor) ? '' : editor.getContents(),
        modified: timestamp(),
      })
      if (popupWindow) {
        popupEditor = popupWindow.popupEditor
        if (popupEditor) popupEditor.setContents(editor.getContents())
      }
    }
  })
  panel.classList.remove('hide')
  positionNotes()
}
/**
 * user has clicked on the sparkles icon in a Factor window
 * return the output from an LLM asked to explain the factor
 */
async function genAINode() {
  alertMsg('Processing...', 'info', true)
  editor.setText('Processing...\n')
  const sparklesElem = elem('sparklesNode')
  sparklesElem.classList.add('rotating')
  const nodeId = network.getSelectedNodes()[0]
  const node = data.nodes.get(nodeId)
  const context = data.nodes
    .get()
    .map((n) => n.label.replaceAll('\n', ' '))
    .join(', ')
  const systemPrompt = `You are to explain a single factor, A.
Follow all the instructions below for your response:

Write a compact, clear explanation of what A is, what it measures and how it may vary.

Structure the output in the following order without using any section headings or a title:
1 Introduction
2 Bullet points explaining features of A.
3 Evidence section containing academic citations and links to relevant web pages
4 Summary
Always begin the answer with: > ***This text has been generated by AI. It needs to be checked carefully.***
Do not begin by repeating the factor that is being explained.
The introduction should be no more than 50 words.
The bullet points should clearly explain features of A in no more than 150 words.
The evidence section should provide academic references that support the explanation, with a brief explanation of how each reference supports it.
All academic references must include full citations (author, date, title, source). Do not include DOIs.  You must verify that all academic references that you mention actually exist.
Always embed links for any referenced web pages.
The summary should concisely restate the main points in no more than 50 words.

Maximum total output length: 200 words.
Format everything in Markdown.`
  const aiResponse = await getAIresponse(
    `Explain ${node.label}. Use these keywords as context: ${context}`,
    systemPrompt
  )
  editor.setContents(aiResponse)
  const modified = timestamp()
  data.nodes.update({
    id: nodeId,
    note: isQuillEmpty(editor) ? '' : editor.getContents(),
    modified,
  })
  elem('nodeModified').innerHTML = `${timeAndDate(modified.time)} by ${modified.user}`
  positionNotes()
  sparklesElem.classList.remove('rotating')
  if (popupWindow) {
    popupEditor = popupWindow.popupEditor
    if (popupEditor) popupEditor.setContents(editor.getContents())
  }
  cancelAlertMsg()
}
/**
 * Make the notes panel resizeable by dragging its corner handles
 * @param {HTMLElement} notePanel
 */
function makeNotesPanelResizeable(notePanel) {
  const notePanelCornerHandle = notePanel.querySelector('.corner-handle')
  const notePanelTopLeftHandle = notePanel.querySelector('.corner-handle-top-left')

  let isResizingCorner = false
  let isResizingTopLeft = false
  let startX = 0
  let startY = 0
  let startWidth = 0
  let startHeight = 0
  let startLeft = 0
  let startTop = 0

  // Bottom-right corner handle
  notePanelCornerHandle.addEventListener('pointerdown', (e) => {
    isResizingCorner = true
    startX = e.clientX
    startY = e.clientY

    const styles = window.getComputedStyle(notePanel)
    startWidth = parseInt(styles.width, 10)
    startHeight = parseInt(styles.height, 10)

    document.body.style.userSelect = 'none'
    // Prevent default touch behaviors like scrolling
    notePanelCornerHandle.style.touchAction = 'none'
  })

  // Top-left corner handle
  notePanelTopLeftHandle.addEventListener('pointerdown', (e) => {
    isResizingTopLeft = true
    startX = e.clientX
    startY = e.clientY

    const styles = window.getComputedStyle(notePanel)
    startWidth = parseInt(styles.width, 10)
    startHeight = parseInt(styles.height, 10)
    startLeft = parseInt(styles.left, 10)
    startTop = parseInt(styles.top, 10)

    document.body.style.userSelect = 'none'
    // Prevent default touch behaviors like scrolling
    notePanelTopLeftHandle.style.touchAction = 'none'
  })

  document.addEventListener('pointermove', (e) => {
    if (isResizingCorner) {
      const dx = e.clientX - startX
      const dy = e.clientY - startY

      const newWidth = startWidth + dx
      const newHeight = startHeight + dy

      if (newWidth > 150) notePanel.style.width = newWidth + 'px'
      if (newHeight > 200) notePanel.style.height = newHeight + 'px'
    } else if (isResizingTopLeft) {
      const dx = e.clientX - startX
      const dy = e.clientY - startY

      const newWidth = startWidth - dx
      const newHeight = startHeight - dy

      if (newWidth > 150) {
        notePanel.style.width = newWidth + 'px'
        notePanel.style.left = startLeft + dx + 'px'
      }
      if (newHeight > 200) {
        notePanel.style.height = newHeight + 'px'
        notePanel.style.top = startTop + dy + 'px'
      }
    }
  })

  document.addEventListener('pointerup', () => {
    isResizingCorner = false
    isResizingTopLeft = false
    document.body.style.userSelect = 'auto'
    positionNotes()
  })
}
/**
 * Show the notes box for an edge
 * @param {integer} edgeId
 */
function showEdgeData(edgeId) {
  const panel = elem('edgeNotePanel')
  edgeId = edgeId || network.getSelectedEdges()[0]
  const edge = data.edges.get(edgeId)
  elem('edgeLabel').innerHTML = edge.label?.trim().length
    ? edge.label
    : `Link from "${shorten(data.nodes.get(edge.from).label)}" to "${shorten(data.nodes.get(edge.to).label)}"`
  if (edge.created) {
    elem('edgeCreated').innerHTML = `${timeAndDate(edge.created.time)} by ${edge.created.user}`
    elem('edgeCreation').style.display = 'flex'
  } else elem('edgeCreation').style.display = 'none'
  if (edge.modified) {
    elem('edgeModified').innerHTML = `${timeAndDate(edge.modified.time)} by ${edge.modified.user}`
    elem('edgeModification').style.display = 'flex'
  } else elem('edgeModification').style.display = 'none'
  editor = new Quill('#edge-notes', {
    modules: {
      toolbar: viewOnly
        ? null
        : [
            'bold',
            'italic',
            'underline',
            'link',
            { list: 'ordered' },
            { list: 'bullet' },
            { indent: '-1' },
            { indent: '+1' },
          ],
    },
    placeholder: 'Notes',
    theme: 'snow',
    readOnly: viewOnly,
  })
  editor.id = edge.id
  window.editor = editor // used by popupEditor to access this editor
  if (edge.note) {
    if (edge.note instanceof Object) editor.setContents(edge.note)
    else editor.setText(edge.note)
  } else editor.setText('')
  editor.on('text-change', (delta, oldDelta, source) => {
    if (source === 'user') {
      data.edges.update({
        id: edgeId,
        note: isQuillEmpty(editor) ? '' : editor.getContents(),
        modified: timestamp(),
      })
      if (popupWindow) {
        popupEditor = popupWindow.popupEditor
        if (popupEditor) popupEditor.setContents(editor.getContents())
      }
    }
  })
  panel.classList.remove('hide')
  positionNotes()
}
/**
 * user has clicked on the sparkles icon in a Link node window
 * return the output from an LLM asked to elaborate on the causal
 * relationship between the two linked factors
 */
async function genAIEdge() {
  alertMsg('Processing...', 'info', true)
  editor.setText('Processing...\n')
  const sparklesElem = elem('sparklesEdge')
  sparklesElem.classList.add('rotating')
  const edgeId = network.getSelectedEdges()[0]
  const edge = data.edges.get(edgeId)
  const context = data.nodes
    .get()
    .map((n) => n.label.replaceAll('\n', ' '))
    .join(', ')
  const systemPrompt = `You are to explain a single causal link of the form A causes B.
Follow all the instructions below for your response:

Write a compact, clear explanation of why or how A causes B.
Do not introduce new factors that are not present in the inputs.
Do not restate or describe the whole system — focus only on the causal pathway from A to B, optionally referencing intermediary factors if necessary to provide a clear causal pathway.
Structure the output in the following order without using any section headings or a title:
1 Introduction
2 Bullet points explaining the causal reasoning
3 Evidence section containing academic citations and links to relevant web pages
4 Summary
Always begin the answer with: > ***This text has been generated by AI. It needs to be checked carefully.***
Do not begin by repeating the causal link that is being explained.
The introduction should be no more than 50 words.
The bullet points should clearly explain the causal reasoning in no more than 150 words.
The evidence section should provide academic references that support the causal link, with a brief explanation of how each reference supports the link.
All academic references must include full citations (author, date, title, source). Do not include DOIs.  You must verify that all academic references that you mention actually exist.
Always embed links for any referenced web pages.
The summary should concisely restate the main points in no more than 50 words.

Maximum total output length: 200 words.
Format everything in Markdown.`
  const aiResponse = await getAIresponse(
    `
Explain the causal link from ${data.nodes.get(edge.from).label} to ${data.nodes.get(edge.to).label}. 
Use these keywords as context: ${context}`,
    systemPrompt
  )
  editor.setContents(aiResponse)
  const modified = timestamp()
  data.edges.update({
    id: edgeId,
    note: isQuillEmpty(editor) ? '' : editor.getContents(),
    modified,
  })
  elem('edgeModified').innerHTML = `${timeAndDate(modified.time)} by ${modified.user}`
  positionNotes()
  sparklesElem.classList.remove('rotating')
  if (popupWindow) {
    popupEditor = popupWindow.popupEditor
    if (popupEditor) popupEditor.setContents(editor.getContents())
  }
  cancelAlertMsg()
}
/**
 * ensure that the panel is not outside the net pane, nor obscuring the Settings panel
 * @param {HTMLElement} notesPanel
 */
function positionNotes() {
  let notesPanel = document.getElementById('nodeNotePanel')
  if (notesPanel.classList.contains('hide')) notesPanel = document.getElementById('edgeNotePanel')
  if (notesPanel.classList.contains('hide')) return
  const netPane = document.getElementById('net-pane')
  const settings = document.getElementById('panel')

  let notesPanelRect = notesPanel.getBoundingClientRect()
  const settingsRect = settings.getBoundingClientRect()
  const netPaneRect = netPane.getBoundingClientRect()
  // if the notes would cover up the settings panel, move the notes to the left of the settings panel
  if (notesPanelRect.right > settingsRect.left && notesPanelRect.top < settingsRect.bottom) {
    notesPanel.style.left = `${settingsRect.left - notesPanelRect.width - 10}px`
    notesPanelRect = notesPanel.getBoundingClientRect()
  }
  // if the notes panel is taller than the net pane, increase its width and reduce its height
  if (notesPanelRect.height > netPaneRect.height - 20) {
    notesPanel.style.width = `$({
						(notesPanelRect.width * notesPanelRect.height) / (netPaneRect.height - 20)
					}px`
    notesPanel.style.height = `${netPaneRect.height - 20}px`
    notesPanel.style.left = `${notesPanelRect.right - notesPanelRect.width}px`
    notesPanel.style.top = 10
    notesPanelRect = notesPanel.getBoundingClientRect()
  }
  // if the notes panel is wider than the net pane, reduce its width
  if (notesPanelRect.width > netPaneRect.width) {
    notesPanel.style.width = `${netPaneRect.width - 20}px`
    notesPanel.style.left = 10
    notesPanelRect = notesPanel.getBoundingClientRect()
  }
  // if the notes panel is outside the boundary of the net pane, shift it into the pane
  if (notesPanelRect.left < netPaneRect.left + 10) {
    notesPanel.style.left = `${netPaneRect.left + 10}px`
  }
  if (notesPanelRect.right > netPaneRect.right - 10) {
    notesPanel.style.left = `${netPaneRect.right - notesPanelRect.width - 10}px`
  }
  const visibleBottom = Math.min(
    notesPanelRect.bottom,
    notesPanelRect.top + notesPanel.offsetHeight
  )
  if (visibleBottom > netPaneRect.bottom - 10) {
    notesPanel.style.top = `${netPaneRect.bottom - notesPanel.offsetHeight - 10}px`
  }
  if (notesPanelRect.top < netPaneRect.top + 10) {
    notesPanel.style.top = `${netPaneRect.top + 10}px`
  }
}
// Network tab

/**
 * Choose and apply a layout algorithm
 */
function autoLayout(e) {
  const option = e.target.value
  const selectElement = elem('layoutSelect')
  selectElement.value = option
  const label = selectElement.options[selectElement.selectedIndex].innerText
  network.storePositions() // record current positions so it can be undone
  if (network.physics.options.enabled) {
    // another layout already in progress - cancel it first
    network.off('stabilized')
    network.stopSimulation()
    network.setOptions({ physics: { enabled: false } })
    network.storePositions()
    alertMsg(`Previous layout cancelled`, 'warn')
  }
  doc.transact(() => {
    switch (option) {
      case 'off': {
        network.setOptions({ physics: { enabled: false } })
        break
      }
      case 'trophic': {
        try {
          trophic(data)
          trophicDistribute()
          data.nodes.update(data.nodes.get())
          elem('layoutSelect').value = 'off'
          statusMsg('Trophic layout applied')
        } catch (e) {
          alertMsg(`Trophic layout: ${e.message}`, 'error')
        }
        break
      }
      case 'fan': {
        {
          const nodes = data.nodes.get().filter((n) => !n.nodeHidden)
          nodes.forEach((n) => (n.level = undefined))
          const selectedNodes = getSelectedAndFixedNodes().map((nId) => data.nodes.get(nId))
          if (selectedNodes.length === 0) {
            alertMsg('At least one Factor needs to be selected', 'error')
            elem('layoutSelect').value = 'off'
            return
          }
          // if Up or Down stream are selected, use those for the direction
          let direction = 'from'
          if (getRadioVal('stream') === 'downstream') direction = 'to'
          else if (getRadioVal('stream') === 'upstream') direction = 'from'
          else {
            // if neither,
            //  and more links from the selected nodes are going upstream then downstream,
            //  put the selected nodes on the right, else on the left
            let nUp = 0
            let nDown = 0
            selectedNodes.forEach((sl) => {
              nUp += network
                .getConnectedNodes(sl.id, 'to')
                .filter((nId) => !data.nodes.get(nId).nodeHidden).length
              nDown += network
                .getConnectedNodes(sl.id, 'from')
                .filter((nId) => !data.nodes.get(nId).nodeHidden).length
            })
            direction = nUp > nDown ? 'to' : 'from'
          }
          const minX = Math.min(...nodes.map((n) => n.x))
          const maxX = Math.max(...nodes.map((n) => n.x))
          selectedNodes.forEach((n) => {
            setZLevel(n, direction)
          })
          nodes.forEach((n) => {
            if (n.level === undefined) n.level = 0
          })
          const maxLevel = Math.max(...nodes.map((n) => n.level))
          const gap = (maxX - minX) / maxLevel
          for (let l = 0; l <= maxLevel; l++) {
            let x = l * gap + minX
            if (direction === 'from') x = maxX - l * gap
            const nodesOnLevel = nodes.filter((n) => n.level === l)
            nodesOnLevel.forEach((n) => (n.x = x))
            const ySpaceNeeded = nodesOnLevel
              .map((n) => {
                const box = network.getBoundingBox(n.id)
                return box.bottom - box.top + 10
              })
              .reduce((a, b) => a + b, 0)
            const yGap = ySpaceNeeded / nodesOnLevel.length
            let newY = -ySpaceNeeded / 2
            nodesOnLevel
              .sort((a, b) => a.y - b.y)
              .forEach((n) => {
                n.y = newY
                newY += yGap
              })
          }
          data.nodes.update(nodes)
          elem('layoutSelect').value = 'off'
          statusMsg('Fan layout applied')
        }
        break
      }
      case 'barnesHut':
      case 'repulsion':
        {
          statusMsg('Working...')
          const options = { physics: { solver: option, stabilization: true } }
          options.physics[option] = {}
          options.physics[option].springLength = avEdgeLength()
          network.setOptions(options)
          // cancel the iterative algorithms as soon as they have stabilized
          network.on('stabilized', () => cancelLayout())
        }
        break
      case 'forceAtlas2Based': {
        statusMsg('Working...')
        const options = {
          physics: {
            solver: 'forceAtlas2Based',
            forceAtlas2Based: {
              theta: 2, // Boundary between consolidated long range forces and individual short range forces
              gravitationalConstant: -500, // Repulsion force (-ve values push nodes apart)
              centralGravity: 0.01, // Pulls nodes toward the center
              springConstant: 0.3, // Controls edge length
              springLength: 0, // Edge attraction force
              damping: 0.8, // Reduces oscillation
              avoidOverlap: 1, // Prevents node overlap
            },
          },
        }
        network.setOptions(options)
        // cancel the iterative algorithms as soon as they have stabilized
        network.on('stabilized', () => cancelLayout())
        network.on('stabilizationProgress', (obj) => {
          statusMsg(`Working... ${obj.iterations} iterations of ${obj.total}`)
        })
        break
      }
      default: {
        console.log('Unknown layout option')
        break
      }
    }
  })
  // if the layout doesn't stabilize, cancel it after 30 seconds
  setTimeout(() => {
    cancelLayout()
  }, 30000)
  logHistory(`applied ${label} layout`)

  /**
   * cancel the iterative layout algorithms
   */
  function cancelLayout() {
    network.setOptions({ physics: { enabled: false } })
    network.storePositions()
    elem('layoutSelect').value = 'off'
    statusMsg(`${label} layout applied`)
    data.nodes.update(data.nodes.get())
  }

  /**
   * set the levels for fan, using a breadth first search
   * @param {object} node root node
   * @param {string} direction either 'from' or 'to', depending on whether the links to use are point from or to the node
   */
  function setZLevel(node, direction) {
    let q = [node]
    let level = 0
    node.level = 0
    while (q.length > 0) {
      const currentNode = q.shift()
      const connectedNodes = data.nodes
        .get(network.getConnectedNodes(currentNode.id, direction))
        .filter((n) => !n.nodeHidden && n.level === undefined)
      if (connectedNodes.length > 0) {
        level = currentNode.level + 1
        connectedNodes.forEach((n) => {
          n.level = level
        })
        q = q.concat(connectedNodes)
      }
    }
  }
  /**
   * find the average length of all edges, as a guide to the layout spring length
   * so that map is roughly as spaced out as before layout
   * @returns average length (in canvas units)
   */
  function avEdgeLength() {
    let edgeSum = 0
    data.edges.forEach((e) => {
      const from = data.nodes.get(e.from)
      const to = data.nodes.get(e.to)
      edgeSum += Math.sqrt((from.x - to.x) ** 2 + (from.y - to.y) ** 2)
    })
    return edgeSum / data.edges.length
  }
  /**
   * At each level for a trophic layout, distribute the Factors equally along the vertical axis,
   * avoiding overlaps
   */
  function trophicDistribute() {
    for (let level = 0; level <= NLEVELS; level++) {
      const nodesOnLevel = data.nodes.get().filter((n) => n.level === level)
      const ySpaceNeeded = nodesOnLevel
        .map((n) => {
          const box = network.getBoundingBox(n.id)
          return box.bottom - box.top + 10
        })
        .reduce((a, b) => a + b, 0)
      const gap = ySpaceNeeded / nodesOnLevel.length
      let newY = -ySpaceNeeded / 2
      nodesOnLevel
        .sort((a, b) => a.y - b.y)
        .forEach((n) => {
          n.y = newY
          newY += gap
        })
    }
  }
}

function snapToGridSwitch(e) {
  snapToGridToggle = e.target.checked
  doSnapToGrid(snapToGridToggle)
  yNetMap.set('snapToGrid', snapToGridToggle)
}

export function doSnapToGrid(toggle) {
  elem('snaptogridswitch').checked = toggle
  if (toggle) {
    data.nodes.update(
      data.nodes.get().map((n) => {
        snapToGrid(n)
        return n
      })
    )
  }
}

function selectCurve(e) {
  const option = e.target.value
  setCurve(option)
  yNetMap.set('curve', option)
}

export function setCurve(option) {
  elem('curveSelect').value = option
  network.setOptions({
    edges: {
      smooth: option === 'Curved' ? { type: 'cubicBezier' } : false,
    },
  })
}

function updateNetBack(color) {
  const ul = elem('underlay')
  ul.style.backgroundColor = color
  // if in drawing mode, make the underlay translucent so that network shows through
  if (elem('toolbox').style.display === 'block') makeTranslucent(ul)
  yNetMap.set('background', color)
}

const backgroundOpacity = 0.6

function makeTranslucent(el) {
  el.style.backgroundColor = getComputedStyle(el)
    .backgroundColor.replace(')', `, ${backgroundOpacity})`)
    .replace('rgb', 'rgba')
}

function makeSolid(el) {
  el.style.backgroundColor = getComputedStyle(el)
    .backgroundColor.replace(`, ${backgroundOpacity})`, ')')
    .replace('rgba', 'rgb')
}
export function setBackground(color) {
  elem('underlay').style.backgroundColor = color
  if (elem('toolbox').style.display === 'block') makeTranslucent(elem('underlay'))
  elem('netBackColorWell').style.backgroundColor = color
}

function toggleDrawingLayer() {
  drawingSwitch = elem('toolbox').style.display === 'block'
  const ul = elem('underlay')
  if (drawingSwitch) {
    // close drawing layer
    deselectTool()
    elem('toolbox').style.display = 'none'
    elem('underlay').style.zIndex = 0
    makeSolid(ul)
    document.querySelector('.upper-canvas').style.zIndex = 0
    inAddMode = false
    elem('buttons').style.visibility = 'visible'
    setButtonDisabledStatus('addNode', false)
    setButtonDisabledStatus('addLink', false)
    undoRedoButtonStatus()
    if (elem('showLegendSwitch').checked) legend()
    if (nChanges) logHistory('drew on the background layer')
    changeCursor('default')
  } else {
    // expose drawing layer
    elem('toolbox').style.display = 'block'
    ul.style.zIndex = 1000
    ul.style.cursor = 'default'
    document.querySelector('.upper-canvas').style.zIndex = 1001
    // make the underlay (which is now overlay) translucent
    makeTranslucent(ul)
    clearLegend()
    inAddMode = 'disabled'
    elem('buttons').style.visibility = 'hidden'
    elem('help-button').style.visibility = 'visible'
    setButtonDisabledStatus('addNode', true)
    setButtonDisabledStatus('addLink', true)
    setButtonDisabledStatus('undo', true)
    setButtonDisabledStatus('redo', true)
  }
  drawingSwitch = !drawingSwitch
  network.redraw()
}
function ensureNotDrawing() {
  if (!drawingSwitch) return
  toggleDrawingLayer()
  elem('drawing').checked = false
}

function selectAllFactors() {
  selectFactors(data.nodes.getIds({ filter: (n) => !n.nodeHidden }))
  showSelected()
}

export function selectFactors(nodeIds) {
  network.selectNodes(nodeIds, false)
  showSelected()
}

function selectAllLinks() {
  selectLinks(data.edges.getIds({ filter: (e) => !e.edgeHidden }))
  showSelected()
}

export function selectLinks(edgeIds) {
  network.selectEdges(edgeIds)
  showSelected()
}

/**
 * Selects all the nodes and edges that have been created or modified by a user
 */
function selectUsersItems(event) {
  event.preventDefault()
  const userName = event.target.dataset.userName
  const usersNodes = data.nodes
    .get()
    .filter((n) => n.created?.user === userName || n.modified?.user === userName)
    .map((n) => n.id)
  const userEdges = data.edges
    .get()
    .filter((e) => e.created?.user === userName || e.modified?.user === userName)
    .map((e) => e.id)
  network.setSelection({ nodes: usersNodes, edges: userEdges })
  showSelected()
}

function legendSwitch(e) {
  const on = e.target.checked
  setLegend(on, true)
  yNetMap.set('legend', on)
}
export function setLegend(on, warn) {
  elem('showLegendSwitch').checked = on
  if (on) legend(warn)
  else clearLegend()
}
function votingSwitch(e) {
  const on = e.target.checked
  setVoting(on)
  yNetMap.set('voting', on)
}
function setVoting(on) {
  elem('showVotingSwitch').checked = on
  showVotingToggle = on
  network.redraw()
}
/************************************************************** Analysis tab ************************************************* */

/**
 *
 * @param {String} name of Radio button group
 * @returns value of the button that is checked
 */
function getRadioVal(name) {
  // get list of radio buttons with specified name
  const radios = document.getElementsByName(name)
  // loop through list of radio buttons
  for (let i = 0, len = radios.length; i < len; i++) {
    if (radios[i].checked) return radios[i].value
  }
}
/**
 *
 * @param {String} name of the radio button group
 * @param {*} value check the button with this value
 */
function setRadioVal(name, value) {
  // get list of radio buttons with specified name
  const radios = document.getElementsByName(name)
  // loop through list of radio buttons and set the check on the one with the value
  for (let i = 0, len = radios.length; i < len; i++) {
    radios[i].checked = radios[i].value === value
  }
}
/**
 * Return an array of the node Ids of Factors that are selected or are locked
 * @returns Array
 */
function getSelectedAndFixedNodes() {
  return [
    ...new Set(
      network.getSelectedNodes().concat(
        data.nodes
          .get()
          .filter((n) => n.fixed)
          .map((n) => n.id)
      )
    ),
  ]
}

/**
 * Sets the Analysis radio buttons and Factor selection according to values in global hiddenNodes
 *  (which is set when yNetMap is loaded, or when a file is read in)
 */
function setAnalysisButtonsFromRemote() {
  if (netLoaded) {
    const selectedNodes = [].concat(hiddenNodes.selected) // ensure that hiddenNodes.selected is an array
    if (
      hiddenNodes.radiusSetting !== 'All' ||
      hiddenNodes.streamSetting !== 'All' ||
      hiddenNodes.pathsSetting !== 'All'
    ) {
      network.selectNodes(selectedNodes, false) // in viewing  only mode, this does nothing
      if (selectedNodes.length > 0) {
        if (!viewOnly) statusMsg(`${listFactors(getSelectedAndFixedNodes())} selected`)
      } else clearStatusBar()
    }
    showNodeOrEdgeData()
    if (hiddenNodes.radiusSetting) setRadioVal('radius', hiddenNodes.radiusSetting)
    if (hiddenNodes.streamSetting) setRadioVal('stream', hiddenNodes.streamSetting)
    if (hiddenNodes.pathsSetting) setRadioVal('paths', hiddenNodes.pathsSetting)
  }
}

function setYMapAnalysisButtons() {
  const selectedNodes = getSelectedAndFixedNodes()
  yNetMap.set('radius', {
    radiusSetting: getRadioVal('radius'),
    selected: selectedNodes,
  })
  yNetMap.set('stream', {
    streamSetting: getRadioVal('stream'),
    selected: selectedNodes,
  })
  yNetMap.set('paths', {
    pathsSetting: getRadioVal('paths'),
    selected: selectedNodes,
  })
}
/**
 * Hide factors and links to show only those closest to the selected factors and/or
 * those up/downstream and/or those on paths between the selected factors
 */
function analyse() {
  const selectedNodes = getSelectedAndFixedNodes()
  setYMapAnalysisButtons()
  // get all nodes and edges and unhide them before hiding those not wanted to be visible
  const nodes = data.nodes
    .get()
    .filter((n) => !n.isCluster)
    .map((n) => {
      setNodeHidden(n, false)
      return n
    })
  const edges = data.edges
    .get()
    .filter((e) => !e.isClusterEdge)
    .map((e) => {
      setEdgeHidden(e, false)
      return e
    })
  cancelHiddenStyles()
  // if showing everything, we are done
  if (
    getRadioVal('radius') === 'All' &&
    getRadioVal('stream') === 'All' &&
    getRadioVal('paths') === 'All'
  ) {
    resetAll()
    showSelected()
    showNodeOrEdgeData()
    return
  }
  // check that at least one factor is selected
  if (selectedNodes.length === 0 && getRadioVal('paths') === 'All') {
    alertMsg('A Factor needs to be selected', 'error')
    resetAll()
    return
  }
  // but paths between factors needs at least two
  if (getRadioVal('paths') !== 'All' && selectedNodes.length < 2) {
    alertMsg('Select at least 2 factors to show paths between them', 'warn')
    resetAll()
    return
  }
  hideNotes()
  // these operations are not commutative (at least for networks with loops), so do them all in order
  if (getRadioVal('radius') !== 'All') {
    hideNodesByRadius(selectedNodes, parseInt(getRadioVal('radius')))
  }
  if (getRadioVal('stream') !== 'All') hideNodesByStream(selectedNodes, getRadioVal('stream'))
  if (getRadioVal('paths') !== 'All') hideNodesByPaths(selectedNodes, getRadioVal('paths'))

  // finally display the map with its hidden factors and edges
  data.nodes.update(nodes)
  data.edges.update(edges)

  // announce what has been done
  let streamMsg = ''
  if (getRadioVal('stream') === 'upstream') streamMsg = 'upstream'
  if (getRadioVal('stream') === 'downstream') streamMsg = 'downstream'
  let radiusMsg = ''
  if (getRadioVal('radius') === '1') radiusMsg = 'within one link'
  if (getRadioVal('radius') === '2') radiusMsg = 'within two links'
  if (getRadioVal('radius') === '3') radiusMsg = 'within three links'
  let pathsMsg = ''
  if (getRadioVal('paths') === 'allPaths') pathsMsg = ': showing all paths'
  if (getRadioVal('paths') === 'shortestPath') pathsMsg = ': showing shortest paths'
  if (getRadioVal('stream') === 'All' && getRadioVal('radius') === 'All') {
    statusMsg(
      `Showing  ${getRadioVal('paths') === 'allPaths' ? 'all paths' : 'shortest paths'} between ${listFactors(
        getSelectedAndFixedNodes(),
        true
      )}`
    )
  } else {
    statusMsg(
      `Factors ${streamMsg} ${streamMsg && radiusMsg ? ' and ' : ''} ${radiusMsg} of ${listFactors(
        getSelectedAndFixedNodes(),
        true
      )}${pathsMsg}`
    )
  }
  /**
   * return all to neutral analysis state
   */
  function resetAll() {
    setRadioVal('radius', 'All')
    setRadioVal('stream', 'All')
    setRadioVal('paths', 'All')
    setYMapAnalysisButtons()
    data.nodes.update(nodes)
    data.edges.update(edges)
  }
  /**
   * Hide factors that are more than radius links distant from those selected
   * @param {string[]} selectedNodes
   * @param {Integer} radius
   */
  function hideNodesByRadius(selectedNodes, radius) {
    const nodeIdsInRadiusSet = new Set()
    const linkIdsInRadiusSet = new Set()

    // put those factors and links within radius links into these sets
    if (getRadioVal('stream') === 'upstream' || getRadioVal('stream') === 'All') {
      inSet(selectedNodes, radius, 'to')
    }
    if (getRadioVal('stream') === 'downstream' || getRadioVal('stream') === 'All') {
      inSet(selectedNodes, radius, 'from')
    }

    // hide all nodes and edges not in radius
    nodes.forEach((n) => {
      if (!nodeIdsInRadiusSet.has(n.id)) setNodeHidden(n, true)
    })
    edges.forEach((e) => {
      if (!linkIdsInRadiusSet.has(e.id)) setEdgeHidden(e, true)
    })
    // add links between factors that are in radius set, to give an ego network
    nodeIdsInRadiusSet.forEach((f) => {
      network.getConnectedEdges(f).forEach((e) => {
        const edge = data.edges.get(e)
        if (nodeIdsInRadiusSet.has(edge.from) && nodeIdsInRadiusSet.has(edge.to)) {
          setEdgeHidden(edge, false)
        }
      })
    })

    /**
     * recursive function to collect Factors and Links within radius links from any of the nodes listed in nodeIds
     * Factor ids are collected in nodeIdsInRadiusSet and links in linkIdsInRadiusSet
     * Links are followed in a consistent direction, i.e. if 'to', only links directed away from the the nodes are followed
     * @param {string[]} nodeIds
     * @param {number} radius
     * @param {string} direction - either 'from' or 'to'
     */
    function inSet(nodeIds, radius, direction) {
      if (radius < 0) return
      nodeIds.forEach((nId) => {
        const linked = []
        nodeIdsInRadiusSet.add(nId)
        const links = network
          .getConnectedEdges(nId)
          .filter((e) => data.edges.get(e)[direction] === nId)
        if (links && radius > 0) {
          links.forEach((lId) => {
            linkIdsInRadiusSet.add(lId)
            linked.push(data.edges.get(lId)[direction === 'to' ? 'from' : 'to'])
          })
        }
        if (linked) inSet(linked, radius - 1, direction)
      })
    }
  }
  /**
   * Hide factors that are not up or downstream from the selected factors.
   * Does not include links or factors that are already hidden
   * @param {string[]} selectedNodes
   * @param {string} direction - 'upstream' or 'downstream'
   */
  function hideNodesByStream(selectedNodes, upOrDown) {
    const nodeIdsInStreamSet = new Set()
    const linkIdsInStreamSet = new Set()

    const radiusVal = getRadioVal('radius')
    let radius = Infinity
    if (radiusVal !== 'All') {
      radius = parseInt(radiusVal)
    }
    let direction = 'to'
    if (upOrDown === 'upstream') direction = 'from'

    // breadth first search for all Factors that are downstream and less than or equal to radius links away
    data.nodes.map((n) => (n.level = undefined))
    selectedNodes.forEach((nodeId) => {
      nodeIdsInStreamSet.add(nodeId)
      const node = data.nodes.get(nodeId)
      let q = [node]
      let level = 0
      node.level = 0
      while (q.length > 0 && level <= radius) {
        const currentNode = q.shift()
        const connectedNodes = data.nodes
          .get(network.getConnectedNodes(currentNode.id, direction))
          .filter((n) => !(n.nodeHidden || nodeIdsInStreamSet.has(n.id)))
        if (connectedNodes.length > 0) {
          level = currentNode.level + 1
          connectedNodes.forEach((n) => {
            nodeIdsInStreamSet.add(n.id)
            n.level = level
          })
          q = q.concat(connectedNodes)
        }
      }
    })

    // hide all nodes and edges not up or down stream
    nodes.forEach((n) => {
      if (!nodeIdsInStreamSet.has(n.id)) setNodeHidden(n, true)
    })
    edges.forEach((e) => {
      if (!linkIdsInStreamSet.has(e.id)) setEdgeHidden(e, true)
    })

    // add links between factors that are in radius set, to give an ego network
    nodeIdsInStreamSet.forEach((f) => {
      network.getConnectedEdges(f).forEach((e) => {
        const edge = data.edges.get(e)
        if (nodeIdsInStreamSet.has(edge.from) && nodeIdsInStreamSet.has(edge.to)) {
          setEdgeHidden(edge, false)
        }
      })
    })
  }

  /**
   * Hide all factors and links that are not on the shortest path (or all paths) between the selected factors
   * Avoids factors or links that are hidden
   * @param {string[]} selectedNodes
   * @param {string} pathType - either 'allPaths' or 'shortestPath'
   */
  function hideNodesByPaths(selectedNodes, pathType) {
    // paths is an array of objects with from and to node ids, or an empty array if there is no path
    const paths = shortestPaths(selectedNodes, pathType === 'allPaths')
    if (paths.length === 0) {
      alertMsg('No path between the selected Factors', 'info')
      setRadioVal('paths', 'All')
      setYMapAnalysisButtons()
      return
    }
    // hide nodes and links that are not included in paths
    const nodeIdsInPathsSet = new Set()
    const linkIdsInPathsSet = new Set()

    paths.forEach((links) => {
      links.forEach((link) => {
        const edge = data.edges.get({
          filter: (e) => e.to === link.to && e.from === link.from,
        })[0]
        linkIdsInPathsSet.add(edge.id)
        nodeIdsInPathsSet.add(edge.from)
        nodeIdsInPathsSet.add(edge.to)
      })
    })
    // hide all factors and links that are not in the set of paths
    nodes.forEach((n) => {
      if (!nodeIdsInPathsSet.has(n.id)) setNodeHidden(n, true)
    })
    edges.forEach((e) => {
      if (!linkIdsInPathsSet.has(e.id)) setEdgeHidden(e, true)
    })

    /**
     * Given two or more selected factors, return a list of all the links that are either on any path between them, or just the ones on the shortest paths between them
     * @param {Boolean} all when true, find all the links that connect to the selected factors; when false, find the shortest paths between the selected factors
     * @returns	Arrays of objects with from: and to: properties for all the links (an empty array if there is no path between any of the selected factors)
     */
    function shortestPaths(selectedNodes, all) {
      const visited = new Map()
      const allPaths = []
      // list of all pairs of the selected factors
      const combos = selectedNodes.flatMap((v, i) => selectedNodes.slice(i + 1).map((w) => [v, w]))
      // for each pair, find the sequences of links in both directions and combine them
      combos.forEach((combo) => {
        const source = combo[0]
        const dest = combo[1]
        let links = pathList(source, dest, all)
        if (links.length > 0) allPaths.push(links)
        links = pathList(dest, source, all)
        if (links.length > 0) allPaths.push(links)
      })
      return allPaths

      /**
       * find the paths (as a list of links) that connect the source and destination
       * @param {String} source
       * @param {String} dest
       * @param {Boolean} all true of all paths between Source and Destination are wanted; false if just the shortest path
       * @returns an array of lists of links that connect the paths
       */
      function pathList(source, dest, all) {
        visited.clear()
        const links = []
        let paths = getPaths(source, dest)
        // if no path found, getPaths return an array of length greater than the total number of factors in the map, or a string
        // in this case, return an empty list
        if (!Array.isArray(paths) || paths.length === data.nodes.length + 1) paths = []
        if (!all) {
          for (let i = 0; i < paths.length - 1; i++) {
            links.push({ from: paths[i], to: paths[i + 1] })
          }
        }
        return links
        /**
         * recursively explore the map starting from source until destination is reached.
         * stop if a factor has already been visited, or at a dead end (zero out-degree)
         * @param {String} source
         * @param {String} dest
         * @returns an array of factors, the path so far followed
         */
        function getPaths(source, dest) {
          if (source === dest) return [dest]
          visited.set(source, true)
          let path = [source]
          // only consider nodes and edges that are not hidden
          const connectedNodes = network
            .getConnectedEdges(source)
            .filter((e) => {
              const edge = data.edges.get(e)
              return !edge.edgeHidden && edge.from === source
            })
            .map((e) => data.edges.get(e).to)
          if (connectedNodes.length === 0) return 'deadend'
          if (all) {
            // all paths between the source and destination
            connectedNodes.forEach((next) => {
              const vis = visited.get(next)
              if (vis === 'onpath') {
                links.push({ from: source, to: next })
                path = path.concat([next])
              } else if (!vis) {
                const p = getPaths(next, dest)
                if (Array.isArray(p) && p.length > 0) {
                  links.push({ from: source, to: next })
                  visited.set(next, 'onpath')
                  path = path.concat(p)
                }
              }
            })
          } else {
            // shortest path between the source and destination
            let bestPath = []
            let bestPathLength = data.nodes.length
            connectedNodes.forEach((next) => {
              let p = visited.get(next)
              if (!p) {
                p = getPaths(next, dest)
                visited.set(next, p)
              }
              if (Array.isArray(p) && p.length > 0) {
                if (p.length < bestPathLength) {
                  bestPath = p
                  bestPathLength = p.length
                }
              }
            })
            path = path.concat(bestPath)
          }
          // if no progress has been made (the path is just the initial source factor), return an empty path
          if (path.length === 1) path = []
          return path
        }
      }
    }
  }
}
/**
 * Unset the indicators on the Settings Factor and Link tabs that show that Factors/Links with
 * these styles are hidden
 * Assumes that the factors and links have already been unhidden - this just  removes the UI indicators
 */
function cancelHiddenStyles() {
  Array.from(document.getElementsByClassName('sampleNode'))
    .filter((n) => n.dataset.hide === 'hidden')
    .forEach((n) => {
      n.dataset.hide = 'visible'
      n.style.opacity = 1.0
    })
  Array.from(document.getElementsByClassName('sampleLink'))
    .filter((e) => e.dataset.hide === 'hidden')
    .forEach((e) => {
      e.dataset.hide = 'visible'
      e.style.opacity = 1.0
    })
}

function sizingSwitch(e) {
  const metric = e.target.value
  sizing(metric)
  yNetMap.set('sizing', metric)
}

/**
 * set the size of the nodes proportional to the selected metric
 * @param {String} metric none, all the same size, in degree, out degree or betweenness centrality
 */
// constants for sizes of nodes
const MIN_WIDTH = 50
const EQUAL_WIDTH = 100
const MAX_WIDTH = 200

export function sizing(metric) {
  const nodesToUpdate = []
  let min = Number.MAX_VALUE
  let max = 0
  data.nodes.forEach((node) => {
    const oldValue = node.val
    switch (metric) {
      case 'Off':
      case 'Equal': {
        node.val = 0
        break
      }
      case 'Inputs': {
        node.val = network.getConnectedNodes(node.id, 'from').length
        break
      }
      case 'Outputs': {
        node.val = network.getConnectedNodes(node.id, 'to').length
        break
      }
      case 'Leverage': {
        const inDegree = network.getConnectedNodes(node.id, 'from').length
        const outDegree = network.getConnectedNodes(node.id, 'to').length
        node.val = inDegree === 0 ? 0 : outDegree / inDegree
        break
      }
      case 'Centrality': {
        node.val = node.bc
        break
      }
    }
    if (node.val < min) min = node.val
    if (node.val > max) max = node.val
    if (metric === 'Off' || metric === 'Equal' || node.val !== oldValue) nodesToUpdate.push(node)
  })
  data.nodes.forEach((node) => {
    switch (metric) {
      case 'Off': {
        node.widthConstraint = node.heightConstraint = false
        node.size = 25
        break
      }
      case 'Equal': {
        node.widthConstraint = node.heightConstraint = node.size = EQUAL_WIDTH
        break
      }
      default:
        node.widthConstraint =
          node.heightConstraint =
          node.size =
            MIN_WIDTH + MAX_WIDTH * scale(min, max, node.val)
    }
  })
  data.nodes.update(nodesToUpdate)
  elem('sizing').value = metric

  function scale(min, max, value) {
    if (max === min) {
      return 0.5
    } else {
      return Math.max(0, (value - min) * (1 / (max - min)))
    }
  }
}

// Note: most of the clustering functionality is in cluster.js
/**
 * User has chosen a clustering option
 * @param {Event} e
 */
function selectClustering(e) {
  const option = e.target.value
  // it doesn't make much sense to cluster while the factors are hidden, so undo that
  setRadioVal('radius', 'All')
  setRadioVal('stream', 'All')
  setRadioVal('paths', 'All')
  setYMapAnalysisButtons()
  doc.transact(() => {
    data.nodes.update(
      data.nodes.get().map((n) => {
        setNodeHidden(n, false)
        return n
      })
    )
    data.edges.update(
      data.edges.get().map((e) => {
        setEdgeHidden(e, false)
        return e
      })
    )
  })
  cluster(option)
  fit()
  yNetMap.set('cluster', option)
}
export function setCluster(option) {
  elem('clustering').value = option
}
/**
 * recreate the Clustering drop down menu to include user attributes as clustering options
 * @param {object} obj {menu value, menu text}
 */
export function recreateClusteringMenu(obj) {
  // remove any old select items, other than the standard ones (which are the first 4: None, Style, Color, Community)
  const select = elem('clustering')
  for (let i = 4, len = select.options.length; i < len; i++) {
    select.remove()
  }
  // append the ones provided
  for (const property in obj) {
    if (obj[property] !== '*deleted*') {
      const opt = document.createElement('option')
      opt.value = property
      opt.text = shorten(obj[property], 12)
      select.add(opt, null)
    }
  }
}

/* ---------------------------------------history window --------------------------------*/
/**
 * display the history log in a window
 */
function showHistory() {
  elem('history-window').style.display = 'block'
  const log = elem('history-log')
  log.innerHTML = yHistory
    .toArray()
    .map(
      (rec) => `<div class="history-time">${timeAndDate(rec.time)}: </div>
		<div class="history-action">${rec.user} ${rec.action}</div>
		<div class="history-rollback" data-time="${rec.time}"></div>`
    )
    .join(' ')
  document.querySelectorAll('div.history-rollback').forEach((e) => addRollbackIcon(e))
  if (log.children.length > 0) {
    // without the timeout, the window does not scroll fully to the bottom
    setTimeout(() => log.lastChild.scrollIntoView(false), 20)
  }
}
/**
 * add a button for rolling back if there is state data corresponding to this log record
 * @param {HTMLElement} e - history record
 * */
async function addRollbackIcon(e) {
  await localForage.getItem(timekey(parseInt(e.dataset.time))).then((state) => {
    if (state) {
      e.id = `hist${e.dataset.time}`
      e.innerHTML = `<div class="tooltip">
				<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-bootstrap-reboot" viewBox="0 0 16 16">
				<path d="M1.161 8a6.84 6.84 0 1 0 6.842-6.84.58.58 0 1 1 0-1.16 8 8 0 1 1-6.556 3.412l-.663-.577a.58.58 0 0 1 .227-.997l2.52-.69a.58.58 0 0 1 .728.633l-.332 2.592a.58.58 0 0 1-.956.364l-.643-.56A6.812 6.812 0 0 0 1.16 8z"/>
				<path d="M6.641 11.671V8.843h1.57l1.498 2.828h1.314L9.377 8.665c.897-.3 1.427-1.106 1.427-2.1 0-1.37-.943-2.246-2.456-2.246H5.5v7.352h1.141zm0-3.75V5.277h1.57c.881 0 1.416.499 1.416 1.32 0 .84-.504 1.324-1.386 1.324h-1.6z"/>
				</svg>
				<span class="tooltiptext rollbacktip">Rollback to before this action</span>
			</div>
		</div>`
      if (elem(e.id)) listen(e.id, 'click', rollback)
    }
  })
}
/**
 * Restores the state of the map to a previous one
 * @param {Event} event
 * @returns null if no rollback possible or cancelled
 */
function rollback(event) {
  const rbTime = parseInt(event.currentTarget.dataset.time)
  localForage.getItem(timekey(rbTime)).then((rb) => {
    if (!rb) return
    if (!confirm(`Roll back the map to what it was before ${timeAndDate(rbTime)}?`)) return
    const state = JSON.parse(decompressFromUTF16(rb))
    data.nodes.clear()
    data.edges.clear()
    data.nodes.update(state.nodes)
    data.edges.update(state.edges)
    doc.transact(() => {
      for (const k in state.net) {
        yNetMap.set(k, state.net[k])
      }
      setMapTitle(state.net.mapTitle)
      for (const k in state.samples) {
        ySamplesMap.set(k, state.samples[k])
      }
      if (state.paint) {
        yPointsArray.delete(0, yPointsArray.length)
        yPointsArray.insert(0, state.paint)
      }
      if (state.drawing) {
        yDrawingMap.clear()
        for (const k in state.drawing) {
          yDrawingMap.set(k, state.drawing[k])
        }
        updateFromDrawingMap()
      }
    })
    localForage.removeItem(timekey(rbTime))
    logHistory(
      `rolled back the map to what it was before ${timeAndDate(rbTime, true)}`,
      null,
      'rollback'
    )
  })
}

function showHistorySwitch() {
  if (elem('showHistorySwitch').checked) showHistory()
  else elem('history-window').style.display = 'none'
}
listen('history-close', 'click', historyClose)
function historyClose() {
  elem('history-window').style.display = 'none'
  elem('showHistorySwitch').checked = false
}

dragElement(elem('history-window'), elem('history-header'))

/* --------------------------------------- avatars and shared cursors--------------------------------*/

let oldViewOnly = viewOnly // save the viewOnly state
/* tell user if they are offline and disconnect websocket server */
window.addEventListener('offline', () => {
  alertMsg('No network connection - working offline (view only)', 'info')
  wsProvider.shouldConnect = false
  network.setOptions({ interaction: { dragNodes: false, hover: false } })
  hideNavButtons()
  sideDrawEditor.enable(false)
  oldViewOnly = viewOnly
  viewOnly = true
})
window.addEventListener('online', () => {
  wsProvider.connect()
  alertMsg('Network connection re-established', 'info')
  viewOnly = oldViewOnly
  if (!viewOnly) showNavButtons()
  sideDrawEditor.enable(true)
  network.setOptions({ interaction: { dragNodes: true, hover: true } })
  showAvatars()
})
/**
 *  set up user monitoring (awareness)
 */
function setUpAwareness() {
  showAvatars()
  yAwareness.on('change', (event) => receiveEvent(event))

  // regularly broadcast our own state, every 20 seconds
  setInterval(() => {
    yAwareness.setLocalStateField('pkt', { time: Date.now() })
  }, 20000)

  // if debug = fake, generate fake mouse events every 200 ms for testing
  if (/fake/.test(debug)) {
    setInterval(() => {
      yAwareness.setLocalStateField('cursor', {
        x: Math.random() * 1000 - 500,
        y: Math.random() * 1000 - 500,
      })
    }, 200)
  }

  // fade out avatar when there has been no movement of the mouse for 15 minutes
  asleep(false)
  let sleepTimer = setTimeout(() => asleep(true), TIMETOSLEEP)

  // throttle mousemove broadcast to avoid overloading server
  let throttled = false
  const THROTTLETIME = 200
  window.addEventListener('mousemove', (e) => {
    // broadcast my mouse movements
    if (throttled) return
    throttled = true
    setTimeout(() => (throttled = false), THROTTLETIME)
    clearTimeout(sleepTimer)
    asleep(false)
    sleepTimer = setTimeout(() => asleep(true), TIMETOSLEEP)
    // broadcast current position of mouse in canvas coordinates
    const box = netPane.getBoundingClientRect()
    yAwareness.setLocalStateField(
      'cursor',
      network.DOMtoCanvas({
        x: Math.round(e.clientX - box.left),
        y: Math.round(e.clientY - box.top),
      })
    )
  })
}

/**
 * Set the awareness local state to show whether this client is sleeping (no mouse movement for 15 minutes)
 * @param {Boolean} isSleeping
 */
function asleep(isSleeping) {
  if (myNameRec.asleep === isSleeping) return
  myNameRec.asleep = isSleeping
  yAwareness.setLocalState({ user: myNameRec })
  showAvatars()
  //disconnect from websocket server to save resources when sleeping
  if (isSleeping) wsProvider.disconnect()
  else wsProvider.connect()
}
/**
 * display the awareness events
 * @param {object} event
 */
function traceUsers(event) {
  let msg = ''
  event.added.forEach((id) => {
    msg += `Added ${user(id)} (${id}) `
  })
  event.updated.forEach((id) => {
    msg += `Updated ${user(id)} (${id}) `
  })
  event.removed.forEach((id) => {
    msg += `Removed (${id}) `
  })
  console.log('yAwareness', exactTime(), msg)

  function user(id) {
    const userRec = yAwareness.getStates().get(id)
    return isEmpty(userRec.user) ? id : userRec.user.name
  }
}
const lastMicePositions = new Map()
const lastAvatarStatus = new Map()
let refreshAvatars = true
/**
 * Despatch to deal with event
 * @param {object} event - from yAwareness.on('change')
 */
function receiveEvent(event) {
  if (/aware/.test(debug)) traceUsers(event)
  if (elem('showUsersSwitch').checked) {
    const box = netPane.getBoundingClientRect()
    const changed = event.added.concat(event.updated)
    changed.forEach((userId) => {
      const rec = yAwareness.getStates().get(userId)
      if (
        userId !== clientID &&
        rec.cursor &&
        !objectEquals(rec.cursor, lastMicePositions.get(userId))
      ) {
        showOtherMouse(userId, rec.cursor, box)
        lastMicePositions.set(userId, rec.cursor)
      }
      if (rec.user) {
        // if anything has changed, redisplay the avatars
        if (refreshAvatars || !objectEquals(rec.user, lastAvatarStatus.get(userId))) showAvatars()
        lastAvatarStatus.set(userId, rec.user)
        // set a timer for this avatar to self-destruct if no update has been received for a minute
        const ava = elem(`ava${userId}`)
        if (ava) {
          clearTimeout(ava.timer)
          ava.timer = setTimeout(removeAvatar, 60000, ava)
        }
      }
      if (userId !== clientID && rec.addingFactor) showGhostFactor(userId, rec.addingFactor)
    })
  }
  if (followme) followUser()
}
/**
 * Display another user's mouse pointers (if they are inside the canvas)
 */
function showOtherMouse(userId, cursor, box) {
  const cursorDiv = elem(userId.toString())
  if (cursorDiv) {
    const p = network.canvasToDOM(cursor)
    p.x += box.left
    p.y += box.top
    cursorDiv.style.top = `${p.y}px`
    cursorDiv.style.left = `${p.x}px`
    cursorDiv.style.display =
      p.x < box.left || p.x > box.right || p.y > box.bottom || p.y < box.top ? 'none' : 'block'
  }
}
/**
 * Place a circle at the top left of the net pane to represent each user who is online
 * Also create a cursor (a div) for each of the users
 */

function showAvatars() {
  refreshAvatars = false
  const recs = Array.from(yAwareness.getStates())
  // remove and save myself (using clientID as the id, not name)
  const me = recs.splice(
    recs.findIndex((a) => a[0] === clientID),
    1
  )
  const nameRecs = recs

    .map(([, value]) => value.user || null)
    .filter((e) => e) // remove any recs without a user record
    .filter((v, i, a) => a.findIndex((t) => t.name === v.name) === i) // remove duplicates, by name
    .sort((a, b) => (a.name.charAt(0).toUpperCase() > b.name.charAt(0).toUpperCase() ? 1 : -1)) // sort names

  if (me.length === 0) return // app is unloading
  nameRecs.unshift(me[0][1].user) // push myself on to the front

  const avatars = elem('avatars')
  const currentCursors = []

  // check that an avatar exists for each name; if not create one.  If it does, check that it is still looking right
  nameRecs.forEach((nameRec) => {
    const ava = elem(`ava${nameRec.id}`)
    const shortName = initials(nameRec.name)
    if (ava === null) {
      makeAvatar(nameRec)
      refreshAvatars = true
    } else {
      // to avoid flashes, don't touch anything that is already correct
      if (ava.dataset.tooltip !== nameRec.name) ava.dataset.tooltip = nameRec.name
      const circle = ava.firstChild
      if (circle.style.backgroundColor !== nameRec.color) {
        circle.style.backgroundColor = nameRec.color
      }
      const circleBorderColor = nameRec.anon ? 'white' : 'black'
      if (circle.style.borderColor !== circleBorderColor) {
        circle.style.borderColor = circleBorderColor
      }
      if (circle.innerText !== shortName) circle.innerText = shortName
      const circleFontColor = circle.style.color
      if (circleFontColor !== (nameRec.isLight ? 'black' : 'white')) {
        circle.style.color = nameRec.isLight ? 'black' : 'white'
      }
      const opacity = nameRec.asleep ? 0.2 : 1.0
      if (circle.style.opacity !== opacity) circle.style.opacity = opacity
    }

    if (nameRec.id !== clientID) {
      // don't create a cursor for myself
      let cursorDiv = elem(nameRec.id)
      if (cursorDiv === null) {
        cursorDiv = makeCursor(nameRec)
      } else {
        if (nameRec.asleep) cursorDiv.style.display = 'none'
        if (cursorDiv.innerText !== shortName) cursorDiv.innerText = shortName
        if (cursorDiv.style.backgroundColor !== nameRec.color) {
          cursorDiv.style.backgroundColor = nameRec.color
        }
      }
      currentCursors.push(cursorDiv)
    }
  })

  // re-order the avatars into alpha order, without gaps, with me at the start

  const df = document.createDocumentFragment()
  nameRecs.forEach((nameRec) => {
    df.appendChild(elem(`ava${nameRec.id}`))
  })
  avatars.replaceChildren(df)

  // delete any cursors that remain from before
  const cursorsToDelete = Array.from(document.querySelectorAll('.shared-cursor')).filter(
    (a) => !currentCursors.includes(a)
  )
  cursorsToDelete.forEach((e) => e.remove())

  /**
   * create an avatar as a div with initials inside
   * @param {object} nameRec
   */
  function makeAvatar(nameRec) {
    const ava = document.createElement('div')
    ava.classList.add('hoverme')
    if (followme === nameRec.id) ava.classList.add('followme')
    ava.id = `ava${nameRec.id}`
    ava.dataset.tooltip = nameRec.name
    // the broadcast awareness sometimes loses a client (i.e. broadcasts that it has been removed)
    // when it actually hasn't (e.g. if there is a comms glitch).  So instead, we set a timer
    // and delete the avatar only if nothing is heard from that user for a minute
    ava.timer = setTimeout(removeAvatar, 60000, ava)
    const circle = document.createElement('div')
    circle.classList.add('round')
    circle.style.backgroundColor = nameRec.color
    if (nameRec.anon) circle.style.borderColor = 'white'
    circle.innerText = initials(nameRec.name)
    circle.style.color = nameRec.isLight ? 'black' : 'white'
    circle.style.opacity = nameRec.asleep ? 0.2 : 1.0
    circle.dataset.client = nameRec.id
    circle.dataset.userName = nameRec.name
    ava.appendChild(circle)
    avatars.appendChild(ava)
    circle.addEventListener('click', nameRec.id === clientID ? renameUser : follow)
    circle.addEventListener('contextmenu', selectUsersItems)
    circle.addEventListener('mouseover', () =>
      statusMsg(
        nameRec.id === clientID
          ? 'Click to change your name. Right click to select all your edits'
          : `Click to follow this person. Right click to select all this person's edits`
      )
    )
    circle.addEventListener('mouseout', () => clearStatusBar())
  }
  /**
   * make a pseudo cursor (a div)
   * @param {object} nameRec
   * @returns a div
   */
  function makeCursor(nameRec) {
    const cursorDiv = document.createElement('div')
    cursorDiv.className = 'shared-cursor'
    cursorDiv.id = nameRec.id
    cursorDiv.style.backgroundColor = nameRec.color
    cursorDiv.innerText = initials(nameRec.name)
    cursorDiv.style.color = nameRec.isLight ? 'black' : 'white'
    cursorDiv.style.display = 'none' // hide it until we get coordinates at next mousemove
    container.appendChild(cursorDiv)
    return cursorDiv
  }
}
/**
 * destroy the avatar - the user is no longer on line
 * @param {HTMLelement} ava
 */
function removeAvatar(ava) {
  refreshAvatars = true
  ava.remove()
}
function showUsersSwitch() {
  const on = elem('showUsersSwitch').checked
  document.querySelectorAll('div.shared-cursor').forEach((node) => {
    node.style.display = on ? 'block' : 'none'
  })
  elem('avatars').style.display = on ? 'flex' : 'none'
}
/**
 * User has clicked on an avatar.  Start following this avatar
 * @param {event} event
 */
function follow(event) {
  if (followme) unFollow()
  const user = parseInt(event.target.dataset.client, 10)
  if (user === clientID) return
  followme = user
  elem(`ava${followme}`).classList.add('followme')
  const userName = elem(`ava${user}`).dataset.tooltip
  alertMsg(`Following ${userName}`, 'info')
}
/**
 * User was following another user, but has now clicked off the avatar, so stop following
 */
function unFollow() {
  if (!followme) return
  elem(`ava${followme}`).classList.remove('followme')
  followme = undefined
  elem('errMsg').classList.remove('fadeInAndOut')
  clearStatusBar()
}
/**
 * move the map so that the followed cursor is always in the centre of the pane
 */
function followUser() {
  const userRec = yAwareness.getStates().get(followme)
  if (!userRec) return
  if (userRec.user.asleep) unFollow()
  const userPosition = userRec.cursor
  if (userPosition) network.moveTo({ position: userPosition })
}
/**
 * User has clicked on their own avatar.  Prompt them to change their own name.
 */
function renameUser() {
  const newName = prompt('Enter your new name', myNameRec.name)
  if (newName) {
    myNameRec.name = newName
    myNameRec.anon = false
    yAwareness.setLocalState({ user: myNameRec })
    showAvatars()
  }
  clearStatusBar()
}
/**
 * show a ghost box where another user is adding a factor
 * addingFactor is an object with properties:
 * state: adding', or 'done' to indicate that the ghost box should be removed
 * pos: a position (of the Add Factor dialog); 'done'
 * name: the name of the other user
 * @param {Integer} userId other user's client Id
 * @param {object} addingFactor
 */
function showGhostFactor(userId, addingFactor) {
  const id = `ghost-factor${userId}`
  switch (addingFactor.state) {
    case 'done': {
      const ghostDiv = elem(id)
      if (ghostDiv) ghostDiv.remove()
      break
    }
    case 'adding': {
      if (!elem(id)) {
        const ghostDiv = document.createElement('div')
        ghostDiv.className = 'ghost-factor'
        ghostDiv.id = `ghost-factor${userId}`
        ghostDiv.innerText = `[New factor\nbeing added by\n${addingFactor.name}]`
        const p = network.canvasToDOM(addingFactor.pos)
        const box = container.getBoundingClientRect()
        p.x += box.left
        p.y += box.top
        ghostDiv.style.top = `${p.y - 50}px`
        ghostDiv.style.left = `${p.x - 187}px`
        ghostDiv.style.display =
          p.x < box.left || p.x > box.right || p.y > box.bottom || p.y < box.top ? 'none' : 'block'
        netPane.appendChild(ghostDiv)
      }
      break
    }
    default:
      console.log(`Bad adding factor: ${addingFactor}`)
  }
}