2020-07-27 01:16:37 -04:00
|
|
|
package d2datadict
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2020-07-30 10:14:15 -04:00
|
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2common"
|
2020-07-27 01:16:37 -04:00
|
|
|
"log"
|
2020-07-30 10:14:15 -04:00
|
|
|
)
|
2020-07-27 01:16:37 -04:00
|
|
|
|
2020-07-30 10:14:15 -04:00
|
|
|
const (
|
|
|
|
maxTreasuresPerRecord = 10
|
|
|
|
treasureItemFmt = "Item%d"
|
|
|
|
treasureProbFmt = "Prob%d"
|
|
|
|
)
|
|
|
|
|
|
|
|
// TreasureDropType indicates the drop type of the treasure
|
|
|
|
type TreasureDropType int
|
|
|
|
|
|
|
|
const (
|
|
|
|
// TreasureNone is default bad case, but nothing should have this
|
|
|
|
TreasureNone TreasureDropType = iota
|
|
|
|
|
|
|
|
// TreasureGold indicates that the treasure drop type is for gold
|
|
|
|
TreasureGold
|
|
|
|
|
|
|
|
// indicates that the drop type resolves directly to an ItemCommonRecord
|
|
|
|
TreasureWeapon
|
|
|
|
TreasureArmor
|
|
|
|
TreasureMisc
|
|
|
|
|
|
|
|
// indicates that the code is for a dynamic item record, because the treasure code has
|
|
|
|
// and item level appended to it. this is for things like `armo63` or `weap24` which does not
|
|
|
|
// explicitly have an item record that matches this code, but we need to resolve this
|
|
|
|
TreasureWeaponDynamic
|
|
|
|
TreasureArmorDynamic
|
|
|
|
TreasureMiscDynamic
|
2020-07-27 01:16:37 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-07-30 10:14:15 -04:00
|
|
|
GoldMultDropCodeStr string = "gld,mul="
|
|
|
|
GoldDropCodeStr = "gld"
|
|
|
|
WeaponDropCodeStr = "weap"
|
|
|
|
ArmorDropCodeStr = "armo"
|
|
|
|
MiscDropCodeStr = "misc"
|
2020-07-27 01:16:37 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// TreasureClassRecord represents a rule for item drops in diablo 2
|
|
|
|
type TreasureClassRecord struct {
|
|
|
|
Name string
|
|
|
|
Group int
|
|
|
|
Level int
|
|
|
|
NumPicks int
|
|
|
|
FreqUnique int
|
|
|
|
FreqSet int
|
|
|
|
FreqRare int
|
|
|
|
FreqMagic int
|
|
|
|
FreqNoDrop int
|
|
|
|
Treasures []*Treasure
|
|
|
|
}
|
|
|
|
|
|
|
|
// Treasure describes a treasure to drop
|
2020-07-30 10:14:15 -04:00
|
|
|
// the Name is either a reference to an item, or to another treasure class
|
2020-07-27 01:16:37 -04:00
|
|
|
type Treasure struct {
|
2020-07-30 10:14:15 -04:00
|
|
|
Code string
|
2020-07-27 01:16:37 -04:00
|
|
|
Probability int
|
|
|
|
}
|
|
|
|
|
2020-07-30 10:14:15 -04:00
|
|
|
// TreasureClass contains all of the TreasureClassRecords
|
2020-07-27 01:16:37 -04:00
|
|
|
var TreasureClass map[string]*TreasureClassRecord //nolint:gochecknoglobals // Currently global by design
|
|
|
|
|
|
|
|
// LoadTreasureClassRecords loads treasure class records from TreasureClassEx.txt
|
|
|
|
//nolint:funlen // Makes no sense to split
|
|
|
|
func LoadTreasureClassRecords(file []byte) {
|
|
|
|
TreasureClass = make(map[string]*TreasureClassRecord)
|
|
|
|
|
|
|
|
d := d2common.LoadDataDictionary(file)
|
|
|
|
|
|
|
|
for d.Next() {
|
|
|
|
record := &TreasureClassRecord{
|
|
|
|
Name: d.String("Treasure Class"),
|
|
|
|
Group: d.Number("group"),
|
|
|
|
Level: d.Number("level"),
|
|
|
|
NumPicks: d.Number("Picks"),
|
|
|
|
FreqUnique: d.Number("Unique"),
|
|
|
|
FreqSet: d.Number("Set"),
|
|
|
|
FreqRare: d.Number("Rare"),
|
|
|
|
FreqMagic: d.Number("Magic"),
|
|
|
|
FreqNoDrop: d.Number("NoDrop"),
|
|
|
|
}
|
|
|
|
|
2020-07-30 10:14:15 -04:00
|
|
|
if record.Name == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for treasureIdx := 0; treasureIdx < maxTreasuresPerRecord; treasureIdx++ {
|
2020-07-27 01:16:37 -04:00
|
|
|
treasureColumnKey := fmt.Sprintf(treasureItemFmt, treasureIdx+1)
|
|
|
|
probColumnKey := fmt.Sprintf(treasureProbFmt, treasureIdx+1)
|
|
|
|
|
|
|
|
treasureName := d.String(treasureColumnKey)
|
|
|
|
if treasureName == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
prob := d.Number(probColumnKey)
|
|
|
|
|
|
|
|
treasure := &Treasure{
|
2020-07-30 10:14:15 -04:00
|
|
|
Code: treasureName,
|
2020-07-27 01:16:37 -04:00
|
|
|
Probability: prob,
|
|
|
|
}
|
|
|
|
|
|
|
|
if record.Treasures == nil {
|
|
|
|
record.Treasures = []*Treasure{treasure}
|
2020-07-30 10:14:15 -04:00
|
|
|
} else {
|
|
|
|
record.Treasures = append(record.Treasures, treasure)
|
2020-07-27 01:16:37 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TreasureClass[record.Name] = record
|
|
|
|
}
|
|
|
|
|
|
|
|
if d.Err != nil {
|
|
|
|
panic(d.Err)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Loaded %d TreasureClass records", len(TreasureClass))
|
|
|
|
}
|