Files
crtr/Assets/Cryville/Crtr/SkinSelectors.cs
2023-01-27 15:28:22 +08:00

166 lines
5.8 KiB
C#

using Cryville.Common;
using Cryville.Common.Pdt;
using Cryville.Crtr.Components;
using Cryville.Crtr.Event;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using UnityEngine;
using CAnchor = Cryville.Crtr.Anchor;
namespace Cryville.Crtr {
public class SkinSelectors {
readonly SkinSelector[] selectors;
public readonly HashSet<string> annotations;
public SkinSelectors(IEnumerable<SkinSelector> s, IEnumerable<string> a) {
selectors = s.ToArray();
annotations = a.ToHashSet();
}
public override string ToString() {
if (selectors.Length == 0) return "";
bool flag = false;
string r = "";
foreach (var a in annotations) {
if (flag) r += " @" + a;
else { r += a; flag = true; }
}
foreach (var s in selectors) {
if (flag) r += " " + s.ToString();
else { r += s.ToString(); flag = true; }
}
return r;
}
public void Optimize(PdtEvaluatorBase etor) {
for (int i = 0; i < selectors.Length; i++) {
selectors[i].Optimize(etor);
}
}
public IEnumerable<SkinContext> MatchStatic(ContainerState h, SkinContext ctx) {
IEnumerable<SkinContext> result = new SkinContext[] { ctx };
foreach (var s in selectors) {
result = result.SelectMany(l => s.MatchStatic(h, l));
}
return result;
}
public bool IsUpdatable(ContainerState h) {
foreach (var s in selectors)
if (!s.IsUpdatable(h)) return false;
return true;
}
public SkinContext MatchDynamic(ContainerState h, SkinContext ctx) {
foreach (var s in selectors) {
ctx = s.MatchDynamic(h, ctx);
if (ctx == null) return null;
}
return ctx;
}
}
public abstract class SkinSelector {
protected SkinSelector() { }
public abstract override string ToString();
public virtual void Optimize(PdtEvaluatorBase etor) { }
public virtual IEnumerable<SkinContext> MatchStatic(ContainerState h, SkinContext c) { throw new SelectorNotAvailableException(); }
public virtual SkinContext MatchDynamic(ContainerState h, SkinContext c) { throw new SelectorNotAvailableException(); }
public virtual bool IsUpdatable(ContainerState h) {
return true;
}
public class CreateObject : SkinSelector {
public CreateObject() { }
public override string ToString() { return "$"; }
public override IEnumerable<SkinContext> MatchStatic(ContainerState h, SkinContext c) {
var obj = new GameObject("__obj__");
obj.transform.SetParent(c.Transform, false);
obj.AddComponent<TransformInterface>();
return new SkinContext[] { new SkinContext(obj.transform) };
}
}
public class Anchor : SkinSelector {
public int Name { get; private set; }
public Anchor(string name) {
Name = IdentifierManager.SharedInstance.Request(name);
}
public override string ToString() { return string.Format(".{0}", IdentifierManager.SharedInstance.Retrieve(Name)); }
public override IEnumerable<SkinContext> MatchStatic(ContainerState h, SkinContext c) {
List<CAnchor> anchors;
if (h.Handler.Anchors.TryGetValue(Name, out anchors)) {
return anchors.Select(a => a.SkinContext);
}
else return Enumerable.Empty<SkinContext>();
}
public override bool IsUpdatable(ContainerState h) {
return h.Handler.OpenedAnchor != null && h.Handler.OpenedAnchor.Name == Name;
}
}
public class AtAnchor : SkinSelector {
public int Name { get; private set; }
public AtAnchor(string name) {
Name = IdentifierManager.SharedInstance.Request(name);
}
public override string ToString() { return string.Format("..{0}", IdentifierManager.SharedInstance.Retrieve(Name)); }
public override SkinContext MatchDynamic(ContainerState h, SkinContext c) {
return IsUpdatable(h) ? c : null;
}
public override bool IsUpdatable(ContainerState h) {
return h.Handler.OpenedAnchor != null && h.Handler.OpenedAnchor.Name == Name;
}
}
public class Property : SkinSelector {
readonly PdtExpression _exp;
readonly PdtOperator _op;
bool _flag;
public Property(PdtExpression exp) {
_exp = exp;
_op = new PropOp.Boolean(v => _flag = v);
}
public override string ToString() { return string.Format("> {{{0}}}", _exp); }
public override void Optimize(PdtEvaluatorBase etor) {
etor.Optimize(_exp);
}
public override IEnumerable<SkinContext> MatchStatic(ContainerState h, SkinContext c) {
var result = Match(c);
if (result != null) return new SkinContext[] { result };
else return Enumerable.Empty<SkinContext>();
}
public override SkinContext MatchDynamic(ContainerState h, SkinContext c) {
return Match(c);
}
public SkinContext Match(SkinContext a) {
ChartPlayer.etor.ContextTransform = a.Transform;
try {
ChartPlayer.etor.Evaluate(_op, _exp);
return _flag ? a : null;
}
catch (Exception ex) {
throw new SelectorNotAvailableException("The expression is not evaluatable under the current context", ex);
}
finally {
ChartPlayer.etor.ContextTransform = null;
}
}
}
public class ElementType : SkinSelector {
readonly string _type;
public ElementType(string type) { _type = type; }
public override string ToString() { return _type; }
public override IEnumerable<SkinContext> MatchStatic(ContainerState h, SkinContext c) {
return h.Handler.TypeName == _type ? new SkinContext[] { c } : Enumerable.Empty<SkinContext>();
}
}
}
public class SelectorNotAvailableException : Exception {
public SelectorNotAvailableException() : base("The selector is not available under the current context") { }
public SelectorNotAvailableException(string message) : base(message) { }
public SelectorNotAvailableException(string message, Exception innerException) : base(message, innerException) { }
protected SelectorNotAvailableException(SerializationInfo info, StreamingContext context) : base(info, context) { }
}
}