Merge remote-tracking branch 'console/master'

This commit is contained in:
Him188 2021-02-11 18:53:14 +08:00
commit 188601fb87
6 changed files with 187 additions and 37 deletions

View File

@ -1,8 +1,8 @@
/* /*
* Copyright 2019-2020 Mamoe Technologies and contributors. * Copyright 2019-2021 Mamoe Technologies and contributors.
* *
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证. * 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link. * Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
* *
* https://github.com/mamoe/mirai/blob/master/LICENSE * https://github.com/mamoe/mirai/blob/master/LICENSE
*/ */

View File

@ -1,8 +1,8 @@
/* /*
* Copyright 2019-2020 Mamoe Technologies and contributors. * Copyright 2019-2021 Mamoe Technologies and contributors.
* *
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证. * 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link. * Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
* *
* https://github.com/mamoe/mirai/blob/master/LICENSE * https://github.com/mamoe/mirai/blob/master/LICENSE
*/ */

View File

@ -1,8 +1,8 @@
/* /*
* Copyright 2019-2020 Mamoe Technologies and contributors. * Copyright 2019-2021 Mamoe Technologies and contributors.
* *
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证. * 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link. * Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
* *
* https://github.com/mamoe/mirai/blob/master/LICENSE * https://github.com/mamoe/mirai/blob/master/LICENSE
*/ */
@ -109,13 +109,14 @@ internal abstract class CompositeListValueImpl<T>(
// workaround to a type inference bug // workaround to a type inference bug
internal fun <K, V> PluginData.createCompositeMapValueImpl( internal fun <K, V> PluginData.createCompositeMapValueImpl(
mapInitializer: (() -> MutableMap<Value<K>, Value<V>>?)? = null,
kToValue: (K) -> Value<K>, kToValue: (K) -> Value<K>,
vToValue: (V) -> Value<V>, vToValue: (V) -> Value<V>,
valueToK: (Value<K>) -> K = Value<K>::value, valueToK: (Value<K>) -> K = Value<K>::value,
valueToV: (Value<V>) -> V = Value<V>::value, valueToV: (Value<V>) -> V = Value<V>::value,
applyToShadowedMap: ((MutableMap<K, V>) -> (MutableMap<K, V>))? = null applyToShadowedMap: ((MutableMap<K, V>) -> (MutableMap<K, V>))? = null
): CompositeMapValueImpl<K, V> { ): CompositeMapValueImpl<K, V> {
return object : CompositeMapValueImpl<K, V>(kToValue, vToValue, valueToK, valueToV, applyToShadowedMap) { return object : CompositeMapValueImpl<K, V>(mapInitializer, kToValue, vToValue, valueToK, valueToV, applyToShadowedMap) {
override fun onChanged() = this@createCompositeMapValueImpl.onValueChanged(this) override fun onChanged() = this@createCompositeMapValueImpl.onValueChanged(this)
} }
} }
@ -123,6 +124,7 @@ internal fun <K, V> PluginData.createCompositeMapValueImpl(
// TODO: 2020/6/24 在一个 Value 被删除后停止追踪其更新. // TODO: 2020/6/24 在一个 Value 被删除后停止追踪其更新.
internal abstract class CompositeMapValueImpl<K, V>( internal abstract class CompositeMapValueImpl<K, V>(
mapInitializer: (() -> MutableMap<Value<K>, Value<V>>?)? = null,
@JvmField internal val kToValue: (K) -> Value<K>, // should override onChanged @JvmField internal val kToValue: (K) -> Value<K>, // should override onChanged
@JvmField internal val vToValue: (V) -> Value<V>, // should override onChanged @JvmField internal val vToValue: (V) -> Value<V>, // should override onChanged
@JvmField internal val valueToK: (Value<K>) -> K = Value<K>::value, @JvmField internal val valueToK: (Value<K>) -> K = Value<K>::value,
@ -130,7 +132,7 @@ internal abstract class CompositeMapValueImpl<K, V>(
applyToShadowedMap: ((MutableMap<K, V>) -> (MutableMap<K, V>))? = null applyToShadowedMap: ((MutableMap<K, V>) -> (MutableMap<K, V>))? = null
) : CompositeMapValue<K, V>, AbstractValueImpl<Map<K, V>>() { ) : CompositeMapValue<K, V>, AbstractValueImpl<Map<K, V>>() {
@JvmField @JvmField
internal val internalList: MutableMap<Value<K>, Value<V>> = mutableMapOf() internal val internalList: MutableMap<Value<K>, Value<V>> = mapInitializer?.invoke() ?: mutableMapOf()
private var _value: MutableMap<K, V> = private var _value: MutableMap<K, V> =
internalList.shadowMap(valueToK, kToValue, valueToV, vToValue).let { internalList.shadowMap(valueToK, kToValue, valueToV, vToValue).let {

View File

@ -1,8 +1,8 @@
/* /*
* Copyright 2019-2020 Mamoe Technologies and contributors. * Copyright 2019-2021 Mamoe Technologies and contributors.
* *
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证. * 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link. * Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
* *
* https://github.com/mamoe/mirai/blob/master/LICENSE * https://github.com/mamoe/mirai/blob/master/LICENSE
*/ */
@ -11,15 +11,28 @@
package net.mamoe.mirai.console.internal.data package net.mamoe.mirai.console.internal.data
import java.util.concurrent.ConcurrentMap
import java.util.function.BiConsumer
import java.util.function.BiFunction
import java.util.function.Function
// TODO: 2020/6/24 优化性能: 引入一个 comparator 之类来替代将 Int 包装为 Value<Int> 后进行 containsKey 比较的方法 // TODO: 2020/6/24 优化性能: 引入一个 comparator 之类来替代将 Int 包装为 Value<Int> 后进行 containsKey 比较的方法
// java.util.function was used in mirai-core
// direct improve apis of java.util.function
@Suppress(
"MANY_IMPL_MEMBER_NOT_IMPLEMENTED", "MANY_INTERFACES_MEMBER_NOT_IMPLEMENTED",
"UNCHECKED_CAST", "USELESS_CAST", "ACCIDENTAL_OVERRIDE", "TYPE_MISMATCH",
"NOTHING_TO_OVERRIDE", "EXPLICIT_OVERRIDE_REQUIRED_IN_MIXED_MODE", "CONFLICTING_INHERITED_JVM_DECLARATIONS",
) // for improve java.util.function apis
internal open class ShadowMap<K, V, KR, VR>( internal open class ShadowMap<K, V, KR, VR>(
private val originMapComputer: () -> MutableMap<K, V>, @JvmField protected val originMapComputer: () -> MutableMap<K, V>,
private val kTransform: (K) -> KR, @JvmField protected val kTransform: (K) -> KR,
private val kTransformBack: (KR) -> K, @JvmField protected val kTransformBack: (KR) -> K,
private val vTransform: (V) -> VR, @JvmField protected val vTransform: (V) -> VR,
private val vTransformBack: (VR) -> V @JvmField protected val vTransformBack: (VR) -> V
) : MutableMap<KR, VR> { ) : MutableMap<KR, VR> {
override val size: Int get() = originMapComputer().size override val size: Int get() = originMapComputer().size
override fun containsKey(key: KR): Boolean = originMapComputer().containsKey(key.let(kTransformBack)) override fun containsKey(key: KR): Boolean = originMapComputer().containsKey(key.let(kTransformBack))
@ -76,6 +89,10 @@ internal open class ShadowMap<K, V, KR, VR>(
} }
override fun remove(key: KR): VR? = originMapComputer().remove(key.let(kTransformBack))?.let(vTransform) override fun remove(key: KR): VR? = originMapComputer().remove(key.let(kTransformBack))?.let(vTransform)
override fun remove(key: KR, value: VR): Boolean =
originMapComputer().remove(key.let(kTransformBack), value?.let(vTransformBack))
override fun toString(): String = originMapComputer().toString() override fun toString(): String = originMapComputer().toString()
override fun hashCode(): Int = originMapComputer().hashCode() override fun hashCode(): Int = originMapComputer().hashCode()
override fun equals(other: Any?): Boolean { override fun equals(other: Any?): Boolean {
@ -92,14 +109,77 @@ internal open class ShadowMap<K, V, KR, VR>(
return true return true
} }
override fun putIfAbsent(key: KR, value: VR): VR? =
originMapComputer().putIfAbsent(key.let(kTransformBack), value.let(vTransformBack))?.let(vTransform)
override fun replace(key: KR, oldValue: VR, newValue: VR): Boolean =
originMapComputer().replace(key.let(kTransformBack), oldValue.let(vTransformBack), newValue.let(vTransformBack))
override fun replace(key: KR, value: VR): VR? =
originMapComputer().replace(key.let(kTransformBack), value.let(vTransformBack))?.let(vTransform)
override fun compute(key: KR, remappingFunction: BiFunction<in KR, in VR?, out VR?>): VR? =
originMapComputer().compute(key.let(kTransformBack)) { k1, v1 ->
remappingFunction.apply(k1.let(kTransform), v1?.let(vTransform))?.let(vTransformBack)
}?.let(vTransform)
override fun computeIfAbsent(key: KR, mappingFunction: Function<in KR, out VR>): VR =
originMapComputer().computeIfAbsent(key.let(kTransformBack)) { k ->
mappingFunction.apply(k.let(kTransform)).let(vTransformBack)
}.let(vTransform)
override fun computeIfPresent(key: KR, remappingFunction: BiFunction<in KR, in VR, out VR?>): VR? =
originMapComputer().computeIfPresent(key.let(kTransformBack)) { k, v ->
remappingFunction.apply(k.let(kTransform), v.let(vTransform))?.let(vTransformBack)
}?.let(vTransform)
override fun merge(key: KR, value: VR, remappingFunction: BiFunction<in VR, in VR, out VR?>): VR? =
originMapComputer().merge(key.let(kTransformBack), value.let(vTransformBack)) { k, v ->
remappingFunction.apply(k.let(vTransform), v.let(vTransform))?.let(vTransformBack)
}?.let(vTransform)
override fun forEach(action: BiConsumer<in KR, in VR>) {
@Suppress("JavaMapForEach")
originMapComputer().forEach { t, u ->
action.accept(t.let(kTransform), u.let(vTransform))
}
}
override fun replaceAll(function: BiFunction<in KR, in VR, out VR>) {
originMapComputer().replaceAll { t, u ->
function.apply(t.let(kTransform), u.let(vTransform))?.let(vTransformBack)
}
}
} }
@Suppress(
"MANY_IMPL_MEMBER_NOT_IMPLEMENTED", "MANY_INTERFACES_MEMBER_NOT_IMPLEMENTED",
"UNCHECKED_CAST", "USELESS_CAST", "ACCIDENTAL_OVERRIDE", "TYPE_MISMATCH",
"EXPLICIT_OVERRIDE_REQUIRED_IN_MIXED_MODE", "CONFLICTING_INHERITED_JVM_DECLARATIONS"
)
internal open class ConcurrentShadowMap<K, V, KR, VR>(
originMapComputer: () -> MutableMap<K, V>,
kTransform: (K) -> KR,
kTransformBack: (KR) -> K,
vTransform: (V) -> VR,
vTransformBack: (VR) -> V
) : ShadowMap<K, V, KR, VR>(
originMapComputer, kTransform, kTransformBack, vTransform, vTransformBack
), ConcurrentMap<KR, VR>
internal fun <K, V, KR, VR> MutableMap<K, V>.shadowMap( internal fun <K, V, KR, VR> MutableMap<K, V>.shadowMap(
kTransform: (K) -> KR, kTransform: (K) -> KR,
kTransformBack: (KR) -> K, kTransformBack: (KR) -> K,
vTransform: (V) -> VR, vTransform: (V) -> VR,
vTransformBack: (VR) -> V vTransformBack: (VR) -> V
): MutableMap<KR, VR> = ShadowMap({ this }, kTransform, kTransformBack, vTransform, vTransformBack) ): MutableMap<KR, VR> = if (this is ConcurrentMap<K, V>) {
ConcurrentShadowMap({ this }, kTransform, kTransformBack, vTransform, vTransformBack)
} else {
ShadowMap({ this }, kTransform, kTransformBack, vTransform, vTransformBack)
}
internal inline fun <E, R> MutableCollection<E>.shadowMap( internal inline fun <E, R> MutableCollection<E>.shadowMap(
crossinline transform: (E) -> R, crossinline transform: (E) -> R,
@ -319,9 +399,16 @@ internal inline fun <T> dynamicMutableSet(crossinline supplier: () -> MutableSet
} }
} }
*/ */
@Suppress("UNCHECKED_CAST", "USELESS_CAST") // type inference bug
internal inline fun <K, V> MutableMap<K, V>.observable(crossinline onChanged: () -> Unit): MutableMap<K, V> { @Suppress(
return object : MutableMap<K, V>, Map<K, V> by (this as Map<K, V>) { "UNCHECKED_CAST", "USELESS_CAST",
"ACCIDENTAL_OVERRIDE", "TYPE_MISMATCH", "NOTHING_TO_OVERRIDE",
"MANY_IMPL_MEMBER_NOT_IMPLEMENTED", "MANY_INTERFACES_MEMBER_NOT_IMPLEMENTED",
"UNCHECKED_CAST", "USELESS_CAST", "ACCIDENTAL_OVERRIDE",
"EXPLICIT_OVERRIDE_REQUIRED_IN_MIXED_MODE", "CONFLICTING_INHERITED_JVM_DECLARATIONS"
) // type inference bug
internal fun <K, V> MutableMap<K, V>.observable(onChanged: () -> Unit): MutableMap<K, V> {
open class ObservableMap : MutableMap<K, V> by (this as MutableMap<K, V>) {
override val keys: MutableSet<K> override val keys: MutableSet<K>
get() = this@observable.keys.observable(onChanged) get() = this@observable.keys.observable(onChanged)
override val values: MutableCollection<V> override val values: MutableCollection<V>
@ -335,7 +422,46 @@ internal inline fun <K, V> MutableMap<K, V>.observable(crossinline onChanged: ()
override fun remove(key: K): V? = this@observable.remove(key).also { onChanged() } override fun remove(key: K): V? = this@observable.remove(key).also { onChanged() }
override fun toString(): String = this@observable.toString() override fun toString(): String = this@observable.toString()
override fun hashCode(): Int = this@observable.hashCode() override fun hashCode(): Int = this@observable.hashCode()
override fun equals(other: Any?): Boolean {
if (other == null) return false
if (other === this) return true
return this@observable == other
} }
override fun remove(key: K, value: V): Boolean = this@observable.remove(key, value).also { onChanged() }
override fun putIfAbsent(key: K, value: V): V? =
this@observable.putIfAbsent(key, value).also { onChanged() }
override fun replace(key: K, oldValue: V, newValue: V): Boolean =
this@observable.replace(key, oldValue, newValue).also { onChanged() }
override fun replace(key: K, value: V): V? =
this@observable.replace(key, value).also { onChanged() }
override fun computeIfAbsent(key: K, mappingFunction: Function<in K, out V>): V =
this@observable.computeIfAbsent(key, mappingFunction).also { onChanged() }
override fun replaceAll(function: BiFunction<in K, in V, out V>) =
this@observable.replaceAll(function).also { onChanged() }
override fun compute(key: K, remappingFunction: BiFunction<in K, in V?, out V?>): V? =
this@observable.compute(key, remappingFunction).also { onChanged() }
override fun computeIfPresent(key: K, remappingFunction: BiFunction<in K, in V, out V?>): V? =
this@observable.computeIfPresent(key, remappingFunction).also { onChanged() }
override fun merge(key: K, value: V, remappingFunction: BiFunction<in V, in V, out V?>): V? =
this@observable.merge(key, value, remappingFunction).also { onChanged() }
}
@Suppress(
"MANY_IMPL_MEMBER_NOT_IMPLEMENTED", "MANY_INTERFACES_MEMBER_NOT_IMPLEMENTED",
"UNCHECKED_CAST", "USELESS_CAST", "ACCIDENTAL_OVERRIDE", "TYPE_MISMATCH",
"EXPLICIT_OVERRIDE_REQUIRED_IN_MIXED_MODE", "CONFLICTING_INHERITED_JVM_DECLARATIONS"
)
return if (this is ConcurrentMap<*, *>) {
object : ConcurrentMap<K, V>, MutableMap<K, V>, ObservableMap() {}
} else ObservableMap()
} }
internal inline fun <T> MutableList<T>.observable(crossinline onChanged: () -> Unit): MutableList<T> { internal inline fun <T> MutableList<T>.observable(crossinline onChanged: () -> Unit): MutableList<T> {

View File

@ -1,8 +1,8 @@
/* /*
* Copyright 2019-2020 Mamoe Technologies and contributors. * Copyright 2019-2021 Mamoe Technologies and contributors.
* *
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证. * 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link. * Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
* *
* https://github.com/mamoe/mirai/blob/master/LICENSE * https://github.com/mamoe/mirai/blob/master/LICENSE
*/ */
@ -22,6 +22,8 @@ import kotlinx.serialization.serializer
import net.mamoe.yamlkt.YamlDynamicSerializer import net.mamoe.yamlkt.YamlDynamicSerializer
import net.mamoe.yamlkt.YamlNullableDynamicSerializer import net.mamoe.yamlkt.YamlNullableDynamicSerializer
import java.lang.reflect.Modifier import java.lang.reflect.Modifier
import java.util.concurrent.ConcurrentHashMap
import java.util.concurrent.ConcurrentMap
import kotlin.reflect.KClass import kotlin.reflect.KClass
import kotlin.reflect.KType import kotlin.reflect.KType
@ -56,6 +58,10 @@ internal fun serializerMirai(type: KType): KSerializer<Any?> {
HashSet::class -> SetSerializer(serializers[0]) HashSet::class -> SetSerializer(serializers[0])
Set::class, MutableSet::class, LinkedHashSet::class -> SetSerializer(serializers[0]) Set::class, MutableSet::class, LinkedHashSet::class -> SetSerializer(serializers[0])
HashMap::class -> MapSerializer(serializers[0], serializers[1]) HashMap::class -> MapSerializer(serializers[0], serializers[1])
ConcurrentHashMap::class, ConcurrentMap::class -> MapSerializer(serializers[0], serializers[1]).map(
serializer = { HashMap(it as Map<*, *>) },
deserializer = { ConcurrentHashMap(it) }
)
Map::class, MutableMap::class, LinkedHashMap::class -> MapSerializer(serializers[0], serializers[1]) Map::class, MutableMap::class, LinkedHashMap::class -> MapSerializer(serializers[0], serializers[1])
Map.Entry::class -> MapEntrySerializer(serializers[0], serializers[1]) Map.Entry::class -> MapEntrySerializer(serializers[0], serializers[1])
Pair::class -> PairSerializer(serializers[0], serializers[1]) Pair::class -> PairSerializer(serializers[0], serializers[1])

View File

@ -1,8 +1,8 @@
/* /*
* Copyright 2019-2020 Mamoe Technologies and contributors. * Copyright 2019-2021 Mamoe Technologies and contributors.
* *
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证. * 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link. * Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
* *
* https://github.com/mamoe/mirai/blob/master/LICENSE * https://github.com/mamoe/mirai/blob/master/LICENSE
*/ */
@ -16,9 +16,12 @@ import net.mamoe.mirai.console.data.PluginData
import net.mamoe.mirai.console.data.SerializableValue.Companion.serializableValueWith import net.mamoe.mirai.console.data.SerializableValue.Companion.serializableValueWith
import net.mamoe.mirai.console.data.SerializerAwareValue import net.mamoe.mirai.console.data.SerializerAwareValue
import net.mamoe.mirai.console.data.valueFromKType import net.mamoe.mirai.console.data.valueFromKType
import java.util.concurrent.ConcurrentHashMap
import java.util.concurrent.ConcurrentMap
import kotlin.contracts.contract import kotlin.contracts.contract
import kotlin.reflect.KClass import kotlin.reflect.KClass
import kotlin.reflect.KType import kotlin.reflect.KType
import kotlin.reflect.full.isSubclassOf
private val primitiveCollectionsImplemented by lazy { private val primitiveCollectionsImplemented by lazy {
false false
@ -43,7 +46,9 @@ internal fun PluginData.valueFromKTypeImpl(type: KType): SerializerAwareValue<*>
MutableMap::class, MutableMap::class,
Map::class, Map::class,
LinkedHashMap::class, LinkedHashMap::class,
HashMap::class HashMap::class,
ConcurrentMap::class,
ConcurrentHashMap::class,
-> { -> {
val keyClass = type.arguments[0].type?.classifier val keyClass = type.arguments[0].type?.classifier
require(keyClass is KClass<*>) require(keyClass is KClass<*>)
@ -57,6 +62,13 @@ internal fun PluginData.valueFromKTypeImpl(type: KType): SerializerAwareValue<*>
TODO() TODO()
} else { } else {
return createCompositeMapValueImpl<Any?, Any?>( return createCompositeMapValueImpl<Any?, Any?>(
mapInitializer = {
if (classifier.cast<KClass<*>>().isSubclassOf(ConcurrentMap::class)) {
ConcurrentHashMap()
} else {
null
}
},
kToValue = { k -> valueFromKType(type.arguments[0].type!!, k) }, kToValue = { k -> valueFromKType(type.arguments[0].type!!, k) },
vToValue = { v -> valueFromKType(type.arguments[1].type!!, v) } vToValue = { v -> valueFromKType(type.arguments[1].type!!, v) }
).serializableValueWith(serializerMirai(type) as KSerializer<Map<Any?, Any?>>) // erased ).serializableValueWith(serializerMirai(type) as KSerializer<Map<Any?, Any?>>) // erased
@ -132,6 +144,10 @@ internal fun KClass<*>.createInstanceSmart(): Any {
HashSet::class HashSet::class
-> LinkedHashSet<Any?>() -> LinkedHashSet<Any?>()
ConcurrentHashMap::class,
ConcurrentMap::class,
-> ConcurrentHashMap<Any?,Any?>()
else -> createInstanceOrNull() else -> createInstanceOrNull()
?: error("Cannot create instance or find a initial value for ${this.qualifiedNameOrTip}") ?: error("Cannot create instance or find a initial value for ${this.qualifiedNameOrTip}")
} }