SwiftUI. Сбой приложения на iPad с iOS 13.4-13.7 при длительном нажатии

#ios #ipad #swiftui #drag #ios13.4

#iOS #iPad #swiftui #перетаскивание #ios13.4

Вопрос:

Это мой первый вопрос по stackoverflow. Пожалуйста, поправьте меня, если я ошибаюсь с форматированием темы.

Я получаю сбой приложения при попытке использовать длительное нажатие (? тактильное касание?) на реальном iPad. Также это воспроизводимо на симуляторах iPad с принудительным нажатием на Magic Trackpad.

На каких устройствах есть эта ошибка?

Каждый iPad с iOS 13.4 до последней версии iOS 13. Каждый симулятор iPad с iOS 13.4 до последней версии iOS 13.

Я никогда не видел эту ошибку на iPhone.

Мои настройки для тестирования: macOS Catalina 10.15.6, xCode 11.7, симулятор iPad mini 2019 5-го поколения iOS 13.7, iPad 2018 6-го поколения iOS 13.7.

Обновление 26.10.2020. Кроме того, я пытаюсь использовать xCode 12.0.1 на macOS Catalina 10.15.7, iPad 2018 6-го поколения iOS 13.7. Я повторно протестировал, похоже, ошибка все еще доступна на iOS 13.4-13.7, но исчезла для iOS 14. Хорошего решения для iOS 13 я не нашел.

Шаги по воспроизведению ошибки.

  1. Я использую SwiftUI view ( TestCrashIPAD ) с разделами List, ForEach и dynamic (исходный код ниже).
  2. Некоторые статические разделы предопределены, некоторые динамические разделы появляются после загрузки с сервера.
  3. ВАЖНО, ошибка воспроизводится только при первом появлении view! Не переходите на другой экран и не сворачивайте приложение!
  4. Просто откройте это представление ( TestCrashIPAD ).
  5. Подождите, пока появятся динамические разделы.
  6. Как только появятся разделы, попробуйте долго нажимать на ячейку с текстом «r1» или «r2» или … (скриншот)
  7. Получить сбой (скриншот)

СБОЙ

Я получаю сбой в AppDelegate с: Thread 1: EXC_BAD_INSTRUCTION (code=EXC_I386_INVOP, subcode=0x0)

Я немного исследовал и обнаружил: эта ошибка воспроизводима с iOS 13.4. Также в iOS 13.4 есть модификатор просмотра onDrag (ссылка). Может быть, это как-то связано, потому что в стеке вызовов я вижу:

 UIDragInteractionLongPressDriver
UIGestureRecognizer
  

Может быть что-то не так с массивом динамических разделов, потому что я вижу:

 SwiftUI.Sections.rowIDs(forSectionAt: Swift.Int)
SwiftUI.ListCoreDataSource.rowIndex(at: Foundation.IndexPath)
  

Вопрос

Как исправить эту проблему? Как обойти это без сбоев в этом случае? Пожалуйста, скажите любую идею.

Исходный код

Пожалуйста, посмотрите исходный код:

 import SwiftUI

struct TestCrashIPAD: View {

    @State var sections = [TestSection]()

    var body: some View {
        List {
            // predefined sections
            Section(header: Text("Section predefined")) {
                Text("1")
                Text("2")
            }
            Section(header: Text("Section predefined")) {
                ForEach(1..<7) {
                    Text("($0)")
                }
            }

            // dynamic content
            ForEach(sections) { section in
                Section(header: Text(section.title)) {
                    ForEach(section.rows) { row in
                        HStack {
                            Text(row.str)
                            Spacer()
                        }
                    }
                }
            }
        }
        .listStyle(GroupedListStyle())
        .navigationBarTitle("iPad crash with long press")
        .onAppear(perform: onAppearAction)
    }

    func onAppearAction() {

        DispatchQueue.global().async {
            // fetch some data from the server
            sleep(3)

            // show data in UI
            DispatchQueue.main.async {
                self.sections = [TestSection](TestSection.array)
            }
        }
    }
}

struct TestSection: Identifiable {
    let id = UUID()
    let title: String
    let rows: [TestRow]

    static var array: [TestSection] {
        var result = [TestSection]()
        for idx in 0..<50 {
            result.append(TestSection(title: "s(idx)", rows: TestRow.array))
        }
        return result
    }
}

struct TestRow: Identifiable {
    let id = UUID()
    let str: String

    static var array: [TestRow] {
        var result = [TestRow]()
        for idx in 0..<Int.random(in: 3..<7) {
            result.append(TestRow(str: "r(idx)"))
        }
        return result
    }
}

  

Комментарии:

1. У нас был такой же сбой. Добавление .onLongPressGesture {} в представление в foreach, казалось, исправило это для нас.

2. @mikemike396, это помогает в xCode 12 с iOS 13.7 на iPad 2018 (6-го поколения). Какую версию вы используете: xCode 11 или 12?

3. Мы используем Xcode 12

Ответ №1:

Похоже, что ответ заключается в добавлении .onLongPressGesture { } в ваши списки, которые выходят из строя (я получил это из комментариев, но это решило мои проблемы, так что похоже на ответ. публикация в качестве ответа, потому что не все читают комментарии