|  | 
|  | 1 | +package main | 
|  | 2 | + | 
|  | 3 | +import ( | 
|  | 4 | +	"fmt" | 
|  | 5 | +	"strings" | 
|  | 6 | + | 
|  | 7 | +	"github.com/iancoleman/strcase" | 
|  | 8 | +	"github.com/tableauio/loader/cmd/protoc-gen-csharp-tableau-loader/helper" | 
|  | 9 | +	"github.com/tableauio/loader/internal/options" | 
|  | 10 | +	"google.golang.org/protobuf/compiler/protogen" | 
|  | 11 | +	"google.golang.org/protobuf/reflect/protoreflect" | 
|  | 12 | +) | 
|  | 13 | + | 
|  | 14 | +const orderedMapSuffix = "_OrderedMap" | 
|  | 15 | +const orderedMapValueSuffix = "_OrderedMapValue" | 
|  | 16 | + | 
|  | 17 | +func genOrderedMapTypeDef(gen *protogen.Plugin, g *protogen.GeneratedFile, md protoreflect.MessageDescriptor, depth int, keys []helper.MapKey, messagerFullName string) { | 
|  | 18 | +	if depth == 1 && !options.NeedGenOrderedMap(md, options.LangCS) { | 
|  | 19 | +		return | 
|  | 20 | +	} | 
|  | 21 | +	for i := 0; i < md.Fields().Len(); i++ { | 
|  | 22 | +		fd := md.Fields().Get(i) | 
|  | 23 | +		if fd.IsMap() { | 
|  | 24 | +			if depth == 1 { | 
|  | 25 | +				g.P("        // OrderedMap types.") | 
|  | 26 | +			} | 
|  | 27 | +			nextKeys := helper.AddMapKey(gen, fd, keys) | 
|  | 28 | +			keyType := nextKeys[len(nextKeys)-1].Type | 
|  | 29 | + | 
|  | 30 | +			if fd.MapValue().Kind() == protoreflect.MessageKind { | 
|  | 31 | +				genOrderedMapTypeDef(gen, g, fd.MapValue().Message(), depth+1, nextKeys, messagerFullName) | 
|  | 32 | +			} | 
|  | 33 | + | 
|  | 34 | +			prefix := parseOrderedMapPrefix(fd, messagerFullName) | 
|  | 35 | +			orderedMap := prefix + orderedMapSuffix | 
|  | 36 | +			orderedMapValue := prefix + orderedMapValueSuffix | 
|  | 37 | + | 
|  | 38 | +			nextMapFD := getNextLevelMapFD(fd.MapValue()) | 
|  | 39 | +			if nextMapFD != nil { | 
|  | 40 | +				currValueType := helper.ParseCsharpType(fd.MapValue()) | 
|  | 41 | +				nextPrefix := parseOrderedMapPrefix(nextMapFD, messagerFullName) | 
|  | 42 | +				nextOrderedMap := nextPrefix + orderedMapSuffix | 
|  | 43 | +				g.P("        public class ", orderedMapValue, " : Tuple<", nextOrderedMap, ", ", currValueType, "?>") | 
|  | 44 | +				g.P("        {") | 
|  | 45 | +				g.P("            public ", orderedMapValue, "(", nextOrderedMap, " item1, ", currValueType, "? item2) : base(item1, item2) { }") | 
|  | 46 | +				g.P("        }") | 
|  | 47 | +				g.P("        public class ", orderedMap, " : SortedDictionary<", keyType, ", ", orderedMapValue, "> { }") | 
|  | 48 | +				g.P() | 
|  | 49 | +			} else { | 
|  | 50 | +				g.P("        public class ", orderedMap, " : SortedDictionary<", keyType, ", ", parseMapValueType(fd), "> { }") | 
|  | 51 | +				g.P() | 
|  | 52 | +			} | 
|  | 53 | +			if depth == 1 { | 
|  | 54 | +				g.P("        private ", orderedMap, " OrderedMap = new ", orderedMap, "();") | 
|  | 55 | +				g.P() | 
|  | 56 | +			} | 
|  | 57 | +			break | 
|  | 58 | +		} | 
|  | 59 | +	} | 
|  | 60 | +} | 
|  | 61 | + | 
|  | 62 | +func genOrderedMapLoader(gen *protogen.Plugin, g *protogen.GeneratedFile, md protoreflect.MessageDescriptor, depth int, messagerFullName string) { | 
|  | 63 | +	if depth == 1 { | 
|  | 64 | +		g.P("            // OrderedMap init.") | 
|  | 65 | +		g.P("            OrderedMap.Clear();") | 
|  | 66 | +	} | 
|  | 67 | +	for i := 0; i < md.Fields().Len(); i++ { | 
|  | 68 | +		fd := md.Fields().Get(i) | 
|  | 69 | +		if fd.IsMap() { | 
|  | 70 | +			prefix := parseOrderedMapPrefix(fd, messagerFullName) | 
|  | 71 | +			// orderedMap := prefix + orderedMapSuffix | 
|  | 72 | +			orderedMapValue := prefix + orderedMapValueSuffix | 
|  | 73 | +			keyName := fmt.Sprintf("key%d", depth) | 
|  | 74 | +			valueName := fmt.Sprintf("value%d", depth) | 
|  | 75 | + | 
|  | 76 | +			tmpOrderedMapName := fmt.Sprintf("ordered_map%d", depth) | 
|  | 77 | + | 
|  | 78 | +			prevContainer := fmt.Sprintf("value%d", depth-1) | 
|  | 79 | +			prevTmpOrderedMapName := fmt.Sprintf("ordered_map%d", depth-1) | 
|  | 80 | +			if depth == 1 { | 
|  | 81 | +				prevContainer = "Data_" | 
|  | 82 | +				prevTmpOrderedMapName = "OrderedMap" | 
|  | 83 | +			} | 
|  | 84 | +			g.P(strings.Repeat("    ", depth+2), "foreach (var (", keyName, ", ", valueName, ") in ", prevContainer, ".", strcase.ToCamel(string(fd.Name())), ")") | 
|  | 85 | +			g.P(strings.Repeat("    ", depth+2), "{") | 
|  | 86 | +			nextMapFD := getNextLevelMapFD(fd.MapValue()) | 
|  | 87 | +			if nextMapFD != nil { | 
|  | 88 | +				nextPrefix := parseOrderedMapPrefix(nextMapFD, messagerFullName) | 
|  | 89 | +				nextOrderedMap := nextPrefix + orderedMapSuffix | 
|  | 90 | +				g.P(strings.Repeat("    ", depth+3), "var ", tmpOrderedMapName, " = new ", nextOrderedMap, "();") | 
|  | 91 | +			} | 
|  | 92 | +			if fd.MapValue().Kind() == protoreflect.MessageKind { | 
|  | 93 | +				genOrderedMapLoader(gen, g, fd.MapValue().Message(), depth+1, messagerFullName) | 
|  | 94 | +			} | 
|  | 95 | + | 
|  | 96 | +			if nextMapFD != nil { | 
|  | 97 | +				g.P(strings.Repeat("    ", depth+3), prevTmpOrderedMapName, "[", keyName, "] = new ", orderedMapValue, "(", tmpOrderedMapName, ", ", valueName, ");") | 
|  | 98 | +			} else { | 
|  | 99 | +				g.P(strings.Repeat("    ", depth+3), prevTmpOrderedMapName, "[", keyName, "] = ", valueName, ";") | 
|  | 100 | +			} | 
|  | 101 | +			g.P(strings.Repeat("    ", depth+2), "}") | 
|  | 102 | +			break | 
|  | 103 | +		} | 
|  | 104 | +	} | 
|  | 105 | +} | 
|  | 106 | + | 
|  | 107 | +func genOrderedMapGetters(gen *protogen.Plugin, g *protogen.GeneratedFile, md protoreflect.MessageDescriptor, depth int, keys []helper.MapKey, messagerFullName string) { | 
|  | 108 | +	if depth == 1 && !options.NeedGenOrderedMap(md, options.LangCS) { | 
|  | 109 | +		return | 
|  | 110 | +	} | 
|  | 111 | +	genGetterName := func(depth int) string { | 
|  | 112 | +		getter := "GetOrderedMap" | 
|  | 113 | +		if depth > 1 { | 
|  | 114 | +			getter = fmt.Sprintf("GetOrderedMap%v", depth-1) | 
|  | 115 | +		} | 
|  | 116 | +		return getter | 
|  | 117 | +	} | 
|  | 118 | +	for i := 0; i < md.Fields().Len(); i++ { | 
|  | 119 | +		fd := md.Fields().Get(i) | 
|  | 120 | +		if fd.IsMap() { | 
|  | 121 | +			g.P() | 
|  | 122 | +			if depth == 1 { | 
|  | 123 | +				g.P("        // OrderedMap accessors.") | 
|  | 124 | +			} | 
|  | 125 | +			getter := genGetterName(depth) | 
|  | 126 | +			prefix := parseOrderedMapPrefix(fd, messagerFullName) | 
|  | 127 | +			orderedMap := prefix + orderedMapSuffix | 
|  | 128 | + | 
|  | 129 | +			if depth == 1 { | 
|  | 130 | +				g.P("        public ref readonly ", orderedMap, " ", getter, "()") | 
|  | 131 | +				g.P("        {") | 
|  | 132 | +				g.P("            return ref OrderedMap;") | 
|  | 133 | +			} else { | 
|  | 134 | +				g.P("        public ", orderedMap, "? ", getter, "(", helper.GenGetParams(keys), ")") | 
|  | 135 | +				g.P("        {") | 
|  | 136 | +				lastKeyName := keys[len(keys)-1].Name | 
|  | 137 | +				if depth == 2 { | 
|  | 138 | +					g.P("            if (OrderedMap.TryGetValue(", lastKeyName, ", out var value))") | 
|  | 139 | +				} else { | 
|  | 140 | +					prevKeys := keys[:len(keys)-1] | 
|  | 141 | +					prevGetter := genGetterName(depth - 1) | 
|  | 142 | +					g.P("            var conf = ", prevGetter, "(", helper.GenGetArguments(prevKeys), ");") | 
|  | 143 | +					g.P("            if (conf != null && conf.TryGetValue(", lastKeyName, ", out var value))") | 
|  | 144 | +				} | 
|  | 145 | +				g.P("            {") | 
|  | 146 | +				g.P("                return value.Item1;") | 
|  | 147 | +				g.P("            }") | 
|  | 148 | +				g.P("            return null;") | 
|  | 149 | +			} | 
|  | 150 | +			g.P("        }") | 
|  | 151 | + | 
|  | 152 | +			keys = helper.AddMapKey(gen, fd, keys) | 
|  | 153 | +			if fd.MapValue().Kind() == protoreflect.MessageKind { | 
|  | 154 | +				genOrderedMapGetters(gen, g, fd.MapValue().Message(), depth+1, keys, messagerFullName) | 
|  | 155 | +			} | 
|  | 156 | +			break | 
|  | 157 | +		} | 
|  | 158 | +	} | 
|  | 159 | +} | 
|  | 160 | + | 
|  | 161 | +func parseOrderedMapPrefix(mapFd protoreflect.FieldDescriptor, messagerFullName string) string { | 
|  | 162 | +	if mapFd.MapValue().Kind() == protoreflect.MessageKind { | 
|  | 163 | +		localMsgProtoName := strings.TrimPrefix(string(mapFd.MapValue().Message().FullName()), messagerFullName+".") | 
|  | 164 | +		return strings.ReplaceAll(localMsgProtoName, ".", "_") | 
|  | 165 | +	} | 
|  | 166 | +	return mapFd.MapValue().Kind().String() | 
|  | 167 | +} | 
0 commit comments