Source: Annotation.js

import { BoundingBox } from './BoundingBox.js'

/**
 * Represents an annotation that can be drawn as an overlay on a canvas.
 * An annotation is a decorative element (text, graphics, glyph) that provides
 * additional context or information for interpreting underlying drawings.
 * 
 * Each annotation includes:
 * - A unique identifier
 * - Optional metadata (description, category, code, label)
 * - Visual representation (SVG, image, or element collection)
 * - Spatial information (region or bounding box)
 * - Style and state properties
 * 
 * Annotations can be serialized to/from JSON-LD format for interoperability
 * with Web Annotation standards.
 */
class Annotation {
  /**
   * Creates a new Annotation instance.
   * @param {Object} [options] - Configuration options for the annotation.
   * @param {string} [options.id] - Unique identifier for the annotation. Auto-generated if not provided.
   * @param {string} [options.code] - A code identifier for the annotation.
   * @param {string} [options.label=''] - Display label for the annotation.
   * @param {string} [options.description] - HTML text containing a comprehensive description.
   * @param {string} [options.class] - Category or classification of the annotation.
   * @param {string} [options.target] - Target element or area this annotation refers to.
   * @param {string} [options.svg] - SVG content for the annotation.
   * @param {Object} [options.image] - Image data associated with the annotation.
   * @param {Object} [options.region] - Region coordinates {x, y, w, h} for the annotation.
   * @param {Object} [options.data={}] - Additional custom data for the annotation.
   * @param {Object} [options.style] - Style configuration for rendering.
   * @param {BoundingBox} [options.bbox] - Bounding box of the annotation.
   * @param {boolean} [options.visible=true] - Visibility state of the annotation.
   * @param {Object} [options.state] - State variables for the annotation.
   * @param {boolean} [options.ready=false] - Indicates if SVG conversion is complete.
   * @param {boolean} [options.needsUpdate=true] - Indicates if annotation needs updating.
   * @param {boolean} [options.editing=false] - Indicates if annotation is being edited.
   */
  constructor(options = {}) {
    // Set default properties
    this.id = options.id ?? Annotation.generateUUID();
    this.code = options.code ?? null;
    this.label = options.label ?? '';
    this.description = options.description ?? null;
    this.class = options.class ?? null;
    this.target = options.target ?? null;
    this.svg = options.svg ?? null;
    this.image = options.image ?? null;
    this.region = options.region ?? null;
    this.data = options.data ?? {};
    this.style = options.style ?? null;
    this.bbox = options.bbox ?? null;
    this.visible = options.visible ?? true;
    this.state = options.state ?? null;
    this.ready = options.ready ?? false;
    this.needsUpdate = options.needsUpdate ?? true;
    this.editing = options.editing ?? false;
    
    // Initialize elements array
    this.elements = Array.isArray(options.elements) ? options.elements : [];
  }

  /**
   * Generates a UUID (Universally Unique Identifier) for annotation instances.
   * @returns {string} A newly generated UUID.
   * @private
   */
  static generateUUID() {
    // Use modern approach for UUID generation
    return 'a' + ([1e7]+-1e3+-4e3+-8e3+-1e11).replace(/[018]/g, c =>
      (c ^ crypto.getRandomValues(new Uint8Array(1))[0] & 15 >> c / 4).toString(16)
    );
  }

  /**
   * Calculates and returns the bounding box of the annotation based on its elements or region.
   * The coordinates are always relative to the top-left corner of the canvas.
   * @returns {BoundingBox} The calculated bounding box of the annotation.
   */
  getBBoxFromElements() {
    // If no elements exist, use region or return empty bounding box
    if (!this.elements.length) {
      if (this.region == null) {
        return new BoundingBox();
      }
      
      const r = this.region;
      return new BoundingBox({ 
        xLow: r.x, 
        yLow: r.y, 
        xHigh: r.x + r.w, 
        yHigh: r.y + r.h 
      });
    }
    
    // Calculate bounding box from elements
    const firstBBox = this.elements[0].getBBox();
    let x = firstBBox.x;
    let y = firstBBox.y;
    let width = firstBBox.width;
    let height = firstBBox.height;
    
    // Expand bounding box to encompass all elements
    for (let i = 1; i < this.elements.length; i++) {
      const { x: sx, y: sy, width: swidth, height: sheight } = this.elements[i].getBBox();
      
      x = Math.min(x, sx);
      y = Math.min(y, sy); // Fixed: comparing y with sy instead of x with sy
      
      const xMax = Math.max(x + width, sx + swidth);
      const yMax = Math.max(y + height, sy + sheight);
      
      width = xMax - x;
      height = yMax - y;
    }
    
    return new BoundingBox({ 
      xLow: x, 
      yLow: y, 
      xHigh: x + width, 
      yHigh: y + height // Fixed: using height instead of width 
    });
  }

  /**
   * Creates an Annotation instance from a JSON-LD format object.
   * @param {Object} entry - The JSON-LD object representing an annotation.
   * @returns {Annotation} A new Annotation instance.
   * @throws {Error} If the entry is not a valid JSON-LD annotation or contains unsupported selectors.
   */
  static fromJsonLd(entry) {
    if (entry.type !== 'Annotation') {
      throw new Error("Not a valid JSON-LD annotation");
    }
    
    const options = { id: entry.id };

    // Map JSON-LD properties to annotation properties
    const propertyMap = { 
      'identifying': 'code', 
      'classifying': 'class', 
      'describing': 'description' 
    };
    
    if (Array.isArray(entry.body)) {
      for (const item of entry.body) {
        const field = propertyMap[item.purpose];
        if (field) {
          options[field] = item.value;
        }
      }
    }
    
    // Process target selector if present
    const selector = entry.target?.selector;
    if (selector) {
      switch (selector.type) {
        case 'SvgSelector':
          options.svg = selector.value;
          options.elements = [];
          break;
        default:
          throw new Error(`Unsupported selector: ${selector.type}`);
      }
    }
    
    return new Annotation(options);
  }

  /**
   * Converts the annotation to a JSON-LD format object.
   * @returns {Object} A JSON-LD representation of the annotation.
   */
  toJsonLd() {
    const body = [];
    
    // Add properties to body if they exist
    if (this.code !== null) {
      body.push({ 
        type: 'TextualBody', 
        value: this.code, 
        purpose: 'identifying' // Fixed: correct spelling
      });
    }
    
    if (this.class !== null) {
      body.push({ 
        type: 'TextualBody', 
        value: this.class, 
        purpose: 'classifying' 
      });
    }
    
    if (this.description !== null) {
      body.push({ 
        type: 'TextualBody', 
        value: this.description, 
        purpose: 'describing' 
      });
    }

    // Create the base JSON-LD object
    const jsonLd = {
      "@context": "http://www.w3.org/ns/anno.jsonld",
      id: this.id,
      type: "Annotation",
      body: body,
      target: { selector: {} }
    };
    
    // Add target information if available
    if (this.target) {
      jsonLd.target.selector.source = this.target;
    }

    // Add SVG representation if elements exist
    if (this.elements.length > 0) {
      // Get the first element or combine them if needed
      const element = this.elements[0]; // Simplified for now
      if (element) {
        const serializer = new XMLSerializer();
        jsonLd.target.selector.type = 'SvgSelector';
        jsonLd.target.selector.value = serializer.serializeToString(element);
      }
    } else if (this.svg) {
      // Use existing SVG if available
      jsonLd.target.selector.type = 'SvgSelector';
      jsonLd.target.selector.value = this.svg;
    }
    
    return jsonLd;
  }
}

export { Annotation }