diagram 0.3.0 → 0.3.2

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.
Files changed (35) hide show
  1. checksums.yaml +4 -4
  2. data/lib/diagram.rb +3 -0
  3. data/lib/diagrams/base.rb +6 -8
  4. data/lib/diagrams/class_diagram.rb +0 -4
  5. data/lib/diagrams/elements/class_entity.rb +0 -4
  6. data/lib/diagrams/elements/edge.rb +0 -4
  7. data/lib/diagrams/elements/event.rb +0 -4
  8. data/lib/diagrams/elements/git_branch.rb +27 -0
  9. data/lib/diagrams/elements/git_commit.rb +36 -0
  10. data/lib/diagrams/elements/node.rb +0 -8
  11. data/lib/diagrams/elements/relationship.rb +0 -4
  12. data/lib/diagrams/elements/slice.rb +0 -4
  13. data/lib/diagrams/elements/state.rb +0 -4
  14. data/lib/diagrams/elements/task.rb +0 -4
  15. data/lib/diagrams/elements/timeline_event.rb +21 -0
  16. data/lib/diagrams/elements/timeline_period.rb +23 -0
  17. data/lib/diagrams/elements/timeline_section.rb +23 -0
  18. data/lib/diagrams/elements/transition.rb +0 -4
  19. data/lib/diagrams/elements.rb +12 -0
  20. data/lib/diagrams/flowchart_diagram.rb +0 -6
  21. data/lib/diagrams/gantt_diagram.rb +0 -3
  22. data/lib/diagrams/gitgraph_diagram.rb +345 -0
  23. data/lib/diagrams/pie_diagram.rb +0 -3
  24. data/lib/diagrams/state_diagram.rb +0 -5
  25. data/lib/diagrams/timeline_diagram.rb +161 -0
  26. data/lib/diagrams/version.rb +1 -1
  27. data/lib/diagrams.rb +6 -1
  28. data/sig/diagrams/elements/git_branch.rbs +19 -0
  29. data/sig/diagrams/elements/git_commit.rbs +23 -0
  30. data/sig/diagrams/elements/timeline_event.rbs +17 -0
  31. data/sig/diagrams/elements/timeline_period.rbs +18 -0
  32. data/sig/diagrams/elements/timeline_section.rbs +18 -0
  33. data/sig/diagrams/gitgraph_diagram.rbs +35 -0
  34. data/sig/diagrams/timeline_diagram.rbs +33 -0
  35. metadata +18 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b4af5a1be5db0b92fbdd6a3c6d41eec3d78dd3bd36c6ba6bb70897f552bfdf27
4
- data.tar.gz: 967838b7e5d58ee6ffa188cfc623a130fdb0223e586118c3e11ea1c66fff16cc
3
+ metadata.gz: b6a7329e68d37cab7fe076fb763b56fddff42b23e81619bd70f3a89ef82ebff8
4
+ data.tar.gz: 3eac3513c7ebb76307967f12f3c9d189e35022203ce817e3fbc90be310ac7b10
5
5
  SHA512:
6
- metadata.gz: 90b0513c6e57596f3c769f01a693b1615cbd9044f8dfc2d97a17432073b9f5415b23b0724c91f121219b767f892d4977eac598e33159d4ca83cd892bf00235f3
7
- data.tar.gz: 1dac456dbecc729f32da2a863bd0605e9ee4b830982920b56c54b68dae4ce27ad635e0ad68bd6fc5029838a5f994553da9592dd09390301f465b1f4ceea09ca9
6
+ metadata.gz: 0b59a02a496318c080f8ff72bbfb6bb224ae4dfeb3657eda3e262ac401cca6871f8a18a487fbe012699c7016ae8ef07cfc565f224e7697cc3a6aba9abb60907b
7
+ data.tar.gz: 6cee2aa87b44ae4aab58a99fcf9cfadfbde2f9496523d7e02e5dcb00f6c4ec14abce8b36ff79c1b276e44eec407fe09db6dcaabd9a14c83a050b7684df854f41
data/lib/diagram.rb ADDED
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'diagrams'
data/lib/diagrams/base.rb CHANGED
@@ -1,9 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'digest'
4
- require 'json'
5
- require 'dry-equalizer'
6
-
7
3
  module Diagrams
8
4
  # Abstract base class for all diagram types.
9
5
  # Provides common functionality like versioning, checksum calculation,
@@ -69,12 +65,14 @@ module Diagrams
69
65
  self_collection = self_elements[type] || []
70
66
  other_collection = other_elements[type] || []
71
67
 
72
- # Determine identifier method (prefer id, then name, fallback to object itself)
73
- identifier_method = if self_collection.first.respond_to?(:id)
68
+ # Determine identifier method (prefer id, then name, then title, then label, fallback to object itself)
69
+ identifier_method = if self_collection.first&.respond_to?(:id)
74
70
  :id
75
- elsif self_collection.first.respond_to?(:name)
71
+ elsif self_collection.first&.respond_to?(:name)
76
72
  :name
77
- elsif self_collection.first.respond_to?(:label) # For Slice
73
+ elsif self_collection.first&.respond_to?(:title) # For TimelineSection
74
+ :title
75
+ elsif self_collection.first&.respond_to?(:label) # For Slice, TimelinePeriod
78
76
  :label
79
77
  else
80
78
  :itself # Fallback to object identity/equality
@@ -1,9 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/class_entity'
5
- require_relative 'elements/relationship'
6
-
7
3
  module Diagrams
8
4
  # Represents a UML Class Diagram consisting of classes and relationships between them.
9
5
  class ClassDiagram < Base
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a class entity in a UML Class Diagram.
10
6
  class ClassEntity < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents an edge or link between two nodes in a diagram.
10
6
  # Typically connects nodes via their IDs and can have an optional label.
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents an event, potentially used in State Diagrams or others.
10
6
  class Event < Dry::Struct
@@ -0,0 +1,27 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a branch in a Gitgraph diagram.
6
+ class GitBranch < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :name, Types::Strict::String.constrained(min_size: 1)
10
+ # head_commit_id can be nil initially if the branch is created before any commits
11
+ attribute :head_commit_id, Types::Strict::String.optional.default(nil)
12
+ attribute :start_commit_id, Types::Strict::String.constrained(min_size: 1)
13
+
14
+ # Returns a hash representation suitable for serialization.
15
+ #
16
+ # @return [Hash{Symbol => String}]
17
+ def to_h
18
+ hash = {
19
+ name:,
20
+ start_commit_id:
21
+ }
22
+ hash[:head_commit_id] = head_commit_id if head_commit_id
23
+ hash
24
+ end
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,36 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a commit in a Gitgraph diagram.
6
+ class GitCommit < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :id, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :parent_ids, Types::Strict::Array.of(Types::Strict::String).default([].freeze)
11
+ attribute :branch_name, Types::Strict::String.constrained(min_size: 1)
12
+ attribute :message, Types::Strict::String.optional.default(nil)
13
+ attribute :tag, Types::Strict::String.optional.default(nil)
14
+ attribute :type, Types::Strict::Symbol.default(:NORMAL).enum(:NORMAL, :REVERSE, :HIGHLIGHT, :MERGE, :CHERRY_PICK)
15
+ attribute :cherry_pick_source_id, Types::Strict::String.optional.default(nil)
16
+
17
+ # Returns a hash representation suitable for serialization.
18
+ # Dry::Struct provides to_h, but explicit definition ensures desired keys/structure.
19
+ # Optional attributes are included only if they have non-nil values.
20
+ #
21
+ # @return [Hash{Symbol => String | Array<String> | Symbol}]
22
+ def to_h
23
+ hash = {
24
+ id:,
25
+ parent_ids:,
26
+ branch_name:,
27
+ type:
28
+ }
29
+ hash[:message] = message if message
30
+ hash[:tag] = tag if tag
31
+ hash[:cherry_pick_source_id] = cherry_pick_source_id if cherry_pick_source_id
32
+ hash
33
+ end
34
+ end
35
+ end
36
+ end
@@ -1,17 +1,9 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require 'dry-types'
5
-
6
3
  module Diagrams
7
4
  # Namespace for diagram element value objects.
8
5
 
9
6
  module Elements
10
- # Common Dry::Types definitions for elements.
11
- module Types
12
- include Dry.Types()
13
- end
14
-
15
7
  # Represents a node in various diagram types (e.g., Flowchart).
16
8
  # Typically has an identifier and a display label.
17
9
  class Node < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a relationship (e.g., association, inheritance) between two classes
10
6
  # in a UML Class Diagram.
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a slice in a Pie Diagram.
10
6
  class Slice < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a state in a State Diagram.
10
6
  class State < Dry::Struct
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a task in a Gantt Diagram.
10
6
  class Task < Dry::Struct
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a single event description within a timeline period.
6
+ class TimelineEvent < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :description, Types::Strict::String.constrained(min_size: 1)
10
+
11
+ # Returns a hash representation suitable for serialization.
12
+ #
13
+ # @return [Hash{Symbol => String}]
14
+ def to_h
15
+ {
16
+ description:
17
+ }
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a specific time period on the timeline, containing one or more events.
6
+ class TimelinePeriod < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :label, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :events, Types::Strict::Array.of(TimelineEvent).constrained(min_size: 1)
11
+
12
+ # Returns a hash representation suitable for serialization.
13
+ #
14
+ # @return [Hash{Symbol => String | Array<Hash>}]
15
+ def to_h
16
+ {
17
+ label:,
18
+ events: events.map(&:to_h)
19
+ }
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ module Elements
5
+ # Represents a section or age within the timeline, grouping multiple time periods.
6
+ class TimelineSection < Dry::Struct
7
+ include Elements::Types
8
+
9
+ attribute :title, Types::Strict::String.constrained(min_size: 1)
10
+ attribute :periods, Types::Strict::Array.of(TimelinePeriod).default([].freeze)
11
+
12
+ # Returns a hash representation suitable for serialization.
13
+ #
14
+ # @return [Hash{Symbol => String | Array<Hash>}]
15
+ def to_h
16
+ {
17
+ title:,
18
+ periods: periods.map(&:to_h)
19
+ }
20
+ end
21
+ end
22
+ end
23
+ end
@@ -1,10 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'dry-struct'
4
- require_relative 'node' # Load Types module defined in node.rb
5
-
6
3
  module Diagrams
7
-
8
4
  module Elements
9
5
  # Represents a transition between two states in a State Diagram.
10
6
  class Transition < Dry::Struct
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ # Namespace for diagram element value objects.
5
+
6
+ module Elements
7
+ # Common Dry::Types definitions for elements.
8
+ module Types
9
+ include Dry.Types()
10
+ end
11
+ end
12
+ end
@@ -1,9 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/node'
5
- require_relative 'elements/edge'
6
-
7
3
  module Diagrams
8
4
  # Represents a flowchart diagram consisting of nodes and edges connecting them.
9
5
  class FlowchartDiagram < Base
@@ -28,7 +24,6 @@ module Diagrams
28
24
  # @raise [ArgumentError] if a node with the same ID already exists.
29
25
  # @return [Element::Node] The added node.
30
26
  def add_node(node)
31
-
32
27
  raise ArgumentError, 'Node must be a Diagrams::Element::Node' unless node.is_a?(Diagrams::Elements::Node)
33
28
  raise ArgumentError, "Node with ID '#{node.id}' already exists" if find_node(node.id)
34
29
 
@@ -43,7 +38,6 @@ module Diagrams
43
38
  # @raise [ArgumentError] if the edge refers to non-existent node IDs.
44
39
  # @return [Element::Edge] The added edge.
45
40
  def add_edge(edge)
46
-
47
41
  raise ArgumentError, 'Edge must be a Diagrams::Element::Edge' unless edge.is_a?(Diagrams::Elements::Edge)
48
42
  unless find_node(edge.source_id) && find_node(edge.target_id)
49
43
  raise ArgumentError, "Edge refers to non-existent node IDs ('#{edge.source_id}' or '#{edge.target_id}')"
@@ -1,8 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/task'
5
-
6
3
  module Diagrams
7
4
  # Represents a Gantt Chart diagram consisting of tasks over time.
8
5
  class GanttDiagram < Base
@@ -0,0 +1,345 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'digest' # For generating default commit IDs if needed
4
+
5
+ module Diagrams
6
+ # Represents a Gitgraph diagram, tracking commits, branches, and their relationships.
7
+ class GitgraphDiagram < Base
8
+ attr_reader :commits, :branches, :commit_order, :current_branch_name
9
+
10
+ # Initializes a new GitgraphDiagram.
11
+ # Starts with a 'master' branch by default.
12
+ #
13
+ # @param version [String, Integer, nil] User-defined version identifier.
14
+ def initialize(version: 1)
15
+ super
16
+ @commits = {} # Hash { commit_id => GitCommit }
17
+ @branches = {} # Hash { branch_name => GitBranch }
18
+ @commit_order = [] # Array<String> - IDs of commits in order of creation/operation
19
+ @current_branch_name = 'master'
20
+
21
+ # Initialize main branch conceptually. Its start/head commit will be set by the first commit.
22
+ # We need a placeholder start_commit_id; using a special value or handling nil in GitBranch might be better.
23
+ # For now, let's use a placeholder that signifies it's the root.
24
+ # A better approach might be to create the branch *during* the first commit. Let's refine this.
25
+ # --> Refinement: Don't create the branch object here. Create it during the first 'commit' or 'branch' operation.
26
+ # Initialize @current_branch_name = 'master' conceptually.
27
+
28
+ update_checksum! # Initial checksum for an empty graph
29
+ end
30
+
31
+ # --- Git Operations ---
32
+
33
+ # Adds a commit to the current branch.
34
+ # Handles the creation of the initial 'master' branch on the first commit.
35
+ #
36
+ # @param id [String, nil] Optional custom ID for the commit. Auto-generated if nil.
37
+ # @param message [String, nil] Optional commit message.
38
+ # @param tag [String, nil] Optional tag for the commit.
39
+ # @param type [Symbol] Type of the commit (:NORMAL, :REVERSE, :HIGHLIGHT).
40
+ # @raise [ArgumentError] if a commit with the given ID already exists.
41
+ # @return [Elements::GitCommit] The created commit object.
42
+ def commit(id: nil, message: nil, tag: nil, type: :NORMAL)
43
+ parent_id = current_head_commit_id
44
+ parent_ids = parent_id ? [parent_id] : []
45
+
46
+ commit_id = id || generate_commit_id(parent_ids, message)
47
+ raise ArgumentError, "Commit with ID '#{commit_id}' already exists" if @commits.key?(commit_id)
48
+
49
+ # Handle first commit: create the master branch
50
+ if @commits.empty? && @current_branch_name == 'master' && !@branches.key?('master')
51
+ # The first commit *is* the starting point of the master branch
52
+ master_branch = Elements::GitBranch.new(name: 'master', start_commit_id: commit_id, head_commit_id: commit_id)
53
+ @branches['master'] = master_branch
54
+ elsif !@branches.key?(@current_branch_name)
55
+ # This case shouldn't typically happen if branch/checkout is used correctly,
56
+ # but defensively handle committing to a non-existent branch (other than initial master).
57
+ raise ArgumentError, "Cannot commit: Branch '#{@current_branch_name}' does not exist."
58
+ end
59
+
60
+ new_commit = Elements::GitCommit.new(
61
+ id: commit_id,
62
+ parent_ids:,
63
+ branch_name: @current_branch_name,
64
+ message:,
65
+ tag:,
66
+ type:
67
+ )
68
+
69
+ @commits[commit_id] = new_commit
70
+ @commit_order << commit_id
71
+
72
+ # Update the head of the current branch
73
+ current_branch = @branches[@current_branch_name]
74
+ current_branch.attributes[:head_commit_id] = commit_id # Update using Dry::Struct's way if needed, direct assign might work
75
+
76
+ update_checksum!
77
+ new_commit
78
+ end
79
+
80
+ # Creates a new branch pointing to a specific commit (or the current head)
81
+ # and switches the current context to the new branch.
82
+ #
83
+ # @param name [String] The name for the new branch.
84
+ # @param start_commit_id [String, nil] Optional ID of the commit where the branch should start.
85
+ # Defaults to the head commit of the current branch.
86
+ # @raise [ArgumentError] if the branch name already exists or if trying to branch before any commits exist.
87
+ # @raise [ArgumentError] if a specified `start_commit_id` does not exist.
88
+ # @return [Elements::GitBranch] The created branch object.
89
+ def branch(name:, start_commit_id: nil)
90
+ raise ArgumentError, "Branch name '#{name}' already exists" if @branches.key?(name)
91
+
92
+ effective_start_commit_id = start_commit_id || current_head_commit_id
93
+
94
+ # Ensure there's a commit to branch from
95
+ raise ArgumentError, 'Cannot create a branch before the first commit' unless effective_start_commit_id
96
+
97
+ unless @commits.key?(effective_start_commit_id)
98
+ raise ArgumentError,
99
+ "Start commit ID '#{effective_start_commit_id}' does not exist"
100
+ end
101
+
102
+ new_branch = Elements::GitBranch.new(
103
+ name:,
104
+ # The new branch initially points to the commit it was created from
105
+ start_commit_id: effective_start_commit_id,
106
+ head_commit_id: effective_start_commit_id
107
+ )
108
+
109
+ @branches[name] = new_branch
110
+ @current_branch_name = name # Switch to the new branch
111
+
112
+ update_checksum!
113
+ new_branch
114
+ end
115
+
116
+ # Switches the current context to an existing branch.
117
+ #
118
+ # @param name [String] The name of the branch to switch to.
119
+ # @raise [ArgumentError] if the branch name does not exist.
120
+ # @return [String] The name of the branch checked out.
121
+ def checkout(name:)
122
+ raise ArgumentError, "Branch '#{name}' does not exist. Cannot checkout." unless @branches.key?(name)
123
+
124
+ @current_branch_name = name
125
+ # NOTE: Checkout does not change the diagram structure itself (commits/branches),
126
+ # so we do NOT update the checksum here.
127
+ name
128
+ end
129
+
130
+ # Merges the head of a specified branch into the current branch.
131
+ # Creates a merge commit on the current branch.
132
+ #
133
+ # @param from_branch_name [String] The name of the branch to merge from.
134
+ # @param id [String, nil] Optional custom ID for the merge commit. Auto-generated if nil.
135
+ # @param tag [String, nil] Optional tag for the merge commit.
136
+ # @param type [Symbol] Type of the merge commit (defaults to :MERGE, can be overridden e.g., :REVERSE).
137
+ # @raise [ArgumentError] if `from_branch_name` does not exist, is the same as the current branch,
138
+ # or if either branch has no commits.
139
+ # @raise [ArgumentError] if a commit with the given ID already exists.
140
+ # @return [Elements::GitCommit] The created merge commit object.
141
+ def merge(from_branch_name:, id: nil, tag: nil, type: :MERGE)
142
+ if from_branch_name == @current_branch_name
143
+ raise ArgumentError,
144
+ "Cannot merge branch '#{from_branch_name}' into itself"
145
+ end
146
+ unless @branches.key?(from_branch_name)
147
+ raise ArgumentError,
148
+ "Branch '#{from_branch_name}' does not exist. Cannot merge."
149
+ end
150
+ unless @branches.key?(@current_branch_name)
151
+ raise ArgumentError, "Current branch '#{@current_branch_name}' does not exist. Cannot merge."
152
+ end
153
+
154
+ target_branch = @branches[@current_branch_name]
155
+ source_branch = @branches[from_branch_name]
156
+
157
+ target_head_id = target_branch.head_commit_id
158
+ source_head_id = source_branch.head_commit_id
159
+
160
+ unless target_head_id
161
+ raise ArgumentError,
162
+ "Current branch '#{@current_branch_name}' has no commits to merge into."
163
+ end
164
+ raise ArgumentError, "Source branch '#{from_branch_name}' has no commits to merge from." unless source_head_id
165
+
166
+ # Merge commit parents are the heads of the two branches being merged
167
+ parent_ids = [target_head_id, source_head_id].sort # Sort for consistent checksumming/comparison
168
+
169
+ merge_commit_id = id || generate_commit_id(parent_ids,
170
+ "Merge branch '#{from_branch_name}' into #{@current_branch_name}")
171
+ raise ArgumentError, "Commit with ID '#{merge_commit_id}' already exists" if @commits.key?(merge_commit_id)
172
+
173
+ merge_commit = Elements::GitCommit.new(
174
+ id: merge_commit_id,
175
+ parent_ids:,
176
+ branch_name: @current_branch_name, # Merge commit belongs to the target branch
177
+ message: "Merge branch '#{from_branch_name}' into #{@current_branch_name}", # Default message
178
+ tag:,
179
+ type: # Use provided type, default :MERGE
180
+ )
181
+
182
+ @commits[merge_commit_id] = merge_commit
183
+ @commit_order << merge_commit_id
184
+
185
+ # Update the head of the current (target) branch
186
+ target_branch.attributes[:head_commit_id] = merge_commit_id
187
+
188
+ update_checksum!
189
+ merge_commit
190
+ end
191
+
192
+ # Cherry-picks an existing commit onto the current branch.
193
+ # Creates a new commit on the current branch that mirrors the specified commit.
194
+ #
195
+ # @param commit_id [String] The ID of the commit to cherry-pick.
196
+ # @param parent_override_id [String, nil] Optional: If cherry-picking a merge commit, specifies which parent lineage to follow.
197
+ # (Note: Basic implementation might ignore this for simplicity initially).
198
+ # @raise [ArgumentError] if the commit_id does not exist, is already on the current branch,
199
+ # or if the current branch has no commits.
200
+ # @return [Elements::GitCommit] The created cherry-pick commit object.
201
+ # Basic implementation ignores parent_override_id for now
202
+ def cherry_pick(commit_id:, parent_override_id: nil)
203
+ unless @commits.key?(commit_id)
204
+ raise ArgumentError,
205
+ "Commit with ID '#{commit_id}' does not exist. Cannot cherry-pick."
206
+ end
207
+
208
+ source_commit = @commits[commit_id]
209
+ current_branch_head_id = current_head_commit_id
210
+
211
+ unless current_branch_head_id
212
+ raise ArgumentError,
213
+ "Current branch '#{@current_branch_name}' has no commits. Cannot cherry-pick onto it."
214
+ end
215
+ if source_commit.branch_name == @current_branch_name
216
+ raise ArgumentError,
217
+ "Commit '#{commit_id}' is already on the current branch '#{@current_branch_name}'. Cannot cherry-pick."
218
+ end
219
+
220
+ # More robust check: walk history? For now, simple branch name check.
221
+
222
+ # TODO: Handle cherry-picking merge commits and parent_override_id if needed later.
223
+ if source_commit.parent_ids.length > 1 && !parent_override_id
224
+ warn "Cherry-picking a merge commit (#{commit_id}) without specifying a parent override is ambiguous. Picking first parent lineage by default."
225
+ # Or raise ArgumentError: "Cherry-picking a merge commit requires specifying parent_override_id."
226
+ end
227
+
228
+ parent_ids = [current_branch_head_id] # Cherry-pick commit's parent is the current head
229
+ new_commit_id = generate_commit_id(parent_ids, "Cherry-pick: #{source_commit.message || source_commit.id}")
230
+ if @commits.key?(new_commit_id)
231
+ raise ArgumentError,
232
+ "Generated commit ID '#{new_commit_id}' conflicts with existing commit."
233
+ end
234
+
235
+ cherry_pick_commit = Elements::GitCommit.new(
236
+ id: new_commit_id,
237
+ parent_ids:,
238
+ branch_name: @current_branch_name,
239
+ message: source_commit.message || "Cherry-pick of #{source_commit.id}", # Copy message or use default
240
+ tag: nil, # Cherry-picks usually don't copy tags directly
241
+ type: :CHERRY_PICK,
242
+ cherry_pick_source_id: commit_id # Link back to the original commit
243
+ )
244
+
245
+ @commits[new_commit_id] = cherry_pick_commit
246
+ @commit_order << new_commit_id
247
+
248
+ # Update the head of the current branch
249
+ current_branch = @branches[@current_branch_name]
250
+ current_branch.attributes[:head_commit_id] = new_commit_id
251
+
252
+ update_checksum!
253
+ cherry_pick_commit
254
+ end
255
+
256
+ # --- Base Class Implementation ---
257
+
258
+ # Returns the specific content of the gitgraph diagram as a hash.
259
+ # @return [Hash]
260
+ def to_h_content
261
+ {
262
+ commits: @commits.values.map(&:to_h),
263
+ branches: @branches.values.map(&:to_h),
264
+ commit_order: @commit_order,
265
+ current_branch_name: @current_branch_name # Useful for resuming state? Maybe not needed in content hash.
266
+ # Consider if current_branch_name should be part of the checksummable content.
267
+ # For now, let's include it for potential deserialization needs.
268
+ }
269
+ end
270
+
271
+ # Returns a hash mapping element types to their collections for diffing.
272
+ # @return [Hash{Symbol => Array<Elements::GitCommit | Elements::GitBranch>}]
273
+ def identifiable_elements
274
+ {
275
+ commits: @commits.values,
276
+ branches: @branches.values
277
+ }
278
+ end
279
+
280
+ # Class method to create a GitgraphDiagram from a hash.
281
+ # @param data_hash [Hash] Hash containing diagram data.
282
+ # @param version [String, Integer, nil] Diagram version.
283
+ # @param checksum [String, nil] Expected checksum (optional).
284
+ # @return [GitgraphDiagram] The instantiated diagram.
285
+ def self.from_h(data_hash, version:, checksum:)
286
+ diagram = new(version:)
287
+
288
+ # Restore commits
289
+ commits_data = data_hash[:commits] || data_hash['commits'] || []
290
+ commits_data.each do |commit_h|
291
+ # Convert type back to symbol if it's a string
292
+ commit_data = commit_h.transform_keys(&:to_sym)
293
+ commit_data[:type] = commit_data[:type].to_sym if commit_data[:type].is_a?(String)
294
+ commit = Elements::GitCommit.new(commit_data)
295
+ diagram.commits[commit.id] = commit
296
+ end
297
+
298
+ # Restore branches
299
+ branches_data = data_hash[:branches] || data_hash['branches'] || []
300
+ branches_data.each do |branch_h|
301
+ branch = Elements::GitBranch.new(branch_h.transform_keys(&:to_sym))
302
+ diagram.branches[branch.name] = branch
303
+ end
304
+
305
+ # Restore commit order
306
+ diagram.instance_variable_set(:@commit_order, data_hash[:commit_order] || data_hash['commit_order'] || [])
307
+
308
+ # Restore current branch name
309
+ diagram.instance_variable_set(:@current_branch_name,
310
+ data_hash[:current_branch_name] || data_hash['current_branch_name'] || 'master')
311
+
312
+ # Recalculate checksum after loading all data
313
+ diagram.send(:update_checksum!) # Use send to call protected method from class scope
314
+
315
+ # Optional: Verify checksum if provided
316
+ if checksum && diagram.checksum != checksum
317
+ warn "Checksum mismatch for loaded GitgraphDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
318
+ end
319
+
320
+ diagram
321
+ end
322
+
323
+ private
324
+
325
+ # Generates a unique ID for a commit if one isn't provided.
326
+ # Placeholder - could use SHA1/SHA256 of content or simple counter.
327
+ # Using a simple counter based on commit order for now.
328
+ def generate_commit_id(parent_ids, message)
329
+ # Simple approach: use commit count + part of parent hash if available
330
+ base = @commit_order.size.to_s
331
+ parent_part = parent_ids.first ? parent_ids.first[0..5] : 'root'
332
+ # NOTE: This is NOT cryptographically secure or git-like. Just for basic uniqueness.
333
+ "commit-#{base}-#{parent_part}-#{Digest::SHA1.hexdigest(message || Time.now.to_s)[0..5]}"
334
+ end
335
+
336
+ # Helper to get the head commit ID of the current branch.
337
+ def current_head_commit_id
338
+ current_branch = @branches[@current_branch_name]
339
+ current_branch&.head_commit_id # Returns nil if branch doesn't exist yet
340
+ end
341
+
342
+ # Protected method access for from_h
343
+ protected :update_checksum!
344
+ end
345
+ end
@@ -1,8 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/slice'
5
-
6
3
  module Diagrams
7
4
  # Represents a Pie Chart diagram consisting of slices.
8
5
  class PieDiagram < Base
@@ -1,10 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative 'base'
4
- require_relative 'elements/state'
5
- require_relative 'elements/transition'
6
- require_relative 'elements/event' # Assuming events are still desired
7
-
8
3
  module Diagrams
9
4
  # Represents a State Diagram consisting of states and transitions between them.
10
5
  class StateDiagram < Base
@@ -0,0 +1,161 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Diagrams
4
+ # Represents a timeline diagram illustrating a chronology of events.
5
+ class TimelineDiagram < Base
6
+ DEFAULT_SECTION_TITLE = 'Default Section'
7
+
8
+ attr_reader :title, :sections
9
+
10
+ # Initializes a new TimelineDiagram.
11
+ #
12
+ # @param title [String, nil] Optional title for the timeline.
13
+ # @param sections [Array<Element::TimelineSection>] Initial sections (optional).
14
+ # @param version [String, Integer, nil] User-defined version identifier.
15
+ def initialize(title: nil, sections: [], version: 1)
16
+ super(version:)
17
+ @title = title&.strip
18
+ @sections = sections.is_a?(Array) ? sections : []
19
+ # Ensure there's always at least a default section if none provided initially
20
+ ensure_default_section if @sections.empty?
21
+ update_checksum!
22
+ end
23
+
24
+ # Sets the title of the timeline.
25
+ #
26
+ # @param new_title [String] The title text.
27
+ # @return [String] The new title.
28
+ def set_title(new_title)
29
+ @title = new_title.strip
30
+ update_checksum!
31
+ @title
32
+ end
33
+
34
+ # Adds a new section to the timeline.
35
+ # Subsequent periods/events will be added to this section.
36
+ #
37
+ # @param section_title [String] The title of the section.
38
+ # @raise [ArgumentError] if a section with the same title already exists.
39
+ # @return [Elements::TimelineSection] The newly added section.
40
+ def add_section(section_title)
41
+ clean_title = section_title.strip
42
+ raise ArgumentError, "Section title '#{clean_title}' cannot be empty" if clean_title.empty?
43
+ if find_section(clean_title)
44
+ raise ArgumentError, "Section with title '#{clean_title}' already exists"
45
+ end
46
+
47
+ # Remove default section if it's empty and we're adding a real one
48
+ if @sections.size == 1 && @sections.first.title == DEFAULT_SECTION_TITLE && @sections.first.periods.empty?
49
+ @sections.clear
50
+ end
51
+
52
+ new_section = Elements::TimelineSection.new(title: clean_title)
53
+ @sections << new_section
54
+ update_checksum!
55
+ new_section
56
+ end
57
+
58
+ # Adds a time period with one or more events to the current (last) section.
59
+ #
60
+ # @param period_label [String] The label for the time period (e.g., "2004", "Bronze Age").
61
+ # @param events [Array<String> | String] A single event description or an array of event descriptions.
62
+ # @raise [ArgumentError] if period_label or any event description is empty.
63
+ # @raise [StandardError] if no sections exist (shouldn't happen due to default section).
64
+ # @return [Elements::TimelinePeriod] The newly added period.
65
+ def add_period(period_label:, events:)
66
+ clean_label = period_label.strip
67
+ raise ArgumentError, "Period label cannot be empty" if clean_label.empty?
68
+
69
+ event_list = Array(events).map(&:strip).reject(&:empty?)
70
+ raise ArgumentError, "Events cannot be empty" if event_list.empty?
71
+
72
+ timeline_events = event_list.map { |desc| Elements::TimelineEvent.new(description: desc) }
73
+ new_period = Elements::TimelinePeriod.new(label: clean_label, events: timeline_events)
74
+
75
+ current_section = @sections.last
76
+ raise StandardError, "Cannot add period: No section available." unless current_section
77
+
78
+ # Add period to the current section's periods array
79
+ # Dry::Struct arrays are immutable, so we need to create a new section object
80
+ updated_periods = current_section.periods + [new_period]
81
+ # Create a completely new section instance with the updated periods array
82
+ updated_section = Elements::TimelineSection.new(title: current_section.title, periods: updated_periods)
83
+
84
+ # Find the index of the current section and update it in place
85
+ # Rebuild the sections array, replacing the modified section
86
+ current_section_title = current_section.title
87
+ # Rebuild the sections array by mapping, replacing the target section
88
+ @sections = @sections.map do |section|
89
+ section.title == current_section_title ? updated_section : section
90
+ end
91
+
92
+ update_checksum!
93
+ new_period
94
+ end
95
+
96
+ # --- Base Class Implementation ---
97
+
98
+ def to_h_content
99
+ content = {
100
+ sections: @sections.map(&:to_h)
101
+ }
102
+ content[:title] = @title if @title
103
+ content
104
+ end
105
+
106
+ def identifiable_elements
107
+ # Sections and Periods are the main identifiable structures. Events are nested.
108
+ # Use section title and period label as identifiers.
109
+ {
110
+ sections: @sections,
111
+ periods: @sections.flat_map(&:periods) # Flatten periods from all sections
112
+ }
113
+ end
114
+
115
+ def self.from_h(data_hash, version:, checksum:)
116
+ title = data_hash[:title] || data_hash['title']
117
+ sections_data = data_hash[:sections] || data_hash['sections'] || []
118
+
119
+ sections = sections_data.map do |section_h|
120
+ section_data = section_h.transform_keys(&:to_sym)
121
+ periods_data = section_data[:periods] || []
122
+ periods = periods_data.map do |period_h|
123
+ period_data = period_h.transform_keys(&:to_sym)
124
+ events_data = period_data[:events] || []
125
+ events = events_data.map do |event_h|
126
+ event_data = event_h.transform_keys(&:to_sym)
127
+ Elements::TimelineEvent.new(event_data)
128
+ end
129
+ Elements::TimelinePeriod.new(period_data.merge(events:))
130
+ end
131
+ Elements::TimelineSection.new(section_data.merge(periods:))
132
+ end
133
+
134
+ diagram = new(title:, sections:, version:)
135
+
136
+ # Optional: Verify checksum
137
+ if checksum && diagram.checksum != checksum
138
+ warn "Checksum mismatch for loaded TimelineDiagram (version: #{version}). Expected #{checksum}, got #{diagram.checksum}."
139
+ end
140
+
141
+ diagram
142
+ end
143
+
144
+ private
145
+
146
+ # Ensures a default section exists if the sections array is empty.
147
+ def ensure_default_section
148
+ unless @sections.any? { |s| s.title == DEFAULT_SECTION_TITLE }
149
+ @sections << Elements::TimelineSection.new(title: DEFAULT_SECTION_TITLE)
150
+ end
151
+ end
152
+
153
+ # Finds a section by its title.
154
+ def find_section(section_title)
155
+ @sections.find { |s| s.title == section_title }
156
+ end
157
+
158
+ # Protected method access for from_h
159
+ protected :update_checksum!
160
+ end
161
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Diagrams
4
- VERSION = '0.3.0'
4
+ VERSION = '0.3.2'
5
5
  end
data/lib/diagrams.rb CHANGED
@@ -1,9 +1,14 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require 'zeitwerk'
4
- require_relative 'diagrams/version' # Keep this for gemspec access before setup
4
+ require 'digest'
5
+ require 'json'
6
+ require 'dry-equalizer'
7
+ require 'dry-struct'
8
+ require_relative 'diagrams/version'
5
9
 
6
10
  loader = Zeitwerk::Loader.for_gem
11
+ loader.ignore("#{__dir__}/diagram.rb")
7
12
  loader.setup
8
13
 
9
14
  # This module handles diagrams creation and manipulation.
@@ -0,0 +1,19 @@
1
+ module Diagrams
2
+ module Elements
3
+ # Represents a branch in a Gitgraph diagram.
4
+ class GitBranch < ::Dry::Struct
5
+ include Elements::Types
6
+
7
+ # Attributes
8
+ attr_reader name: String
9
+ attr_reader head_commit_id: String?
10
+ attr_reader start_commit_id: String
11
+
12
+ # Methods
13
+ def initialize: (name: String, ?head_commit_id: String?, start_commit_id: String) -> void
14
+ | (Hash[Symbol, untyped]) -> void # Allow hash initialization
15
+
16
+ def to_h: () -> Hash[Symbol, String]
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,23 @@
1
+ module Diagrams
2
+ module Elements
3
+ # Represents a commit in a Gitgraph diagram.
4
+ class GitCommit < ::Dry::Struct
5
+ include Elements::Types
6
+
7
+ # Attributes
8
+ attr_reader id: String
9
+ attr_reader parent_ids: Array[String]
10
+ attr_reader branch_name: String
11
+ attr_reader message: String?
12
+ attr_reader tag: String?
13
+ attr_reader type: Symbol # :NORMAL | :REVERSE | :HIGHLIGHT | :MERGE | :CHERRY_PICK
14
+ attr_reader cherry_pick_source_id: String?
15
+
16
+ # Methods
17
+ def initialize: (id: String, ?parent_ids: Array[String], branch_name: String, ?message: String?, ?tag: String?, ?type: Symbol, ?cherry_pick_source_id: String?) -> void
18
+ | (Hash[Symbol, untyped]) -> void # Allow hash initialization
19
+
20
+ def to_h: () -> Hash[Symbol, String | Array[String] | Symbol]
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,17 @@
1
+ module Diagrams
2
+ module Elements
3
+ # Represents a single event description within a timeline period.
4
+ class TimelineEvent < ::Dry::Struct
5
+ include Elements::Types
6
+
7
+ # Attributes
8
+ attr_reader description: String
9
+
10
+ # Methods
11
+ def initialize: (description: String) -> void
12
+ | (Hash[Symbol, untyped]) -> void # Allow hash initialization
13
+
14
+ def to_h: () -> Hash[Symbol, String]
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,18 @@
1
+ module Diagrams
2
+ module Elements
3
+ # Represents a specific time period on the timeline, containing one or more events.
4
+ class TimelinePeriod < ::Dry::Struct
5
+ include Elements::Types
6
+
7
+ # Attributes
8
+ attr_reader label: String
9
+ attr_reader events: Array[TimelineEvent]
10
+
11
+ # Methods
12
+ def initialize: (label: String, events: Array[TimelineEvent]) -> void
13
+ | (Hash[Symbol, untyped]) -> void # Allow hash initialization
14
+
15
+ def to_h: () -> Hash[Symbol, String | Array[Hash[Symbol, String]]]
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,18 @@
1
+ module Diagrams
2
+ module Elements
3
+ # Represents a section or age within the timeline, grouping multiple time periods.
4
+ class TimelineSection < ::Dry::Struct
5
+ include Elements::Types
6
+
7
+ # Attributes
8
+ attr_reader title: String
9
+ attr_reader periods: Array[TimelinePeriod]
10
+
11
+ # Methods
12
+ def initialize: (title: String, ?periods: Array[TimelinePeriod]) -> void
13
+ | (Hash[Symbol, untyped]) -> void # Allow hash initialization
14
+
15
+ def to_h: () -> Hash[Symbol, String | Array[Hash[Symbol, untyped]]]
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,35 @@
1
+ module Diagrams
2
+ # Represents a Gitgraph diagram, tracking commits, branches, and their relationships.
3
+ class GitgraphDiagram < Base
4
+ # Instance Variables (via attr_reader)
5
+ attr_reader commits: Hash[String, Elements::GitCommit]
6
+ attr_reader branches: Hash[String, Elements::GitBranch]
7
+ attr_reader commit_order: Array[String]
8
+ attr_reader current_branch_name: String
9
+
10
+ # Initialization
11
+ def initialize: (?version: String | Integer) -> void
12
+
13
+ # --- Git Operations ---
14
+ def commit: (?id: String?, ?message: String?, ?tag: String?, ?type: Symbol) -> Elements::GitCommit
15
+ def branch: (name: String, ?start_commit_id: String?) -> Elements::GitBranch
16
+ def checkout: (name: String) -> String
17
+ def merge: (from_branch_name: String, ?id: String?, ?tag: String?, ?type: Symbol) -> Elements::GitCommit
18
+ def cherry_pick: (commit_id: String, ?parent_override_id: String?) -> Elements::GitCommit
19
+
20
+ # --- Base Class Implementation ---
21
+ def to_h_content: () -> Hash[Symbol, untyped] # More specific: Hash[Symbol, Array[Hash[Symbol, untyped]] | Array[String] | String]
22
+ def identifiable_elements: () -> Hash[Symbol, Array[Elements::GitCommit | Elements::GitBranch]]
23
+
24
+ # Class method for deserialization
25
+ def self.from_h: (Hash[Symbol, untyped] data_hash, version: String | Integer | nil, checksum: String?) -> GitgraphDiagram
26
+
27
+ # --- Private/Protected Methods ---
28
+ private
29
+ def generate_commit_id: (Array[String] parent_ids, String? message) -> String
30
+ def current_head_commit_id: () -> String?
31
+
32
+ # Inherited protected method
33
+ # def update_checksum!: () -> String # Defined in Base, but accessible via send in from_h
34
+ end
35
+ end
@@ -0,0 +1,33 @@
1
+ module Diagrams
2
+ # Represents a timeline diagram illustrating a chronology of events.
3
+ class TimelineDiagram < Base
4
+ DEFAULT_SECTION_TITLE: String
5
+
6
+ # Instance Variables (via attr_reader)
7
+ attr_reader title: String?
8
+ attr_reader sections: Array[Elements::TimelineSection]
9
+
10
+ # Initialization
11
+ def initialize: (?title: String?, ?sections: Array[Elements::TimelineSection], ?version: String | Integer) -> void
12
+
13
+ # Public Methods
14
+ def set_title: (String new_title) -> String
15
+ def add_section: (String section_title) -> Elements::TimelineSection
16
+ def add_period: (period_label: String, events: String | Array[String]) -> Elements::TimelinePeriod
17
+
18
+ # --- Base Class Implementation ---
19
+ def to_h_content: () -> Hash[Symbol, untyped] # More specific: Hash[:title?, String | :sections, Array[Hash]]
20
+ def identifiable_elements: () -> Hash[Symbol, Array[Elements::TimelineSection | Elements::TimelinePeriod]]
21
+
22
+ # Class method for deserialization
23
+ def self.from_h: (Hash[Symbol, untyped] data_hash, version: String | Integer | nil, checksum: String?) -> TimelineDiagram
24
+
25
+ # --- Private Methods ---
26
+ private
27
+ def ensure_default_section: () -> void
28
+ def find_section: (String section_title) -> Elements::TimelineSection?
29
+
30
+ # Inherited protected method
31
+ # def update_checksum!: () -> String
32
+ end
33
+ end
metadata CHANGED
@@ -1,13 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: diagram
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.3.0
4
+ version: 0.3.2
5
5
  platform: ruby
6
6
  authors:
7
7
  - Abdelkader Boudih
8
8
  bindir: bin
9
9
  cert_chain: []
10
- date: 2025-03-31 00:00:00.000000000 Z
10
+ date: 2025-04-01 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: dry-equalizer
@@ -212,39 +212,55 @@ executables: []
212
212
  extensions: []
213
213
  extra_rdoc_files: []
214
214
  files:
215
+ - lib/diagram.rb
215
216
  - lib/diagrams.rb
216
217
  - lib/diagrams/base.rb
217
218
  - lib/diagrams/class_diagram.rb
219
+ - lib/diagrams/elements.rb
218
220
  - lib/diagrams/elements/class_entity.rb
219
221
  - lib/diagrams/elements/edge.rb
220
222
  - lib/diagrams/elements/event.rb
223
+ - lib/diagrams/elements/git_branch.rb
224
+ - lib/diagrams/elements/git_commit.rb
221
225
  - lib/diagrams/elements/node.rb
222
226
  - lib/diagrams/elements/relationship.rb
223
227
  - lib/diagrams/elements/slice.rb
224
228
  - lib/diagrams/elements/state.rb
225
229
  - lib/diagrams/elements/task.rb
230
+ - lib/diagrams/elements/timeline_event.rb
231
+ - lib/diagrams/elements/timeline_period.rb
232
+ - lib/diagrams/elements/timeline_section.rb
226
233
  - lib/diagrams/elements/transition.rb
227
234
  - lib/diagrams/flowchart_diagram.rb
228
235
  - lib/diagrams/gantt_diagram.rb
236
+ - lib/diagrams/gitgraph_diagram.rb
229
237
  - lib/diagrams/pie_diagram.rb
230
238
  - lib/diagrams/state_diagram.rb
239
+ - lib/diagrams/timeline_diagram.rb
231
240
  - lib/diagrams/version.rb
232
241
  - sig/diagrams/base.rbs
233
242
  - sig/diagrams/class_diagram.rbs
234
243
  - sig/diagrams/elements/class_entity.rbs
235
244
  - sig/diagrams/elements/edge.rbs
236
245
  - sig/diagrams/elements/event.rbs
246
+ - sig/diagrams/elements/git_branch.rbs
247
+ - sig/diagrams/elements/git_commit.rbs
237
248
  - sig/diagrams/elements/node.rbs
238
249
  - sig/diagrams/elements/relationship.rbs
239
250
  - sig/diagrams/elements/slice.rbs
240
251
  - sig/diagrams/elements/state.rbs
241
252
  - sig/diagrams/elements/task.rbs
253
+ - sig/diagrams/elements/timeline_event.rbs
254
+ - sig/diagrams/elements/timeline_period.rbs
255
+ - sig/diagrams/elements/timeline_section.rbs
242
256
  - sig/diagrams/elements/transition.rbs
243
257
  - sig/diagrams/elements/types.rbs
244
258
  - sig/diagrams/flowchart_diagram.rbs
245
259
  - sig/diagrams/gantt_diagram.rbs
260
+ - sig/diagrams/gitgraph_diagram.rbs
246
261
  - sig/diagrams/pie_diagram.rbs
247
262
  - sig/diagrams/state_diagram.rbs
263
+ - sig/diagrams/timeline_diagram.rbs
248
264
  homepage: https://github.com/seuros/diagram-ruby
249
265
  licenses:
250
266
  - MIT