y-rb 0.6.0-aarch64-linux-musl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/ext/yrb/Cargo.toml +21 -0
- data/ext/yrb/extconf.rb +6 -0
- data/ext/yrb/src/lib.rs +649 -0
- data/ext/yrb/src/utils.rs +64 -0
- data/ext/yrb/src/yany.rs +46 -0
- data/ext/yrb/src/yarray.rs +191 -0
- data/ext/yrb/src/yattrs.rs +39 -0
- data/ext/yrb/src/yawareness.rs +151 -0
- data/ext/yrb/src/ydiff.rs +19 -0
- data/ext/yrb/src/ydoc.rs +113 -0
- data/ext/yrb/src/ymap.rs +175 -0
- data/ext/yrb/src/ytext.rs +235 -0
- data/ext/yrb/src/ytransaction.rs +127 -0
- data/ext/yrb/src/yvalue.rs +256 -0
- data/ext/yrb/src/yxml_element.rs +280 -0
- data/ext/yrb/src/yxml_fragment.rs +129 -0
- data/ext/yrb/src/yxml_text.rs +177 -0
- data/lib/3.1/yrb.so +0 -0
- data/lib/3.2/yrb.so +0 -0
- data/lib/3.3/yrb.so +0 -0
- data/lib/3.4/yrb.so +0 -0
- data/lib/y/array.rb +371 -0
- data/lib/y/awareness.rb +290 -0
- data/lib/y/diff.rb +38 -0
- data/lib/y/doc.rb +313 -0
- data/lib/y/map.rb +199 -0
- data/lib/y/text.rb +383 -0
- data/lib/y/transaction.rb +189 -0
- data/lib/y/version.rb +5 -0
- data/lib/y/xml.rb +1141 -0
- data/lib/y-rb.rb +24 -0
- data/lib/y.rb +3 -0
- metadata +143 -0
    
        data/lib/y/map.rb
    ADDED
    
    | @@ -0,0 +1,199 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "json"
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            module Y
         | 
| 6 | 
            +
              # A map can be used to store and retrieve key-value pairs.
         | 
| 7 | 
            +
              #
         | 
| 8 | 
            +
              # The map is the replicated counterpart to a Hash. It supports a subset
         | 
| 9 | 
            +
              # of the Hash operations, like adding, getting and deleting values by key.
         | 
| 10 | 
            +
              #
         | 
| 11 | 
            +
              # Someone should not instantiate a map directly, but use {Y::Doc#get_map}
         | 
| 12 | 
            +
              # instead.
         | 
| 13 | 
            +
              #
         | 
| 14 | 
            +
              # @example
         | 
| 15 | 
            +
              #   doc = Y::Doc.new
         | 
| 16 | 
            +
              #   map = doc.get_map("my map")
         | 
| 17 | 
            +
              #
         | 
| 18 | 
            +
              #   map[:hello] = "world"
         | 
| 19 | 
            +
              #   puts map[:hello]
         | 
| 20 | 
            +
              class Map
         | 
| 21 | 
            +
                include Enumerable
         | 
| 22 | 
            +
             | 
| 23 | 
            +
                # @!attribute [r] document
         | 
| 24 | 
            +
                #
         | 
| 25 | 
            +
                # @return [Y::Doc] The document this map belongs to
         | 
| 26 | 
            +
                attr_accessor :document
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                # Create a new map instance
         | 
| 29 | 
            +
                #
         | 
| 30 | 
            +
                # @param doc [Y::Doc]
         | 
| 31 | 
            +
                def initialize(doc = nil)
         | 
| 32 | 
            +
                  @document = doc || Y::Doc.new
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                  super()
         | 
| 35 | 
            +
                end
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                # Attach a listener to get notified about any changes to the map
         | 
| 38 | 
            +
                #
         | 
| 39 | 
            +
                # @param callback [Proc]
         | 
| 40 | 
            +
                # @param block [Block]
         | 
| 41 | 
            +
                # @return [Integer]
         | 
| 42 | 
            +
                def attach(callback, &block)
         | 
| 43 | 
            +
                  return ymap_observe(callback) unless callback.nil?
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                  ymap_observe(block.to_proc) unless block.nil?
         | 
| 46 | 
            +
                end
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                # Removes all map entries
         | 
| 49 | 
            +
                #
         | 
| 50 | 
            +
                # @return [Self]
         | 
| 51 | 
            +
                def clear
         | 
| 52 | 
            +
                  document.current_transaction { |tx| ymap_clear(tx) }
         | 
| 53 | 
            +
                  self
         | 
| 54 | 
            +
                end
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                # rubocop:disable Layout/LineLength
         | 
| 57 | 
            +
             | 
| 58 | 
            +
                # Deletes the entry for the given key and returns its associated value.
         | 
| 59 | 
            +
                #
         | 
| 60 | 
            +
                # @example Deletes the entry and return associated value
         | 
| 61 | 
            +
                #
         | 
| 62 | 
            +
                #   m = doc.get_map("my map")
         | 
| 63 | 
            +
                #   m[:bar] = 1
         | 
| 64 | 
            +
                #   m.delete(:bar) # => 1
         | 
| 65 | 
            +
                #   m # => {}
         | 
| 66 | 
            +
                #
         | 
| 67 | 
            +
                # @example Unknown key is handled in block
         | 
| 68 | 
            +
                #
         | 
| 69 | 
            +
                #   m = doc.get_map("my map")
         | 
| 70 | 
            +
                #   m.delete(:nosuch) { |key| "Key #{key} not found" }# => "Key nosuch not found"
         | 
| 71 | 
            +
                #   m # => {}
         | 
| 72 | 
            +
                #
         | 
| 73 | 
            +
                # @param key [String, Symbol]
         | 
| 74 | 
            +
                # @return [void]
         | 
| 75 | 
            +
                def delete(key)
         | 
| 76 | 
            +
                  value = document.current_transaction { |tx| ymap_remove(tx, key) }
         | 
| 77 | 
            +
                  if block_given? && key?(key)
         | 
| 78 | 
            +
                    yield key
         | 
| 79 | 
            +
                  else
         | 
| 80 | 
            +
                    value
         | 
| 81 | 
            +
                  end
         | 
| 82 | 
            +
                end
         | 
| 83 | 
            +
             | 
| 84 | 
            +
                # rubocop:enable Layout/LineLength
         | 
| 85 | 
            +
             | 
| 86 | 
            +
                # Detach listener
         | 
| 87 | 
            +
                #
         | 
| 88 | 
            +
                # @param subscription_id [Integer]
         | 
| 89 | 
            +
                # @return [void]
         | 
| 90 | 
            +
                def detach(subscription_id)
         | 
| 91 | 
            +
                  ymap_unobserve(subscription_id)
         | 
| 92 | 
            +
                end
         | 
| 93 | 
            +
             | 
| 94 | 
            +
                # @return [void]
         | 
| 95 | 
            +
                def each(&block)
         | 
| 96 | 
            +
                  document.current_transaction { |tx| ymap_each(tx, block) }
         | 
| 97 | 
            +
                end
         | 
| 98 | 
            +
             | 
| 99 | 
            +
                # @return [true, false]
         | 
| 100 | 
            +
                def key?(key)
         | 
| 101 | 
            +
                  document.current_transaction { |tx| ymap_contains(tx, key) }
         | 
| 102 | 
            +
                end
         | 
| 103 | 
            +
             | 
| 104 | 
            +
                alias has_key? key?
         | 
| 105 | 
            +
             | 
| 106 | 
            +
                # @return [Object]
         | 
| 107 | 
            +
                def [](key)
         | 
| 108 | 
            +
                  document.current_transaction { |tx| ymap_get(tx, key) }
         | 
| 109 | 
            +
                end
         | 
| 110 | 
            +
             | 
| 111 | 
            +
                # @return [void]
         | 
| 112 | 
            +
                def []=(key, val)
         | 
| 113 | 
            +
                  document.current_transaction { |tx| ymap_insert(tx, key, val) }
         | 
| 114 | 
            +
                end
         | 
| 115 | 
            +
             | 
| 116 | 
            +
                # Returns size of map
         | 
| 117 | 
            +
                #
         | 
| 118 | 
            +
                # @return [Integer]
         | 
| 119 | 
            +
                def size
         | 
| 120 | 
            +
                  document.current_transaction { |tx| ymap_size(tx) }
         | 
| 121 | 
            +
                end
         | 
| 122 | 
            +
             | 
| 123 | 
            +
                # Returns a Hash representation of this map
         | 
| 124 | 
            +
                #
         | 
| 125 | 
            +
                # @return [Hash]
         | 
| 126 | 
            +
                def to_h
         | 
| 127 | 
            +
                  document.current_transaction { |tx| ymap_to_h(tx) }
         | 
| 128 | 
            +
                end
         | 
| 129 | 
            +
             | 
| 130 | 
            +
                # Returns a JSON representation of map
         | 
| 131 | 
            +
                #
         | 
| 132 | 
            +
                # @return [String] JSON string
         | 
| 133 | 
            +
                def to_json(*_args)
         | 
| 134 | 
            +
                  to_h.to_json
         | 
| 135 | 
            +
                end
         | 
| 136 | 
            +
             | 
| 137 | 
            +
                # @!method ymap_clear(tx)
         | 
| 138 | 
            +
                #   Removes all key-value pairs from Map
         | 
| 139 | 
            +
                #
         | 
| 140 | 
            +
                # @param tx [Y::Transaction]
         | 
| 141 | 
            +
             | 
| 142 | 
            +
                # @!method ymap_contains(tx, key)
         | 
| 143 | 
            +
                #   Check if a certain key is in the Map
         | 
| 144 | 
            +
                #
         | 
| 145 | 
            +
                # @param tx [Y::Transaction]
         | 
| 146 | 
            +
                # @param key [String, Symbol]
         | 
| 147 | 
            +
                # @return [Boolean] True, if and only if the key exists
         | 
| 148 | 
            +
             | 
| 149 | 
            +
                # @!method ymap_each(tx, proc)
         | 
| 150 | 
            +
                #   Iterates over all key-value pairs in Map by calling the provided proc
         | 
| 151 | 
            +
                #   with the key and the value as arguments.
         | 
| 152 | 
            +
                #
         | 
| 153 | 
            +
                # @param tx [Y::Transaction]
         | 
| 154 | 
            +
                # @param proc [Proc<String, Any>] A proc that is called for every element
         | 
| 155 | 
            +
             | 
| 156 | 
            +
                # @!method ymap_get(tx, key)
         | 
| 157 | 
            +
                #   Returns stored value for key or nil if none is present
         | 
| 158 | 
            +
                #
         | 
| 159 | 
            +
                # @param tx [Y::Transaction]
         | 
| 160 | 
            +
                # @param key [String, Symbol]
         | 
| 161 | 
            +
                # @return [Object, nil] Value or nil
         | 
| 162 | 
            +
             | 
| 163 | 
            +
                # @!method ymap_insert(tx, key, value)
         | 
| 164 | 
            +
                #   Insert value for key. In case the key already exists, the previous value
         | 
| 165 | 
            +
                #   will be overwritten.
         | 
| 166 | 
            +
                #
         | 
| 167 | 
            +
                # @param tx [Y::Transaction]
         | 
| 168 | 
            +
                # @param key [String, Symbol]
         | 
| 169 | 
            +
                # @param value [Object]
         | 
| 170 | 
            +
             | 
| 171 | 
            +
                # @!method ymap_observe(callback)
         | 
| 172 | 
            +
                #
         | 
| 173 | 
            +
                # @param callback [Proc]
         | 
| 174 | 
            +
                # @return [Integer]
         | 
| 175 | 
            +
             | 
| 176 | 
            +
                # @!method ymap_remove(tx, key)
         | 
| 177 | 
            +
                #   Removes key-value pair from Map if key exists.
         | 
| 178 | 
            +
                #
         | 
| 179 | 
            +
                # @param tx [Y::Transaction]
         | 
| 180 | 
            +
                # @param key [String, Symbol]
         | 
| 181 | 
            +
             | 
| 182 | 
            +
                # @!method ymap_size(tx)
         | 
| 183 | 
            +
                #   Returns number of key-value pairs stored in map
         | 
| 184 | 
            +
                #
         | 
| 185 | 
            +
                # @param tx [Y::Transaction]
         | 
| 186 | 
            +
                # @return [Integer] Number of key-value pairs
         | 
| 187 | 
            +
             | 
| 188 | 
            +
                # @!method ymap_to_h(tx)
         | 
| 189 | 
            +
                #   Returns a Hash representation of the Map
         | 
| 190 | 
            +
                #
         | 
| 191 | 
            +
                # @param tx [Y::Transaction]
         | 
| 192 | 
            +
                # @return [Hash] Hash representation of Map
         | 
| 193 | 
            +
             | 
| 194 | 
            +
                # @!method ymap_unobserve(subscription_id)
         | 
| 195 | 
            +
                #
         | 
| 196 | 
            +
                # @param subscription_id [Integer]
         | 
| 197 | 
            +
                # @return [void]
         | 
| 198 | 
            +
              end
         | 
| 199 | 
            +
            end
         | 
    
        data/lib/y/text.rb
    ADDED
    
    | @@ -0,0 +1,383 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module Y
         | 
| 4 | 
            +
              # A text can be used insert and remove string fragments. It also supports
         | 
| 5 | 
            +
              # formatting and the concept of embeds, which are supported data types that
         | 
| 6 | 
            +
              # added as metadata.
         | 
| 7 | 
            +
              #
         | 
| 8 | 
            +
              # The text is the replicated counterpart to a String. It supports a subset
         | 
| 9 | 
            +
              # of String operations, like appending, insert at position and slicing.
         | 
| 10 | 
            +
              #
         | 
| 11 | 
            +
              # Someone should not instantiate a text directly, but use {Y::Doc#get_text}
         | 
| 12 | 
            +
              # instead.
         | 
| 13 | 
            +
              #
         | 
| 14 | 
            +
              # @example
         | 
| 15 | 
            +
              #   doc = Y::Doc.new
         | 
| 16 | 
            +
              #   text = doc.get_text("my text")
         | 
| 17 | 
            +
              #
         | 
| 18 | 
            +
              #   text << "Hello, World!"
         | 
| 19 | 
            +
              #   puts text.to_s
         | 
| 20 | 
            +
              class Text
         | 
| 21 | 
            +
                # @!attribute [r] document
         | 
| 22 | 
            +
                #
         | 
| 23 | 
            +
                # @return [Y::Doc] The document this text belongs to
         | 
| 24 | 
            +
                attr_accessor :document
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                # Create a new text instance
         | 
| 27 | 
            +
                #
         | 
| 28 | 
            +
                # @param doc [Y::Doc]
         | 
| 29 | 
            +
                def initialize(doc = nil)
         | 
| 30 | 
            +
                  @document = doc || Y::Doc.new
         | 
| 31 | 
            +
             | 
| 32 | 
            +
                  super()
         | 
| 33 | 
            +
                end
         | 
| 34 | 
            +
             | 
| 35 | 
            +
                # Appends a string at the end of the text
         | 
| 36 | 
            +
                #
         | 
| 37 | 
            +
                # @param str [String]
         | 
| 38 | 
            +
                # @return [void]
         | 
| 39 | 
            +
                def <<(str)
         | 
| 40 | 
            +
                  document.current_transaction { |tx| ytext_push(tx, str) }
         | 
| 41 | 
            +
                end
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                # Attach listener to text changes
         | 
| 44 | 
            +
                #
         | 
| 45 | 
            +
                # @example Listen to changes in text type
         | 
| 46 | 
            +
                #   local = Y::Doc.new
         | 
| 47 | 
            +
                #
         | 
| 48 | 
            +
                #   text = local.get_text("my text")
         | 
| 49 | 
            +
                #   text.attach(->(delta) { pp delta }) # { insert: "Hello, World!" }
         | 
| 50 | 
            +
                #
         | 
| 51 | 
            +
                #   local.transact do
         | 
| 52 | 
            +
                #     text << "Hello, World!"
         | 
| 53 | 
            +
                #   end
         | 
| 54 | 
            +
                #
         | 
| 55 | 
            +
                # @example Listen to changes in text type
         | 
| 56 | 
            +
                #   local = Y::Doc.new
         | 
| 57 | 
            +
                #
         | 
| 58 | 
            +
                #   text = local.get_text("my text")
         | 
| 59 | 
            +
                #   text.attach(->(delta) { pp delta }) # { insert: "Hello, World!" }
         | 
| 60 | 
            +
                #
         | 
| 61 | 
            +
                #   text << "Hello, World!"
         | 
| 62 | 
            +
                #
         | 
| 63 | 
            +
                #   # todo: required, otherwise segfault
         | 
| 64 | 
            +
                #   local.commit
         | 
| 65 | 
            +
                #
         | 
| 66 | 
            +
                # @param callback [Proc]
         | 
| 67 | 
            +
                # @param block [Block]
         | 
| 68 | 
            +
                # @return [Integer]
         | 
| 69 | 
            +
                def attach(callback, &block)
         | 
| 70 | 
            +
                  return ytext_observe(callback) unless callback.nil?
         | 
| 71 | 
            +
             | 
| 72 | 
            +
                  ytext_observe(block.to_proc) unless block.nil?
         | 
| 73 | 
            +
                end
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                # Detach listener
         | 
| 76 | 
            +
                #
         | 
| 77 | 
            +
                # @param subscription_id [Integer]
         | 
| 78 | 
            +
                # @return [void]
         | 
| 79 | 
            +
                def detach(subscription_id)
         | 
| 80 | 
            +
                  ytext_unobserve(subscription_id)
         | 
| 81 | 
            +
                end
         | 
| 82 | 
            +
             | 
| 83 | 
            +
                # Diff
         | 
| 84 | 
            +
                #
         | 
| 85 | 
            +
                # @return[Array<YDiff>]
         | 
| 86 | 
            +
                def diff
         | 
| 87 | 
            +
                  document.current_transaction do |tx|
         | 
| 88 | 
            +
                    ytext_diff(tx)
         | 
| 89 | 
            +
                  end
         | 
| 90 | 
            +
                end
         | 
| 91 | 
            +
             | 
| 92 | 
            +
                # Checks if text is empty
         | 
| 93 | 
            +
                #
         | 
| 94 | 
            +
                # @example Check if text is empty
         | 
| 95 | 
            +
                #   doc = Y::Doc.new
         | 
| 96 | 
            +
                #   text = doc.get_text("my text")
         | 
| 97 | 
            +
                #
         | 
| 98 | 
            +
                #   text.empty? # true
         | 
| 99 | 
            +
                #
         | 
| 100 | 
            +
                # @return [TrueClass,FalseClass]
         | 
| 101 | 
            +
                def empty?
         | 
| 102 | 
            +
                  length.zero?
         | 
| 103 | 
            +
                end
         | 
| 104 | 
            +
             | 
| 105 | 
            +
                # rubocop:disable Metrics/MethodLength
         | 
| 106 | 
            +
             | 
| 107 | 
            +
                # Insert a value at position and with optional attributes. This method is
         | 
| 108 | 
            +
                # similar to [String#insert](https://ruby-doc.org/core-3.1.2/String.html),
         | 
| 109 | 
            +
                # except for the optional third `attrs` argument.
         | 
| 110 | 
            +
                #
         | 
| 111 | 
            +
                # @example Insert a string at position
         | 
| 112 | 
            +
                #   doc = Y::Doc.new
         | 
| 113 | 
            +
                #   text = doc.get_text("my text")
         | 
| 114 | 
            +
                #   text << "Hello, "
         | 
| 115 | 
            +
                #
         | 
| 116 | 
            +
                #   text.insert(7, "World!")
         | 
| 117 | 
            +
                #
         | 
| 118 | 
            +
                #   puts text.to_s == "Hello, World!" # true
         | 
| 119 | 
            +
                #
         | 
| 120 | 
            +
                # The value can be any of the supported types:
         | 
| 121 | 
            +
                # - Boolean
         | 
| 122 | 
            +
                # - String
         | 
| 123 | 
            +
                # - Numeric
         | 
| 124 | 
            +
                # - Array (where element types must be supported)
         | 
| 125 | 
            +
                # - Hash (where the the types of key and values must be supported)
         | 
| 126 | 
            +
                #
         | 
| 127 | 
            +
                # @param index [Integer]
         | 
| 128 | 
            +
                # @param value [String, Numeric, Array, Hash]
         | 
| 129 | 
            +
                # @param attrs [Hash, nil]
         | 
| 130 | 
            +
                # @return [void]
         | 
| 131 | 
            +
                def insert(index, value, attrs = nil)
         | 
| 132 | 
            +
                  document.current_transaction do |tx|
         | 
| 133 | 
            +
                    if value.is_a?(String)
         | 
| 134 | 
            +
                      ytext_insert(tx, index, value) if attrs.nil?
         | 
| 135 | 
            +
                      unless attrs.nil?
         | 
| 136 | 
            +
                        ytext_insert_with_attributes(tx, index, value, attrs)
         | 
| 137 | 
            +
                      end
         | 
| 138 | 
            +
                      return nil
         | 
| 139 | 
            +
                    end
         | 
| 140 | 
            +
             | 
| 141 | 
            +
                    if can_insert?(value)
         | 
| 142 | 
            +
                      ytext_insert_embed(tx, index, value) if attrs.nil?
         | 
| 143 | 
            +
                      unless attrs.nil?
         | 
| 144 | 
            +
                        ytext_insert_embed_with_attributes(tx, index, value, attrs)
         | 
| 145 | 
            +
                      end
         | 
| 146 | 
            +
                      return nil
         | 
| 147 | 
            +
                    end
         | 
| 148 | 
            +
             | 
| 149 | 
            +
                    raise ArgumentError,
         | 
| 150 | 
            +
                          "Can't insert value. `#{value.class.name}` isn't supported."
         | 
| 151 | 
            +
                  end
         | 
| 152 | 
            +
                end
         | 
| 153 | 
            +
             | 
| 154 | 
            +
                # rubocop:enable Metrics/MethodLength
         | 
| 155 | 
            +
             | 
| 156 | 
            +
                # Applies formatting to text
         | 
| 157 | 
            +
                #
         | 
| 158 | 
            +
                # @example Add formatting to first word
         | 
| 159 | 
            +
                #   doc = Y::Doc.new
         | 
| 160 | 
            +
                #   text = doc.get_text("my text")
         | 
| 161 | 
            +
                #
         | 
| 162 | 
            +
                #   attrs = {format: "bold"}
         | 
| 163 | 
            +
                #   text.format(0, 2, attrs)
         | 
| 164 | 
            +
                #
         | 
| 165 | 
            +
                # @param index [Integer]
         | 
| 166 | 
            +
                # @param length [Integer]
         | 
| 167 | 
            +
                # @param attrs [Hash]
         | 
| 168 | 
            +
                # @return [void]
         | 
| 169 | 
            +
                def format(index, length, attrs)
         | 
| 170 | 
            +
                  document.current_transaction do |tx|
         | 
| 171 | 
            +
                    ytext_format(tx, index, length, attrs)
         | 
| 172 | 
            +
                  end
         | 
| 173 | 
            +
                end
         | 
| 174 | 
            +
             | 
| 175 | 
            +
                # Returns length of text
         | 
| 176 | 
            +
                #
         | 
| 177 | 
            +
                # @return [Integer] Length of text
         | 
| 178 | 
            +
                def length
         | 
| 179 | 
            +
                  document.current_transaction { |tx| ytext_length(tx) }
         | 
| 180 | 
            +
                end
         | 
| 181 | 
            +
             | 
| 182 | 
            +
                alias size length
         | 
| 183 | 
            +
             | 
| 184 | 
            +
                # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength
         | 
| 185 | 
            +
             | 
| 186 | 
            +
                # Removes a part from text
         | 
| 187 | 
            +
                #
         | 
| 188 | 
            +
                # **Attention:** In comparison to String#slice, {Text#slice!} will not
         | 
| 189 | 
            +
                # return the substring that gets removed. Even this being technically
         | 
| 190 | 
            +
                # possible, it requires us to read the substring before removing it, which
         | 
| 191 | 
            +
                # is not desirable in most situations.
         | 
| 192 | 
            +
                #
         | 
| 193 | 
            +
                # @example Removes a single character
         | 
| 194 | 
            +
                #   doc = Y::Doc.new
         | 
| 195 | 
            +
                #
         | 
| 196 | 
            +
                #   text = doc.get_text("my text")
         | 
| 197 | 
            +
                #   text << "Hello"
         | 
| 198 | 
            +
                #
         | 
| 199 | 
            +
                #   text.slice!(0)
         | 
| 200 | 
            +
                #
         | 
| 201 | 
            +
                #   text.to_s == "ello" # true
         | 
| 202 | 
            +
                #
         | 
| 203 | 
            +
                # @example Removes a range of characters
         | 
| 204 | 
            +
                #   doc = Y::Doc.new
         | 
| 205 | 
            +
                #
         | 
| 206 | 
            +
                #   text = doc.get_text("my text")
         | 
| 207 | 
            +
                #   text << "Hello"
         | 
| 208 | 
            +
                #
         | 
| 209 | 
            +
                #   text.slice!(1..2)
         | 
| 210 | 
            +
                #   text.to_s == "Hlo" # true
         | 
| 211 | 
            +
                #
         | 
| 212 | 
            +
                #   text.slice!(1...2)
         | 
| 213 | 
            +
                #   text.to_s == "Ho" # true
         | 
| 214 | 
            +
                #
         | 
| 215 | 
            +
                # @example Removes a range of chars from start and for given length
         | 
| 216 | 
            +
                #   doc = Y::Doc.new
         | 
| 217 | 
            +
                #
         | 
| 218 | 
            +
                #   text = doc.get_text("my text")
         | 
| 219 | 
            +
                #   text << "Hello"
         | 
| 220 | 
            +
                #
         | 
| 221 | 
            +
                #   text.slice!(0, 3)
         | 
| 222 | 
            +
                #
         | 
| 223 | 
            +
                #   text.to_s == "lo" # true
         | 
| 224 | 
            +
                #
         | 
| 225 | 
            +
                # @overload slice!(index)
         | 
| 226 | 
            +
                #   Removes a single character at index
         | 
| 227 | 
            +
                #
         | 
| 228 | 
            +
                # @overload slice!(start, length)
         | 
| 229 | 
            +
                #   Removes a range of characters
         | 
| 230 | 
            +
                #
         | 
| 231 | 
            +
                # @overload slice!(range)
         | 
| 232 | 
            +
                #   Removes a range of characters
         | 
| 233 | 
            +
                #
         | 
| 234 | 
            +
                # @return [void]
         | 
| 235 | 
            +
                def slice!(*args)
         | 
| 236 | 
            +
                  document.current_transaction do |tx|
         | 
| 237 | 
            +
                    if args.empty?
         | 
| 238 | 
            +
                      raise ArgumentError,
         | 
| 239 | 
            +
                            "Provide one of `index`, `range`, `start, length` as arguments"
         | 
| 240 | 
            +
                    end
         | 
| 241 | 
            +
             | 
| 242 | 
            +
                    if args.size == 1
         | 
| 243 | 
            +
                      arg = args.first
         | 
| 244 | 
            +
             | 
| 245 | 
            +
                      if arg.is_a?(Range)
         | 
| 246 | 
            +
                        ytext_remove_range(tx, arg.first, arg.last - arg.first)
         | 
| 247 | 
            +
                        return nil
         | 
| 248 | 
            +
                      end
         | 
| 249 | 
            +
             | 
| 250 | 
            +
                      if arg.is_a?(Numeric)
         | 
| 251 | 
            +
                        ytext_remove_range(tx, arg.to_int, 1)
         | 
| 252 | 
            +
                        return nil
         | 
| 253 | 
            +
                      end
         | 
| 254 | 
            +
                    end
         | 
| 255 | 
            +
             | 
| 256 | 
            +
                    if args.size == 2
         | 
| 257 | 
            +
                      start, length = args
         | 
| 258 | 
            +
             | 
| 259 | 
            +
                      if start.is_a?(Numeric) && length.is_a?(Numeric)
         | 
| 260 | 
            +
                        ytext_remove_range(tx, start, length)
         | 
| 261 | 
            +
                        return nil
         | 
| 262 | 
            +
                      end
         | 
| 263 | 
            +
                    end
         | 
| 264 | 
            +
             | 
| 265 | 
            +
                    raise ArgumentError, "Please check your arguments, can't slice."
         | 
| 266 | 
            +
                  end
         | 
| 267 | 
            +
                end
         | 
| 268 | 
            +
             | 
| 269 | 
            +
                # rubocop:enable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength
         | 
| 270 | 
            +
             | 
| 271 | 
            +
                # Returns string representation of text
         | 
| 272 | 
            +
                #
         | 
| 273 | 
            +
                # @example
         | 
| 274 | 
            +
                #   doc = Y::Doc.new
         | 
| 275 | 
            +
                #   text = doc.get_text("my text")
         | 
| 276 | 
            +
                #   text << "Hello"
         | 
| 277 | 
            +
                #
         | 
| 278 | 
            +
                #   puts text.to_s # "Hello"
         | 
| 279 | 
            +
                #
         | 
| 280 | 
            +
                # @return [String]
         | 
| 281 | 
            +
                def to_s
         | 
| 282 | 
            +
                  document.current_transaction { |tx| ytext_to_s(tx) }
         | 
| 283 | 
            +
                end
         | 
| 284 | 
            +
             | 
| 285 | 
            +
                private
         | 
| 286 | 
            +
             | 
| 287 | 
            +
                def can_insert?(value)
         | 
| 288 | 
            +
                  value.is_a?(NilClass) ||
         | 
| 289 | 
            +
                    value.is_a?(Symbol) ||
         | 
| 290 | 
            +
                    [true, false].include?(value) ||
         | 
| 291 | 
            +
                    value.is_a?(Numeric) ||
         | 
| 292 | 
            +
                    value.is_a?(Enumerable) ||
         | 
| 293 | 
            +
                    value.is_a?(Hash)
         | 
| 294 | 
            +
                end
         | 
| 295 | 
            +
             | 
| 296 | 
            +
                # @!method ytext_diff(tx)
         | 
| 297 | 
            +
                #   Returns text changes as list of diffs
         | 
| 298 | 
            +
                #
         | 
| 299 | 
            +
                # @param transaction [Y::Transaction]
         | 
| 300 | 
            +
                # @return [Array<YDiff>]
         | 
| 301 | 
            +
             | 
| 302 | 
            +
                # @!method ytext_insert(tx, index, chunk)
         | 
| 303 | 
            +
                #   Insert into text at position
         | 
| 304 | 
            +
                #
         | 
| 305 | 
            +
                # @param transaction [Y::Transaction]
         | 
| 306 | 
            +
                # @param index [Integer]
         | 
| 307 | 
            +
                # @param chunk [String]
         | 
| 308 | 
            +
                # @return [nil]
         | 
| 309 | 
            +
             | 
| 310 | 
            +
                # @!method ytext_insert_embed(tx, index, content)
         | 
| 311 | 
            +
                #   Insert into text at position
         | 
| 312 | 
            +
                #
         | 
| 313 | 
            +
                # @param tx [Y::Transaction]
         | 
| 314 | 
            +
                # @param index [Integer]
         | 
| 315 | 
            +
                # @param content [Y::Text|Y::Array|Y::Map]
         | 
| 316 | 
            +
                # @return [nil]
         | 
| 317 | 
            +
             | 
| 318 | 
            +
                # @!method ytext_insert_embed_with_attributes(tx, index, embed, attrs)
         | 
| 319 | 
            +
                #   Insert into text at position
         | 
| 320 | 
            +
                #
         | 
| 321 | 
            +
                # @param tx [Y::Transaction]
         | 
| 322 | 
            +
                # @param index [Integer]
         | 
| 323 | 
            +
                # @param embed [Y::Text, Y::Array, Y::Map]
         | 
| 324 | 
            +
                # @param attrs [Hash]
         | 
| 325 | 
            +
                # @return [nil]
         | 
| 326 | 
            +
             | 
| 327 | 
            +
                # @!method ytext_insert_with_attributes(tx, index, chunk, attrs)
         | 
| 328 | 
            +
                #   Insert into text at position
         | 
| 329 | 
            +
                #
         | 
| 330 | 
            +
                # @param tx [Y::Transaction]
         | 
| 331 | 
            +
                # @param index [Integer]
         | 
| 332 | 
            +
                # @param chunk [String]
         | 
| 333 | 
            +
                # @param attrs [Hash]
         | 
| 334 | 
            +
                # @return [nil]
         | 
| 335 | 
            +
             | 
| 336 | 
            +
                # @!method ytext_push(tx, value)
         | 
| 337 | 
            +
                #   Returns length of text
         | 
| 338 | 
            +
                #
         | 
| 339 | 
            +
                # @param tx [Y::Transaction]
         | 
| 340 | 
            +
                # @param value [String]
         | 
| 341 | 
            +
                # @return [nil]
         | 
| 342 | 
            +
             | 
| 343 | 
            +
                # @!method ytext_remove_range(tx, index, length)
         | 
| 344 | 
            +
                #   Removes a range from text
         | 
| 345 | 
            +
                #
         | 
| 346 | 
            +
                # @param tx [Y::Transaction]
         | 
| 347 | 
            +
                # @param index [Integer]
         | 
| 348 | 
            +
                # @param length [Integer]
         | 
| 349 | 
            +
                # @return [nil]
         | 
| 350 | 
            +
             | 
| 351 | 
            +
                # @!method ytext_format(tx, index, length, attrs)
         | 
| 352 | 
            +
                #   Formats a text range
         | 
| 353 | 
            +
                #
         | 
| 354 | 
            +
                # @param tx [Y::Transaction]
         | 
| 355 | 
            +
                # @param index [Integer]
         | 
| 356 | 
            +
                # @param length [Integer]
         | 
| 357 | 
            +
                # @param attrs [Hash]
         | 
| 358 | 
            +
                # @return [nil]
         | 
| 359 | 
            +
             | 
| 360 | 
            +
                # @!method ytext_length(tx)
         | 
| 361 | 
            +
                #   Returns length of text
         | 
| 362 | 
            +
                #
         | 
| 363 | 
            +
                # @param tx [Y::Transaction]
         | 
| 364 | 
            +
                # @return [Integer]
         | 
| 365 | 
            +
             | 
| 366 | 
            +
                # @!method ytext_observe(proc)
         | 
| 367 | 
            +
                #   Observe text changes
         | 
| 368 | 
            +
                #
         | 
| 369 | 
            +
                # @param proc [Proc]
         | 
| 370 | 
            +
                # @return [Integer]
         | 
| 371 | 
            +
             | 
| 372 | 
            +
                # @!method ytext_to_s()
         | 
| 373 | 
            +
                #   Returns string representation of text
         | 
| 374 | 
            +
                #
         | 
| 375 | 
            +
                # @return [String]
         | 
| 376 | 
            +
             | 
| 377 | 
            +
                # @!method ytext_unobserve(subscription_id)
         | 
| 378 | 
            +
                #   Detach listener
         | 
| 379 | 
            +
                #
         | 
| 380 | 
            +
                # @param subscription_id [Integer]
         | 
| 381 | 
            +
                # @return [void]
         | 
| 382 | 
            +
              end
         | 
| 383 | 
            +
            end
         |