Slide 1

Slide 1 text

#iOSDC2018 .BSLEPXOΛ ϦΞϧλΠϜʹ ղੳ͢Δ

Slide 2

Slide 2 text

#iOSDC2018

Slide 3

Slide 3 text

!OBLBKJKBQBO 4PGUXBSF&OHJOFFS $".1'*3&J04%$4UB⒎/,+*OUFSOFU J04"OESPJENBD04

Slide 4

Slide 4 text

No content

Slide 5

Slide 5 text

.BSLEPXOΛ ϦΞϧλΠϜʹ ղੳ͢Δ

Slide 6

Slide 6 text

.BSLEPXOΛ ϦΞϧλΠϜʹ ղੳ͢Δ ͖͔͚ͬ

Slide 7

Slide 7 text

Izumo Markdown Note Tool

Slide 8

Slide 8 text

J04 NBD04

Slide 9

Slide 9 text

͖͔͚ͬ wࣗ෼ʹͱͬͯྑ͍ΤσΟλ͕ͳ͔ͬͨ wϦονͷදݱํ๏ʹෆຬ w৭͚ͩมΘͬͯ͘ΕΕ͹͍͍ wಉظͯ͠ཉ͍͠

Slide 10

Slide 10 text

΍Γ͍ͨ͜ͱ

Slide 11

Slide 11 text

wΧϥʔϦϯά wฤूิࠤ

Slide 12

Slide 12 text

ΧϥʔϦϯά ߦ໨͸λΠτϧߦɺݟग़͠ɺڧௐɺΠϯ ϥΠϯදࣔɺϦετɺ50%0Ϧετɺը૾ ଧͪফ͠ɺςʔϒϧɺਫฏઢɺҾ༻ɺίʔ υϒϩοΫɺҾ༻

Slide 13

Slide 13 text

ฤूิࠤ wࣗಈΠϯσϯτ wλϒͰͷϦετҠಈ

Slide 14

Slide 14 text

ௐࠪ

Slide 15

Slide 15 text

IUUQTHJUIVCDPN TFBSDI RNBSLEPXOTXJGU

Slide 16

Slide 16 text

No content

Slide 17

Slide 17 text

wϦονͳදݱʹม׵ͯ͠͠·͏ wશจΛಡΈࠐΉඞཁ͕͋Γ

Slide 18

Slide 18 text

࠷খݶͷൣғͰղੳ ॲཧΛߦ͍͍ͨ

Slide 19

Slide 19 text

w͍ͭʁ wԿΛʁ wͲͷΑ͏ʹͯ͠ʁ

Slide 20

Slide 20 text

͍ͭʁ

Slide 21

Slide 21 text

wςΩετ͕ೖྗ͞Εͨ wॳظԽ͞Εͨ wϑΥϯτ͕มߋ͞Εͨ

Slide 22

Slide 22 text

wςΩετ͕ೖྗ͞Εͨ wॳظԽ͞Εͨ wϑΥϯτ͕มߋ͞Εͨ

Slide 23

Slide 23 text

/45FYU4UPSBHF%FMFHBUF

Slide 24

Slide 24 text

// MARK: - NSTextStorageDelegate extension TextViewController: NSTextStorageDelegate { func textStorage(_ textStorage: NSTextStorage, willProcessEditing editedMask: NSTextStorageEditActions, range editedRange: NSRange, changeInLength delta: Int) { self.editedRange = editedRange } func textStorage(_ textStorage: NSTextStorage, didProcessEditing editedMask: NSTextStorageEditActions, range editedRange: NSRange, changeInLength delta: Int) { } }

Slide 25

Slide 25 text

ԿΛʁ

Slide 26

Slide 26 text

ม׵͢Δ΂͖ྖҬ

Slide 27

Slide 27 text

// MARK: - NSTextStorageDelegate extension TextViewController: NSTextStorageDelegate { func textStorage(_ textStorage: NSTextStorage, willProcessEditing editedMask: NSTextStorageEditActions, range editedRange: NSRange, changeInLength delta: Int) { self.editedRange = editedRange } func textStorage(_ textStorage: NSTextStorage, didProcessEditing editedMask: NSTextStorageEditActions, range editedRange: NSRange, changeInLength delta: Int) { } }

Slide 28

Slide 28 text

// MARK: - NSTextStorageDelegate extension TextViewController: NSTextStorageDelegate { func textStorage(_ textStorage: NSTextStorage, willProcessEditing editedMask: NSTextStorageEditActions, range editedRange: NSRange, changeInLength delta: Int) { self.editedRange = editedRange } func textStorage(_ textStorage: NSTextStorage, didProcessEditing editedMask: NSTextStorageEditActions, range editedRange: NSRange, changeInLength delta: Int) { } } ⚠ฤूͨ͠ྖҬ

Slide 29

Slide 29 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩ

Slide 30

Slide 30 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩʂ

Slide 31

Slide 31 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩ

Slide 32

Slide 32 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩʂ ໌೔΋J04%$ͩʂ ໌ޙ೔΋J04%$ͩ

Slide 33

Slide 33 text

Ұؾʹෳ਺ߦ͕ ฤू͞Εͨ৔߹ Λߟྀ͢Δ

Slide 34

Slide 34 text

ͲͷΑ͏ʹͯ͠ʁ

Slide 35

Slide 35 text

ͲͷΑ͏ʹͯ͠ʁ wฤूͨ͠ൣғΛऔಘ͢Δ wߦຖʹ෼ׂ͢Δ wߦͷશͯͷൣғΛऔಘ͢Δ wม׵ wશମͷஔ׵

Slide 36

Slide 36 text

ͲͷΑ͏ʹͯ͠ʁ wฤूͨ͠ൣғΛऔಘ͢Δ wߦຖʹ෼ׂ͢Δ wߦͷશͯͷൣғΛऔಘ͢Δ wม׵ wશମͷஔ׵ ෳ਺ߦͷ৔߹͕͋Δ

Slide 37

Slide 37 text

ͲͷΑ͏ʹͯ͠ʁ wฤूͨ͠ൣғΛऔಘ͢Δ wߦຖʹ෼ׂ͢Δ wߦͷશͯͷൣғΛऔಘ͢Δ wม׵ wશମͷஔ׵

Slide 38

Slide 38 text

J04%$ J04%$

Slide 39

Slide 39 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩ

Slide 40

Slide 40 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩʂ

Slide 41

Slide 41 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩʂ

Slide 42

Slide 42 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩʂ ໌೔΋J04%$ͩʂ ໌ޙ೔΋J04%$ͩ

Slide 43

Slide 43 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩʂ ໌೔΋J04%$ͩʂ ໌ޙ೔΋J04%$ͩ

Slide 44

Slide 44 text

func rangeOfCurrentLine(location: Int) -> NSRange? { let string = internalTextStorage.string let endNSRange = NSRange(location: location, length: 0) let lineRange: Range if let endRange = endNSRange.range(string: string) { lineRange = string.lineRange(for: endRange) } else { return nil } let lower = String.UTF16View.Index(lineRange.lowerBound, within: string.utf16)! let upper = String.UTF16View.Index(lineRange.upperBound, within: string.utf16)! let lineNSRange = NSRange(lower..

Slide 45

Slide 45 text

func rangeOfCurrentLine(location: Int) -> NSRange? { let string = internalTextStorage.string let endNSRange = NSRange(location: location, length: 0) let lineRange: Range if let endRange = endNSRange.range(string: string) { lineRange = string.lineRange(for: endRange) } else { return nil } let lower = String.UTF16View.Index(lineRange.lowerBound, within: string.utf16)! let upper = String.UTF16View.Index(lineRange.upperBound, within: string.utf16)! let lineNSRange = NSRange(lower..

Slide 46

Slide 46 text

func rangeOfCurrentLine(location: Int) -> NSRange? { let string = internalTextStorage.string let endNSRange = NSRange(location: location, length: 0) let lineRange: Range if let endRange = endNSRange.range(string: string) { lineRange = string.lineRange(for: endRange) } else { return nil } let lower = String.UTF16View.Index(lineRange.lowerBound, within: string.utf16)! let upper = String.UTF16View.Index(lineRange.upperBound, within: string.utf16)! let lineNSRange = NSRange(lower..

Slide 47

Slide 47 text

func rangeOfCurrentLine(location: Int) -> NSRange? { let string = internalTextStorage.string let endNSRange = NSRange(location: location, length: 0) let lineRange: Range if let endRange = endNSRange.range(string: string) { lineRange = string.lineRange(for: endRange) } else { return nil } let lower = String.UTF16View.Index(lineRange.lowerBound, within: string.utf16)! let upper = String.UTF16View.Index(lineRange.upperBound, within: string.utf16)! let lineNSRange = NSRange(lower..

Slide 48

Slide 48 text

func rangeOfCurrentLine(location: Int) -> NSRange? { let string = internalTextStorage.string let endNSRange = NSRange(location: location, length: 0) let lineRange: Range if let endRange = endNSRange.range(string: string) { lineRange = string.lineRange(for: endRange) } else { return nil } let lower = String.UTF16View.Index(lineRange.lowerBound, within: string.utf16)! let upper = String.UTF16View.Index(lineRange.upperBound, within: string.utf16)! let lineNSRange = NSRange(lower..

Slide 49

Slide 49 text

func rangeOfCurrentLine(location: Int) -> NSRange? { let string = internalTextStorage.string let endNSRange = NSRange(location: location, length: 0) let lineRange: Range if let endRange = endNSRange.range(string: string) { lineRange = string.lineRange(for: endRange) } else { return nil } let lower = String.UTF16View.Index(lineRange.lowerBound, within: string.utf16)! let upper = String.UTF16View.Index(lineRange.upperBound, within: string.utf16)! let lineNSRange = NSRange(lower..

Slide 50

Slide 50 text

lineRange(for:) Returns the range of characters representing the line or lines containing a given range. func lineRange(for range: NSRange) -> NSRange ศ ར

Slide 51

Slide 51 text

ͲͷΑ͏ʹͯ͠ʁ wฤूͨ͠ൣғΛऔಘ͢Δ wߦຖʹ෼ׂ͢Δ wߦͷશͯͷൣғΛऔಘ͢Δ wม׵ wશମͷஔ׵

Slide 52

Slide 52 text

ͲͷΑ͏ʹͯ͠ʁ wฤूͨ͠ൣғΛऔಘ͢Δ wߦຖʹ෼ׂ͢Δ wߦͷશͯͷൣғΛऔಘ͢Δ wม׵ wશମͷஔ׵ ίʔυϒϩοΫ͸શ ମΛݕࡧඞཁ͕͋Δ

Slide 53

Slide 53 text

ͲͷΑ͏ʹͯ͠ʁ wฤूͨ͠ൣғΛऔಘ͢Δ wߦຖʹ෼ׂ͢Δ wߦͷશͯͷൣғΛऔಘ͢Δ wม׵ wશମͷஔ׵ ޙ͸֘౰͢ΔจࣈྻΛந ग़ͯ͠ద੾ͳ৭ʹม׵͢ Δ͚ͩɻ

Slide 54

Slide 54 text

࠷খݶͷൣғͰղ ੳ͢Δ͜ͱ͕େ੾

Slide 55

Slide 55 text

No content

Slide 56

Slide 56 text

#$%&

Slide 57

Slide 57 text

ֆจࣈ w/44USJOHɺ/43BOHFͷར༻࣌ w୯७ʹจࣈྻΛΧ΢ϯτͯ͠͸❌ w65'Λҙࣝͨ͠จࣈྻૢ࡞Λ͢Δ

Slide 58

Slide 58 text

No content

Slide 59

Slide 59 text

6' Y%% Y%& 6OJDPEF 65'

Slide 60

Slide 60 text

"཮".count // 1 "཮".unicodeScalars.count // 1 "཮".utf16.count // 1 "཮".utf8.count // 3 "".count // 1 "".unicodeScalars.count // 1 "".utf16.count // 2 "".utf8.count // 4 ")".count // 1 ")".unicodeScalars.count // 2 ")".utf16.count // 4 ")".utf8.count // 8

Slide 61

Slide 61 text

#$%&

Slide 62

Slide 62 text

/44USJOH

Slide 63

Slide 63 text

w಺෦తʹ͸65'Ͱό ΠτྻΛ؅ཧ /44USJOH

Slide 64

Slide 64 text

let snowy = "❄ Let it snow! ☃" let nsrange = NSRange(location: 3, length: 12) if let range = Range(nsrange, in: snowy) { print(snowy[range]) } // Prints "Let it snow!"

Slide 65

Slide 65 text

NSRange(location: 0, length: line.utf16.count)

Slide 66

Slide 66 text

4USJOH

Slide 67

Slide 67 text

wόΠτྻ͸Ӆณ 4USJOH

Slide 68

Slide 68 text

let snowy = "❄ Let it snow! ☃" print(snowy.prefix(5)) // Prints "❄ Let" print(snowy.suffix(1)) // Prints "☃"

Slide 69

Slide 69 text

No content

Slide 70

Slide 70 text

ೖྗิࠤ

Slide 71

Slide 71 text

wΦʔτΠϯσϯτ wλϒͰͷҠಈ

Slide 72

Slide 72 text

No content

Slide 73

Slide 73 text

ΠϕϯτϋϯυϦϯά wλϒ wόοΫλϒ wվߦ

Slide 74

Slide 74 text

J04NBD04

Slide 75

Slide 75 text

w/45FYU7JFX w6*5FYU7JFX

Slide 76

Slide 76 text

/45FYU7JFX

Slide 77

Slide 77 text

No content

Slide 78

Slide 78 text

ઐ༻ͷؔ਺ ͕༻ҙ͞Ε͍ͯΔ

Slide 79

Slide 79 text

class NSTextView { func insertTab(_ sender: Any?) func insertBacktab(_ sender: Any?) func insertNewline(_ sender: Any?) } public protocol NSTextInputClient { public func insertText(_ string: Any, replacementRange: NSRange) } طʹظ଴͞Εͨؔ਺͕༻ҙ͞Ε͍ͯΔ ΧελϚΠζͨ͠จࣈྻΛૠೖ͢Δ

Slide 80

Slide 80 text

class TextView: NS { override func insertTab(_ sender: Any?) { var indent = "" let (spaces, lineRange) = getPrefixStringIfLineHasListTag(regex: TextRegex.insertTab) if useIndentWithSpaceKey { indent = String(repeating: " ", count: indentWidth) if !spaces.isEmpty { super.insertText(indent, replacementRange: NSRange(location: lineRange.location, length: 0)) return } super.insertText(indent, replacementRange: self.rangeForUserTextChange) return } indent = "\t" if !spaces.isEmpty { super.insertText(indent, replacementRange: NSRange(location: lineRange.location, length: 0)) return } super.insertTab(sender) } }

Slide 81

Slide 81 text

class TextView: NS { override func insertTab(_ sender: Any?) { var indent = "" let (spaces, lineRange) = getPrefixStringIfLineHasListTag(regex: TextRegex.insertTab) if useIndentWithSpaceKey { indent = String(repeating: " ", count: indentWidth) if !spaces.isEmpty { super.insertText(indent, replacementRange: NSRange(location: lineRange.location, length: 0)) return } super.insertText(indent, replacementRange: self.rangeForUserTextChange) return } indent = "\t" if !spaces.isEmpty { super.insertText(indent, replacementRange: NSRange(location: lineRange.location, length: 0)) return } super.insertTab(sender) } } Πϯσϯτର৅ͱͳ Δઌ಄෦෼Λऔಘ

Slide 82

Slide 82 text

લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩ લ೔͸લ໷ࡇͩͬͨɻ ࠓ೔͸J04%$ͩ

Slide 83

Slide 83 text

class TextView: NS { override func insertTab(_ sender: Any?) { var indent = "" let (spaces, lineRange) = getPrefixStringIfLineHasListTag(regex: TextRegex.insertTab) if useIndentWithSpaceKey { indent = String(repeating: " ", count: indentWidth) if !spaces.isEmpty { super.insertText(indent, replacementRange: NSRange(location: lineRange.location, length: 0)) return } super.insertText(indent, replacementRange: self.rangeForUserTextChange) return } indent = "\t" if !spaces.isEmpty { super.insertText(indent, replacementRange: NSRange(location: lineRange.location, length: 0)) return } super.insertTab(sender) } } ਌ॲཧʹ೚ͤΔ

Slide 84

Slide 84 text

6*5FYU7JFX

Slide 85

Slide 85 text

No content

Slide 86

Slide 86 text

class NSTextView { func insertTab(_ sender: Any?) func insertBacktab(_ sender: Any?) func insertNewline(_ sender: Any?) } public protocol NSTextInputClient { public func insertText(_ string: Any, replacementRange: NSRange) } ༻ҙ͞Ε͍ͯͳ͍ɻɻɻ

Slide 87

Slide 87 text

class NSTextView { func insertTab(_ sender: Any?) func insertBacktab(_ sender: Any?) func insertNewline(_ sender: Any?) } public protocol NSTextInputClient { public func insertText(_ string: Any, replacementRange: NSRange) } • ΩʔϘʔυʹλϒ͸ແ͍ • λϒϘλϯͰಉ౳ͷػೳΛ࣮૷

Slide 88

Slide 88 text

insertTab insertBackTab

Slide 89

Slide 89 text

ςΩετૠೖ࣌

Slide 90

Slide 90 text

6*5FYU7JFXͷࢠΫϥε let stringRange = lineRange.range(string: text)! text.insert(contentsOf: indent, at: stringRange.lowerBound)

Slide 91

Slide 91 text

6*5FYU7JFXͷࢠΫϥε let stringRange = lineRange.range(string: text)! text.insert(contentsOf: indent, at: stringRange.lowerBound)

Slide 92

Slide 92 text

UITextViewͷࢠΫϥε let stringRange = lineRange.range(string: text)! text.insert(contentsOf: indent, at: stringRange.lowerBound) wFEJUFE3BOHFͰฤूൣғ͕શͯʹͳͬ ͯ͠·͏ wUFYU%JE$IBOHF͸ίʔϧ͞Εͳ͍ wظ଴͞ΕͨҐஔʹΧʔιϧ͕Ҡಈ͞Ε ͳ͍

Slide 93

Slide 93 text

public protocol UIKeyInput : UITextInputTraits { public var hasText: Bool { get } public func insertText(_ text: String) public func deleteBackward() }

Slide 94

Slide 94 text

func insertTab(rangeForTextChange: NSRange) { var indent = "" let (spaces, lineRange) = getPrefixStringIfLineHasList(regex: TextInputRegex.insertTab, rangeForTextChange: rangeForTextChange) if useIndentWithSpaceKey { indent = String(repeating: " ", count: indentWidth) if !spaces.isEmpty { selectedRange = NSRange(location: lineRange.location, length: 0) insertText(indent) selectedRange = NSRange(location: rangeForTextChange.location + indentWidth, length: 0) return } super.insertText(indent) return } indent = "\t" if !spaces.isEmpty { selectedRange = NSRange(location: lineRange.location, length: 0) insertText(indent) selectedRange = NSRange(location: rangeForTextChange.location + 1, length: 0) return } }

Slide 95

Slide 95 text

func insertTab(rangeForTextChange: NSRange) { var indent = "" let (spaces, lineRange) = getPrefixStringIfLineHasList(regex: TextInputRegex.insertTab, rangeForTextChange: rangeForTextChange) if useIndentWithSpaceKey { indent = String(repeating: " ", count: indentWidth) if !spaces.isEmpty { selectedRange = NSRange(location: lineRange.location, length: 0) insertText(indent) selectedRange = NSRange(location: rangeForTextChange.location + indentWidth, length: 0) return } super.insertText(indent) return } indent = "\t" if !spaces.isEmpty { selectedRange = NSRange(location: lineRange.location, length: 0) insertText(indent) selectedRange = NSRange(location: rangeForTextChange.location + 1, length: 0) return } } Χʔιϧ͕ظ଴ͨ͠Ґஔʹ͍ͳ͍ͷͰઌ಄ʹҠಈɺૠೖޙɺ ૠೖޙͷҐஔʹҠಈ͢Δඞཁ͕͋Δɻ

Slide 96

Slide 96 text

·ͱΊ

Slide 97

Slide 97 text

·ͱΊ wղੳ͸ͳΔ΂͘࠷খݶͷൣғͰղੳ͢Δ

Slide 98

Slide 98 text

·ͱΊ wղੳ͸ͳΔ΂͘࠷খݶͷൣғͰߦ͏ wจࣈίʔυΛҙࣝͨ͠จࣈྻૢ࡞ w⚠/44USJOH /43BOHF

Slide 99

Slide 99 text

·ͱΊ wղੳ͸ͳΔ΂͘࠷খݶͷൣғͰߦ͏ wจࣈίʔυΛҙࣝͨ͠จࣈྻૢ࡞ w⚠/44USJOH /43BOHF wೖྗิࠤ࣌ͷJ04Ͱͷઃܭ wΧελϚΠζ͕ඞਢ

Slide 100

Slide 100 text

5IBOLT

Slide 101

Slide 101 text

&OKPZ