Files
yjs/attribution-manager.md
2025-07-25 16:08:42 +02:00

6.9 KiB

Attribution Feature

The Attribution feature extends Yjs types to provide rich metadata about content changes, including information about who created, deleted, or formatted content. This enables powerful collaborative editing features such as authorship tracking and change visualization. The information about who performed which changes can be handled by a separate CRDT (which is part of the attribution manager).

Core Concepts

Attribution Manager

The attributionManager is the central component that tracks and manages attribution data. It must be passed to methods that support attribution to enable the feature.

Different implementations of AttributionManager are available for different use cases:

  • DiffingAttributionManager: Highlights the differences between two Yjs documents
  • SnapshotAttributionManager: Highlights the differences between two snapshots

Attributed Content

Attributed content includes standard Yjs operations enhanced with attribution metadata:

// Standard content
[{ insert: 'hello world' }]

// Attributed content
[
  { insert: 'hello', attribution: { insert: ['kevin'] } },
  { insert: ' world', attribution: { insert: ['alice'] } }
]

Delete Attribution

Deleted content is represented in attributed results to maintain authorship information and proper position tracking:

// Shows deleted content with attribution
[
  { insert: 'hello ', attribution: { delete: ['kevin'] } },
  { insert: 'world' }
]

API Reference

YText

getDelta([attributionManager])

Returns the delta representation of the YText content, optionally with attribution information.

Parameters:

  • attributionManager (optional): The attribution manager instance

Returns:

  • Array of delta operations, with attribution metadata if attributionManager is provided

Examples:

const ytext = new Y.Text()
// Content is inserted during collaborative editing
// Attribution is handled automatically by the server

// Without attribution
const delta = ytext.getDelta()
// [{ insert: 'hello world' }]

// With attribution
const attributedDelta = ytext.getDelta(attributionManager)
// [
//   { insert: 'hello', attribution: { insert: ['kevin'] } },
//   { insert: ' world', attribution: { insert: ['alice'] } }
// ]

getContent([attributionManager])

Returns the content representation with optional attribution information.

Parameters:

  • attributionManager (optional): The attribution manager instance

Returns:

  • Content representation with attribution metadata if attributionManager is provided

YArray

getContent([attributionManager])

Returns the array content with optional attribution information for each element.

Parameters:

  • attributionManager (optional): The attribution manager instance

Returns:

  • Array content with attribution metadata if attributionManager is provided

YMap

getContent([attributionManager])

Returns the map content with optional attribution information for each key-value pair.

Parameters:

  • attributionManager (optional): The attribution manager instance

Returns:

  • Map content with attribution metadata if attributionManager is provided

Position Adjustments

When working with attributed content, position calculations must account for deleted content that appears in the attributed representation but not in the standard representation.

Example: Position Adjustment

// Standard content (length: 5)
ytext.toString() // "world"

// Attributed content (includes deleted content)
ytext.getDelta(attributionManager)
// [
//   { insert: 'hello ', attribution: { delete: ['kevin'] } },  // positions 0-5
//   { insert: 'world' }                                         // positions 6-10
// ]

// To insert after "world":
// - Standard position: 5 (after "world")
// - Attributed position: 11 (after "world" accounting for deleted "hello ")

Use Cases

Events in Yjs are enhanced to work with attributed content, automatically adjusting positions when attribution is considered.

Event Position Adjustment

When an attributionManager is used, event positions are automatically adjusted to account for deleted content.

Example:

// Initial content: "hello world"
// User deletes "hello " (positions 0-6)
// Current visible content: "world"

ytext.observe((event, transaction) => {
  // User wants to insert "!" after "world"
  
  // Standard event (without attribution)
  const standardDelta = event.getDelta()
  // Shows insertion at position 5 (after "world" in visible content)
  
  // Attributed event (with attribution manager)
  const attributedDelta = event.getDelta(attributionManager)
  // Shows insertion at position 11 (accounting for deleted "hello ")
  // [
  //   { insert: 'hello ', attribution: { delete: ['kevin'] } },
  //   { insert: 'world' },
  //   { insert: '!' }  // inserted at attributed position 11
  // ]
})

Use Cases

Authorship Visualization

Display content with visual indicators of who created each part:

function renderWithAuthorship(ytext, attributionManager) {
  const attributedDelta = ytext.getDelta(attributionManager)
  
  return attributedDelta.map(op => {
    const author = op.attribution?.insert?.[0] || 'unknown'
    const isDeleted = op.attribution?.delete
    
    return {
      content: op.insert,
      author,
      isDeleted,
      className: `author-${author} ${isDeleted ? 'deleted' : ''}`
    }
  })
}

Change Tracking

Track who made specific changes to content:

function trackChanges(ytext, attributionManager) {
  ytext.observe((event, transaction) => {
    const changes = event.changes.getAttributedDelta?.(attributionManager) || event.changes.delta
    
    changes.forEach(change => {
      if (change.attribution) {
        console.log(`Change by ${change.attribution.insert?.[0] || change.attribution.delete?.[0]}:`, change)
      }
    })
  })
}

Best Practices

Attribution Manager Lifecycle

  • Create one attribution manager per document or collaboration session
  • Ensure the attribution manager is consistently used across all operations
  • Pass the same attribution manager instance to all methods that need attribution

Migration Guide

Upgrading Existing Code

To add attribution support to existing Yjs applications:

  1. Add attribution manager: Create and configure an attribution manager
  2. Update method calls: Add the attribution manager parameter to relevant method calls
  3. Handle attributed content: Update code to handle the new attribution metadata format
  4. Adjust position calculations: Update position calculations to account for deleted content

Backward Compatibility

The Attribution feature is fully backward compatible:

  • All existing methods work without the attribution manager parameter
  • Existing code continues to work unchanged
  • Attribution is opt-in and doesn't affect performance when not used