Я использую Rectangle() для добавления нижней границы в TextField (SwiftUI)
Но я хочу использовать protocol TextFieldStyle для нижней строки в стиле TextField, например RoundedBorderTextFieldStyle.
Как я могу создать собственный стиль для TextField без использования Rectangle?
https://developer.apple.com/documentation/swiftui/staticmember
struct ContentView : View {
@State private var username = "Text Hellow"
var body: some View {
VStack() {
TextField($username)
.foregroundColor(Color.yellow)
Rectangle()
.frame(height: 1.0, alignment: .bottom)
.relativeWidth(1)
.foregroundColor(Color.red)
}
.padding()
}
func editChanged() {
}
}
#if DEBUG
struct ContentView_Previews : PreviewProvider {
static var previews: some View {
ContentView()
}
}
#endif





Чтобы определить собственный стиль, вы можете использовать приведенный ниже код. В конечном итоге прямоугольник используется, но внутри пользовательского стиля, а не в коде вашего представления. Это то, что вы хотите?
Примечание. Протестировано на бета-версии 3.
Чтобы использовать собственный инициализатор:
struct ContentView : View {
@State private var username = ""
var body: some View {
VStack {
TextField("Enter text", text: $username)
.textFieldStyle(.myOwnStyle)
}.frame(width:300)
}
}
Ваша собственная реализация инициализатора:
public struct MyOwnTextFieldStyle: TextFieldStyle {
public func _body(configuration: TextField<Self.Label>) -> some View {
VStack() {
configuration
.border(Color.blue, width: 2)
.foregroundColor(Color.yellow)
Rectangle()
.frame(height: 1.0, alignment: .bottom)
.relativeWidth(1)
.foregroundColor(Color.red)
}
}
}
extension StaticMember where Base : TextFieldStyle {
public static var myOwnStyle: MyOwnTextFieldStyle.Member {
StaticMember<MyOwnTextFieldStyle>(MyOwnTextFieldStyle())
}
}
Больше никаких StaticMember в бета-версии 6. Любая замена?
Однако решение контики не работает с Бета 6.
Поэтому я создал решение, в котором я встраиваю TextField и нарисованную нижнюю строку в новое представление. Вы можете просто скопировать код и использовать его, написав TextFieldWithBottomLine(placeholder: "My placeholder") Используется в представлении, которое выглядит так:
Первое, что я создаю, это горизонтальная линия:
import SwiftUI
struct HorizontalLineShape: Shape {
func path(in rect: CGRect) -> Path {
let fill = CGRect(x: 0, y: 0, width: rect.size.width, height: rect.size.height)
var path = Path()
path.addRoundedRect(in: fill, cornerSize: CGSize(width: 2, height: 2))
return path
}
}
struct HorizontalLine: View {
private var color: Color? = nil
private var height: CGFloat = 1.0
init(color: Color, height: CGFloat = 1.0) {
self.color = color
self.height = height
}
var body: some View {
HorizontalLineShape().fill(self.color!).frame(minWidth: 0, maxWidth: .infinity, minHeight: height, maxHeight: height)
}
}
struct HorizontalLine_Previews: PreviewProvider {
static var previews: some View {
HorizontalLine(color: .black)
}
}
Затем я создаю представление, содержащее TextField и HorizontalLine ниже:
import SwiftUI
struct TextFieldWithBottomLine: View {
@State var text: String = ""
private var placeholder = ""
private let lineThickness = CGFloat(2.0)
init(placeholder: String) {
self.placeholder = placeholder
}
var body: some View {
VStack {
TextField(placeholder, text: $text)
HorizontalLine(color: .black)
}.padding(.bottom, lineThickness)
}
}
struct TextFieldWithBottomLine_Previews: PreviewProvider {
static var previews: some View {
TextFieldWithBottomLine(placeholder: "My placeholder")
}
}
Чтобы увидеть это в действии, я создал образец представления:
import SwiftUI
struct SampleView: View {
@State var text: String = ""
@ObservedObject var model: LoginModel
init() {
self.model = LoginModel()
}
init(model: LoginModel) {
self.model = model
}
var body: some View {
TextFieldWithBottomLine(placeholder: "My placeholder").padding(24)
}
func initialActions() {
}
func reactOnButtonClick() {
}
}
struct SampleView_Previews: PreviewProvider {
static var previews: some View {
SampleView()
}
}
Если анимация не будет поддерживаться, мы можем сделать color и heightprivate let.
Соответствие протоколу TextFieldStyle
struct BottomLineTextFieldStyle: TextFieldStyle {
func _body(configuration: TextField<Self._Label>) -> some View {
VStack() {
configuration
Rectangle()
.frame(height: 0.5, alignment: .bottom)
.foregroundColor(Color.secondary)
}
}
}
Применение:
TextField("Name", text: $presenter.name).textFieldStyle(BottomLineTextFieldStyle())
Попробуйте по-другому
struct ContentView: View {
@State var name: String = ""
var body: some View {
VStack {
// change frame as per requirement
TextField("Please enter text", text: $name)
.frame(width: 300, height: 30, alignment: .center)
.background(Color.gray.opacity(0.2))
LineView().frame(width: 300, height: 1.0, alignment: .center)
}
}
}
LineView : создание пользовательских оберток для UIView
struct LineView: UIViewRepresentable {
typealias UIViewType = UIView
func makeUIView(context: UIViewRepresentableContext<LineView>) -> UIView {
let view = UIView()
view.backgroundColor = UIColor.lightGray
return view
}
func updateUIView(_ uiView: UIView, context: UIViewRepresentableContext<LineView>) {
}
}
Мое решение состоит в том, чтобы добавить разделитель под текстовое поле:
private let textFieldLength:CGFloat = 200
var body: some View {
VStack {
TextField("placeHolder", text: .constant("")).frame(width: textFieldLength, height: 30, alignment: .center)
Divider().frame(width: textFieldLength, height: 2, alignment: .center)
}
}
или использовать оверлей
TextField("", text: .constant(""))
.frame(width: textFieldLength, height: 30, alignment: .center)
.multilineTextAlignment(.leading)
.overlay(Divider().frame(width: textFieldLength, height: 2, alignment: .center), alignment: .bottom)
Я думаю, что решения оверлея чище, чем другие
Мне нравится простота первого фрагмента. Превосходно.
Делитель
Визуальный элемент, который можно использовать для отделения другого контента.
Вы можете установить color и height
Divider()
.frame(height: 1)
.padding(.horizontal, 30)
.background(Color.red)
struct LoginField: View {
@State private var email: String = ""
@State private var password: String = ""
var body: some View {
VStack {
TextField("Email", text: $email)
.padding(.horizontal, 30).padding(.top, 20)
Divider()
.padding(.horizontal, 30)
TextField("Password", text: $password)
.padding(.horizontal, 30).padding(.top, 20)
Divider()
.padding(.horizontal, 30)
Spacer()
}
}
}
Просто обратите внимание, что вы получаете предложенный платформой интервал, используя .padding(.horizontal) (без указания явной длины)
Использование .overlay
TextField("[email protected]", text: $email)
.overlay(VStack{Divider().offset(x: 0, y: 15)})
использование VStack заключается в том, чтобы Divider всегда был горизонтальным.
Просто гениально. Лучший ответ.
почему бы не использовать просто .overlay(Divider(), alignment: .bottom)
Да, это то, что я ищу. Спасибо.