gix_merge/blob/builtin_driver/text/mod.rs
1use bstr::BStr;
2use std::num::NonZeroU8;
3
4/// The way the built-in [text driver](crate::blob::BuiltinDriver::Text) will express
5/// merge conflicts in the resulting file.
6#[derive(Default, Copy, Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Hash)]
7pub enum ConflictStyle {
8 /// Only show the zealously minified conflicting lines of the local changes and the incoming (other) changes,
9 /// hiding the base version entirely.
10 ///
11 /// ```text
12 /// line1-changed-by-both
13 /// <<<<<<< local
14 /// line2-to-be-changed-in-incoming
15 /// =======
16 /// line2-changed
17 /// >>>>>>> incoming
18 /// ```
19 #[default]
20 Merge,
21 /// Show non-minimized hunks of local changes, the base, and the incoming (other) changes.
22 ///
23 /// This mode does not hide any information.
24 ///
25 /// ```text
26 /// <<<<<<< local
27 /// line1-changed-by-both
28 /// line2-to-be-changed-in-incoming
29 /// ||||||| 9a8d80c
30 /// line1-to-be-changed-by-both
31 /// line2-to-be-changed-in-incoming
32 /// =======
33 /// line1-changed-by-both
34 /// line2-changed
35 /// >>>>>>> incoming
36 /// ```
37 Diff3,
38 /// Like [`Diff3](Self::Diff3), but will show *minimized* hunks of local change and the incoming (other) changes,
39 /// as well as non-minimized hunks of the base.
40 ///
41 /// ```text
42 /// line1-changed-by-both
43 /// <<<<<<< local
44 /// line2-to-be-changed-in-incoming
45 /// ||||||| 9a8d80c
46 /// line1-to-be-changed-by-both
47 /// line2-to-be-changed-in-incoming
48 /// =======
49 /// line2-changed
50 /// >>>>>>> incoming
51 /// ```
52 ZealousDiff3,
53}
54
55/// The set of labels to annotate conflict markers with.
56///
57/// That way it becomes clearer where the content of conflicts are originating from.
58#[derive(Default, Copy, Clone, Debug, Eq, PartialEq)]
59pub struct Labels<'a> {
60 /// The label for the common *ancestor*.
61 pub ancestor: Option<&'a BStr>,
62 /// The label for the *current* (or *our*) side.
63 pub current: Option<&'a BStr>,
64 /// The label for the *other* (or *their*) side.
65 pub other: Option<&'a BStr>,
66}
67
68/// Options for the builtin [text driver](crate::blob::BuiltinDriver::Text).
69#[derive(Copy, Clone, Debug, Eq, PartialEq)]
70pub struct Options {
71 /// Determine of the diff will be performed.
72 /// Defaults to [`imara_diff::Algorithm::Myers`].
73 pub diff_algorithm: imara_diff::Algorithm,
74 /// Decide what to do to automatically resolve conflicts, or to keep them.
75 pub conflict: Conflict,
76}
77
78impl Default for Options {
79 fn default() -> Self {
80 Options {
81 conflict: Default::default(),
82 diff_algorithm: imara_diff::Algorithm::Myers,
83 }
84 }
85}
86
87/// What to do to resolve a conflict.
88#[derive(Copy, Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Hash)]
89pub enum Conflict {
90 /// Keep the conflict by marking it in the source file.
91 Keep {
92 /// How to visualize conflicts in merged files.
93 style: ConflictStyle,
94 /// The amount of markers to draw, defaults to 7, i.e. `<<<<<<<`
95 marker_size: NonZeroU8,
96 },
97 /// Chose our side to resolve a conflict.
98 ResolveWithOurs,
99 /// Chose their side to resolve a conflict.
100 ResolveWithTheirs,
101 /// Place our and their lines one after another, in any order
102 ResolveWithUnion,
103}
104
105impl Conflict {
106 /// The amount of conflict marker characters to print by default.
107 // TODO: use NonZeroU8::new().unwrap() here once the MSRV supports it.
108 pub const DEFAULT_MARKER_SIZE: u8 = 7;
109
110 /// The amount of conflict markers to print if this instance contains them, or `None` otherwise
111 pub fn marker_size(&self) -> Option<u8> {
112 match self {
113 Conflict::Keep { marker_size, .. } => Some(marker_size.get()),
114 Conflict::ResolveWithOurs | Conflict::ResolveWithTheirs | Conflict::ResolveWithUnion => None,
115 }
116 }
117}
118
119impl Default for Conflict {
120 fn default() -> Self {
121 Conflict::Keep {
122 style: Default::default(),
123 marker_size: Conflict::DEFAULT_MARKER_SIZE.try_into().unwrap(),
124 }
125 }
126}
127
128pub(super) mod function;
129mod utils;