作者 吴孟雨

登录注册,绑定手机号,获取验证码,轮播图,本周实验,往期实验,立即预约,实验详情

1 <?xml version="1.0" encoding="UTF-8"?> 1 <?xml version="1.0" encoding="UTF-8"?>
2 <project version="4"> 2 <project version="4">
3 <component name="ChangeListManager"> 3 <component name="ChangeListManager">
4 - <list default="true" id="632eecf6-91e3-4185-aea4-06d243e51935" name="Default" comment="上传到远程wumengyu"> 4 + <list default="true" id="632eecf6-91e3-4185-aea4-06d243e51935" name="Default" comment="合并master">
  5 + <change afterPath="$PROJECT_DIR$/utils/qqmap-wx-jssdk.min.js" afterDir="false" />
  6 + <change afterPath="$PROJECT_DIR$/wxParse/html2json.js" afterDir="false" />
  7 + <change afterPath="$PROJECT_DIR$/wxParse/htmlparser.js" afterDir="false" />
  8 + <change afterPath="$PROJECT_DIR$/wxParse/showdown.js" afterDir="false" />
  9 + <change afterPath="$PROJECT_DIR$/wxParse/wxDiscode.js" afterDir="false" />
  10 + <change afterPath="$PROJECT_DIR$/wxParse/wxParse.js" afterDir="false" />
  11 + <change afterPath="$PROJECT_DIR$/wxParse/wxParse.wxml" afterDir="false" />
  12 + <change afterPath="$PROJECT_DIR$/wxParse/wxParse.wxss" afterDir="false" />
5 <change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" /> 13 <change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" />
6 <change beforePath="$PROJECT_DIR$/app.js" beforeDir="false" afterPath="$PROJECT_DIR$/app.js" afterDir="false" /> 14 <change beforePath="$PROJECT_DIR$/app.js" beforeDir="false" afterPath="$PROJECT_DIR$/app.js" afterDir="false" />
7 - <change beforePath="$PROJECT_DIR$/app.json" beforeDir="false" afterPath="$PROJECT_DIR$/app.json" afterDir="false" /> 15 + <change beforePath="$PROJECT_DIR$/pages/clock/clock.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/clock/clock.js" afterDir="false" />
  16 + <change beforePath="$PROJECT_DIR$/pages/clock/clock.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/clock/clock.wxml" afterDir="false" />
  17 + <change beforePath="$PROJECT_DIR$/pages/clock/clock.wxss" beforeDir="false" afterPath="$PROJECT_DIR$/pages/clock/clock.wxss" afterDir="false" />
8 <change beforePath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js" afterDir="false" /> 18 <change beforePath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js" afterDir="false" />
9 <change beforePath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml" afterDir="false" /> 19 <change beforePath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml" afterDir="false" />
  20 + <change beforePath="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js" afterDir="false" />
  21 + <change beforePath="$PROJECT_DIR$/pages/index/go-appointment/go-appointment.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/go-appointment/go-appointment.js" afterDir="false" />
  22 + <change beforePath="$PROJECT_DIR$/pages/index/go-appointment/go-appointment.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/go-appointment/go-appointment.wxml" afterDir="false" />
10 <change beforePath="$PROJECT_DIR$/pages/index/index.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/index.js" afterDir="false" /> 23 <change beforePath="$PROJECT_DIR$/pages/index/index.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/index.js" afterDir="false" />
11 <change beforePath="$PROJECT_DIR$/pages/index/index.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/index.wxml" afterDir="false" /> 24 <change beforePath="$PROJECT_DIR$/pages/index/index.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/index.wxml" afterDir="false" />
  25 + <change beforePath="$PROJECT_DIR$/pages/index/phone-code/phone-code.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/phone-code/phone-code.js" afterDir="false" />
  26 + <change beforePath="$PROJECT_DIR$/pages/index/phone-code/phone-code.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/index/phone-code/phone-code.wxml" afterDir="false" />
  27 + <change beforePath="$PROJECT_DIR$/pages/my/my.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/my/my.js" afterDir="false" />
  28 + <change beforePath="$PROJECT_DIR$/pages/my/my.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/my/my.wxml" afterDir="false" />
  29 + <change beforePath="$PROJECT_DIR$/pages/my/my.wxss" beforeDir="false" afterPath="$PROJECT_DIR$/pages/my/my.wxss" afterDir="false" />
  30 + <change beforePath="$PROJECT_DIR$/pages/my/protocol/protocol.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/my/protocol/protocol.js" afterDir="false" />
  31 + <change beforePath="$PROJECT_DIR$/pages/my/protocol/protocol.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/pages/my/protocol/protocol.wxml" afterDir="false" />
12 <change beforePath="$PROJECT_DIR$/pages/start/start.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/start/start.js" afterDir="false" /> 32 <change beforePath="$PROJECT_DIR$/pages/start/start.js" beforeDir="false" afterPath="$PROJECT_DIR$/pages/start/start.js" afterDir="false" />
13 <change beforePath="$PROJECT_DIR$/project.config.json" beforeDir="false" afterPath="$PROJECT_DIR$/project.config.json" afterDir="false" /> 33 <change beforePath="$PROJECT_DIR$/project.config.json" beforeDir="false" afterPath="$PROJECT_DIR$/project.config.json" afterDir="false" />
14 - <change beforePath="$PROJECT_DIR$/templates/templates.wxml" beforeDir="false" afterPath="$PROJECT_DIR$/templates/templates.wxml" afterDir="false" /> 34 + <change beforePath="$PROJECT_DIR$/utils/util.js" beforeDir="false" afterPath="$PROJECT_DIR$/utils/util.js" afterDir="false" />
15 </list> 35 </list>
16 <ignored path="$PROJECT_DIR$/.tmp/" /> 36 <ignored path="$PROJECT_DIR$/.tmp/" />
17 <ignored path="$PROJECT_DIR$/temp/" /> 37 <ignored path="$PROJECT_DIR$/temp/" />
@@ -26,44 +46,49 @@ @@ -26,44 +46,49 @@
26 <session id="-381596063"> 46 <session id="-381596063">
27 <usages-collector id="statistics.lifecycle.project"> 47 <usages-collector id="statistics.lifecycle.project">
28 <counts> 48 <counts>
29 - <entry key="project.closed" value="3" /> 49 + <entry key="project.closed" value="5" />
30 <entry key="project.open.time.0" value="3" /> 50 <entry key="project.open.time.0" value="3" />
31 - <entry key="project.open.time.1" value="1" />  
32 - <entry key="project.opened" value="4" /> 51 + <entry key="project.open.time.1" value="3" />
  52 + <entry key="project.opened" value="6" />
33 </counts> 53 </counts>
34 </usages-collector> 54 </usages-collector>
35 <usages-collector id="statistics.file.extensions.open"> 55 <usages-collector id="statistics.file.extensions.open">
36 <counts> 56 <counts>
37 - <entry key="js" value="6" />  
38 - <entry key="json" value="4" />  
39 - <entry key="wxml" value="9" />  
40 - <entry key="wxss" value="5" /> 57 + <entry key="js" value="25" />
  58 + <entry key="json" value="11" />
  59 + <entry key="wxml" value="21" />
  60 + <entry key="wxss" value="10" />
41 </counts> 61 </counts>
42 </usages-collector> 62 </usages-collector>
43 <usages-collector id="statistics.file.types.open"> 63 <usages-collector id="statistics.file.types.open">
44 <counts> 64 <counts>
45 - <entry key="CSS" value="5" />  
46 - <entry key="HTML" value="9" />  
47 - <entry key="JSON" value="4" />  
48 - <entry key="JavaScript" value="6" /> 65 + <entry key="CSS" value="10" />
  66 + <entry key="HTML" value="21" />
  67 + <entry key="JSON" value="11" />
  68 + <entry key="JavaScript" value="25" />
49 </counts> 69 </counts>
50 </usages-collector> 70 </usages-collector>
51 <usages-collector id="statistics.file.extensions.edit"> 71 <usages-collector id="statistics.file.extensions.edit">
52 <counts> 72 <counts>
53 - <entry key="js" value="564" />  
54 - <entry key="json" value="27" />  
55 - <entry key="txt" value="36" />  
56 - <entry key="wxml" value="244" />  
57 - <entry key="wxss" value="21" /> 73 + <entry key="js" value="3215" />
  74 + <entry key="json" value="34" />
  75 + <entry key="txt" value="63" />
  76 + <entry key="wxml" value="748" />
  77 + <entry key="wxss" value="23" />
58 </counts> 78 </counts>
59 </usages-collector> 79 </usages-collector>
60 <usages-collector id="statistics.file.types.edit"> 80 <usages-collector id="statistics.file.types.edit">
61 <counts> 81 <counts>
62 - <entry key="CSS" value="21" />  
63 - <entry key="HTML" value="244" />  
64 - <entry key="JSON" value="27" />  
65 - <entry key="JavaScript" value="564" />  
66 - <entry key="PLAIN_TEXT" value="36" /> 82 + <entry key="CSS" value="23" />
  83 + <entry key="HTML" value="748" />
  84 + <entry key="JSON" value="34" />
  85 + <entry key="JavaScript" value="3215" />
  86 + <entry key="PLAIN_TEXT" value="63" />
  87 + </counts>
  88 + </usages-collector>
  89 + <usages-collector id="statistics.vcs.git.usages">
  90 + <counts>
  91 + <entry key="git.branch.merge" value="1" />
67 </counts> 92 </counts>
68 </usages-collector> 93 </usages-collector>
69 </session> 94 </session>
@@ -71,100 +96,104 @@ @@ -71,100 +96,104 @@
71 <component name="FileEditorManager"> 96 <component name="FileEditorManager">
72 <leaf SIDE_TABS_SIZE_LIMIT_KEY="300"> 97 <leaf SIDE_TABS_SIZE_LIMIT_KEY="300">
73 <file pinned="false" current-in-tab="false"> 98 <file pinned="false" current-in-tab="false">
74 - <entry file="file://$PROJECT_DIR$/pages/index/index.wxml"> 99 + <entry file="file://$PROJECT_DIR$/pages/index/index.js">
75 <provider selected="true" editor-type-id="text-editor"> 100 <provider selected="true" editor-type-id="text-editor">
76 - <state relative-caret-position="880">  
77 - <caret line="40" lean-forward="true" selection-start-line="40" selection-end-line="40" /> 101 + <state relative-caret-position="-418">
  102 + <caret line="8" column="23" selection-start-line="8" selection-start-column="23" selection-end-line="8" selection-end-column="23" />
  103 + <folding>
  104 + <element signature="n#!!doc" expanded="true" />
  105 + </folding>
78 </state> 106 </state>
79 </provider> 107 </provider>
80 </entry> 108 </entry>
81 </file> 109 </file>
82 <file pinned="false" current-in-tab="false"> 110 <file pinned="false" current-in-tab="false">
83 - <entry file="file://$PROJECT_DIR$/pages/index/index.wxss"> 111 + <entry file="file://$PROJECT_DIR$/pages/start/start.js">
84 <provider selected="true" editor-type-id="text-editor"> 112 <provider selected="true" editor-type-id="text-editor">
85 - <state relative-caret-position="154">  
86 - <caret line="7" column="21" lean-forward="true" selection-start-line="7" selection-start-column="21" selection-end-line="7" selection-end-column="21" /> 113 + <state relative-caret-position="676">
  114 + <caret line="37" column="25" lean-forward="true" selection-start-line="37" selection-start-column="25" selection-end-line="37" selection-end-column="25" />
87 </state> 115 </state>
88 </provider> 116 </provider>
89 </entry> 117 </entry>
90 </file> 118 </file>
91 <file pinned="false" current-in-tab="false"> 119 <file pinned="false" current-in-tab="false">
92 - <entry file="file://$PROJECT_DIR$/pages/index/index.js"> 120 + <entry file="file://$PROJECT_DIR$/app.js">
93 <provider selected="true" editor-type-id="text-editor"> 121 <provider selected="true" editor-type-id="text-editor">
94 - <state relative-caret-position="506">  
95 - <caret line="98" column="64" selection-start-line="98" selection-start-column="64" selection-end-line="98" selection-end-column="64" /> 122 + <state relative-caret-position="344">
  123 + <caret line="140" column="12" selection-start-line="140" selection-start-column="4" selection-end-line="140" selection-end-column="12" />
96 <folding> 124 <folding>
97 <element signature="n#!!doc" expanded="true" /> 125 <element signature="n#!!doc" expanded="true" />
98 - <element signature="e#1018#1103#0" />  
99 - <element signature="e#1121#1191#0" />  
100 - <element signature="e#1219#1288#0" />  
101 - <element signature="e#1321#1468#0" />  
102 - <element signature="e#1499#1619#0" />  
103 - <element signature="e#1750#1766#0" />  
104 </folding> 126 </folding>
105 </state> 127 </state>
106 </provider> 128 </provider>
107 </entry> 129 </entry>
108 </file> 130 </file>
109 <file pinned="false" current-in-tab="false"> 131 <file pinned="false" current-in-tab="false">
110 - <entry file="file://$PROJECT_DIR$/templates/templates.wxml"> 132 + <entry file="file://$PROJECT_DIR$/utils/util.js">
111 <provider selected="true" editor-type-id="text-editor"> 133 <provider selected="true" editor-type-id="text-editor">
112 - <state relative-caret-position="374">  
113 - <caret line="29" column="65" selection-start-line="29" selection-start-column="65" selection-end-line="29" selection-end-column="65" /> 134 + <state relative-caret-position="946">
  135 + <caret line="43" column="15" selection-start-line="43" selection-start-column="2" selection-end-line="43" selection-end-column="15" />
114 </state> 136 </state>
115 </provider> 137 </provider>
116 </entry> 138 </entry>
117 </file> 139 </file>
118 - <file pinned="false" current-in-tab="true">  
119 - <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml"> 140 + <file pinned="false" current-in-tab="false">
  141 + <entry file="file://$PROJECT_DIR$/pages/index/go-appointment/go-appointment.wxml">
120 <provider selected="true" editor-type-id="text-editor"> 142 <provider selected="true" editor-type-id="text-editor">
121 - <state relative-caret-position="198">  
122 - <caret line="9" column="35" selection-start-line="9" selection-start-column="35" selection-end-line="9" selection-end-column="35" /> 143 + <state relative-caret-position="638">
  144 + <caret line="41" column="90" selection-start-line="41" selection-start-column="72" selection-end-line="41" selection-end-column="90" />
123 </state> 145 </state>
124 </provider> 146 </provider>
125 </entry> 147 </entry>
126 </file> 148 </file>
127 <file pinned="false" current-in-tab="false"> 149 <file pinned="false" current-in-tab="false">
128 - <entry file="file://$PROJECT_DIR$/pages/start/start.wxml"> 150 + <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js">
129 <provider selected="true" editor-type-id="text-editor"> 151 <provider selected="true" editor-type-id="text-editor">
130 - <state relative-caret-position="132">  
131 - <caret line="6" column="15" lean-forward="true" selection-start-line="6" selection-start-column="15" selection-end-line="6" selection-end-column="15" /> 152 + <state relative-caret-position="374">
  153 + <caret line="29" column="44" lean-forward="true" selection-start-line="29" selection-start-column="44" selection-end-line="29" selection-end-column="44" />
132 </state> 154 </state>
133 </provider> 155 </provider>
134 </entry> 156 </entry>
135 </file> 157 </file>
136 - <file pinned="false" current-in-tab="false">  
137 - <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.json">  
138 - <provider selected="true" editor-type-id="text-editor" />  
139 - </entry>  
140 - </file>  
141 - <file pinned="false" current-in-tab="false">  
142 - <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js"> 158 + <file pinned="false" current-in-tab="true">
  159 + <entry file="file://$PROJECT_DIR$/pages/index/go-appointment/go-appointment.js">
143 <provider selected="true" editor-type-id="text-editor"> 160 <provider selected="true" editor-type-id="text-editor">
144 - <state relative-caret-position="550">  
145 - <caret line="25" column="32" selection-start-line="25" selection-start-column="32" selection-end-line="25" selection-end-column="32" /> 161 + <state relative-caret-position="198">
  162 + <caret line="125" column="20" lean-forward="true" selection-start-line="125" selection-start-column="20" selection-end-line="125" selection-end-column="20" />
  163 + <folding>
  164 + <element signature="e#243#454#0" />
  165 + <element signature="e#462#673#0" />
  166 + <element signature="e#681#893#0" />
  167 + <element signature="e#901#1114#0" />
  168 + <element signature="e#1177#2298#0" />
  169 + </folding>
146 </state> 170 </state>
147 </provider> 171 </provider>
148 </entry> 172 </entry>
149 </file> 173 </file>
150 <file pinned="false" current-in-tab="false"> 174 <file pinned="false" current-in-tab="false">
151 - <entry file="file://$PROJECT_DIR$/pages/start/start.js"> 175 + <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js">
152 <provider selected="true" editor-type-id="text-editor"> 176 <provider selected="true" editor-type-id="text-editor">
153 - <state relative-caret-position="374">  
154 - <caret line="29" column="49" lean-forward="true" selection-start-line="29" selection-start-column="12" selection-end-line="29" selection-end-column="49" /> 177 + <state relative-caret-position="302">
  178 + <caret line="44" column="20" lean-forward="true" selection-start-line="44" selection-start-column="20" selection-end-line="44" selection-end-column="20" />
155 </state> 179 </state>
156 </provider> 180 </provider>
157 </entry> 181 </entry>
158 </file> 182 </file>
159 <file pinned="false" current-in-tab="false"> 183 <file pinned="false" current-in-tab="false">
160 - <entry file="file://$PROJECT_DIR$/app.json"> 184 + <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxml">
161 <provider selected="true" editor-type-id="text-editor"> 185 <provider selected="true" editor-type-id="text-editor">
162 - <state relative-caret-position="440">  
163 - <caret line="20" column="38" selection-start-line="20" selection-start-column="38" selection-end-line="20" selection-end-column="38" /> 186 + <state relative-caret-position="852">
  187 + <caret line="50" column="42" selection-start-line="50" selection-start-column="42" selection-end-line="50" selection-end-column="42" />
164 </state> 188 </state>
165 </provider> 189 </provider>
166 </entry> 190 </entry>
167 </file> 191 </file>
  192 + <file pinned="false" current-in-tab="false">
  193 + <entry file="file://$PROJECT_DIR$/pages/index/go-appointment/go-appointment.json">
  194 + <provider selected="true" editor-type-id="text-editor" />
  195 + </entry>
  196 + </file>
168 </leaf> 197 </leaf>
169 </component> 198 </component>
170 <component name="FindInProjectRecents"> 199 <component name="FindInProjectRecents">
@@ -172,6 +201,21 @@ @@ -172,6 +201,21 @@
172 <find>chooseYes</find> 201 <find>chooseYes</find>
173 <find>baseUrl</find> 202 <find>baseUrl</find>
174 <find>getUserInfo</find> 203 <find>getUserInfo</find>
  204 + <find>协议</find>
  205 + <find>serviceProtocol</find>
  206 + <find>goTestDetail</find>
  207 + <find>modal</find>
  208 + <find>goPhoneCode</find>
  209 + <find>QQMapWX</find>
  210 + <find>XX</find>
  211 + <find>showM</find>
  212 + <find>goAppointment</find>
  213 + <find>bindPickerChange</find>
  214 + <find>show_submit_btn</find>
  215 + <find>submit</find>
  216 + <find>current_test_id</find>
  217 + <find>current_session_id</find>
  218 + <find>currentId</find>
175 </findStrings> 219 </findStrings>
176 </component> 220 </component>
177 <component name="Git.Settings"> 221 <component name="Git.Settings">
@@ -182,16 +226,30 @@ @@ -182,16 +226,30 @@
182 <list> 226 <list>
183 <option value="$PROJECT_DIR$/templates/templates.wxss" /> 227 <option value="$PROJECT_DIR$/templates/templates.wxss" />
184 <option value="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxss" /> 228 <option value="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxss" />
185 - <option value="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js" />  
186 <option value="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxml" /> 229 <option value="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxml" />
187 - <option value="$PROJECT_DIR$/app.js" />  
188 - <option value="$PROJECT_DIR$/pages/start/start.js" />  
189 - <option value="$PROJECT_DIR$/pages/index/index.wxml" />  
190 <option value="$PROJECT_DIR$/templates/templates.wxml" /> 230 <option value="$PROJECT_DIR$/templates/templates.wxml" />
191 - <option value="$PROJECT_DIR$/pages/index/index.js" /> 231 + <option value="$PROJECT_DIR$/pages/my/protocol/protocol.wxml" />
  232 + <option value="$PROJECT_DIR$/pages/my/protocol/protocol.js" />
  233 + <option value="$PROJECT_DIR$/project.config.json" />
  234 + <option value="$PROJECT_DIR$/pages/start/start.js" />
  235 + <option value="$PROJECT_DIR$/pages/clock/clock.wxml" />
  236 + <option value="$PROJECT_DIR$/pages/my/my.wxml" />
  237 + <option value="$PROJECT_DIR$/pages/clock/clock.wxss" />
  238 + <option value="$PROJECT_DIR$/pages/my/my.wxss" />
  239 + <option value="$PROJECT_DIR$/pages/my/my.js" />
  240 + <option value="$PROJECT_DIR$/pages/clock/clock.js" />
192 <option value="$PROJECT_DIR$/app.json" /> 241 <option value="$PROJECT_DIR$/app.json" />
193 - <option value="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js" /> 242 + <option value="$PROJECT_DIR$/pages/index/phone-code/phone-code.wxml" />
  243 + <option value="$PROJECT_DIR$/pages/index/phone-code/phone-code.js" />
  244 + <option value="$PROJECT_DIR$/pages/index/index.js" />
  245 + <option value="$PROJECT_DIR$/pages/index/index.wxml" />
194 <option value="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml" /> 246 <option value="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml" />
  247 + <option value="$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js" />
  248 + <option value="$PROJECT_DIR$/app.js" />
  249 + <option value="$PROJECT_DIR$/utils/util.js" />
  250 + <option value="$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js" />
  251 + <option value="$PROJECT_DIR$/pages/index/go-appointment/go-appointment.wxml" />
  252 + <option value="$PROJECT_DIR$/pages/index/go-appointment/go-appointment.js" />
195 </list> 253 </list>
196 </option> 254 </option>
197 </component> 255 </component>
@@ -243,7 +301,14 @@ @@ -243,7 +301,14 @@
243 <item name="ScienceCaptain" type="462c0819:PsiDirectoryNode" /> 301 <item name="ScienceCaptain" type="462c0819:PsiDirectoryNode" />
244 <item name="pages" type="462c0819:PsiDirectoryNode" /> 302 <item name="pages" type="462c0819:PsiDirectoryNode" />
245 <item name="index" type="462c0819:PsiDirectoryNode" /> 303 <item name="index" type="462c0819:PsiDirectoryNode" />
246 - <item name="phone-code" type="462c0819:PsiDirectoryNode" /> 304 + <item name="confirm-appointment-info" type="462c0819:PsiDirectoryNode" />
  305 + </path>
  306 + <path>
  307 + <item name="ScienceCaptain" type="b2602c69:ProjectViewProjectNode" />
  308 + <item name="ScienceCaptain" type="462c0819:PsiDirectoryNode" />
  309 + <item name="pages" type="462c0819:PsiDirectoryNode" />
  310 + <item name="index" type="462c0819:PsiDirectoryNode" />
  311 + <item name="go-appointment" type="462c0819:PsiDirectoryNode" />
247 </path> 312 </path>
248 <path> 313 <path>
249 <item name="ScienceCaptain" type="b2602c69:ProjectViewProjectNode" /> 314 <item name="ScienceCaptain" type="b2602c69:ProjectViewProjectNode" />
@@ -254,7 +319,7 @@ @@ -254,7 +319,7 @@
254 <path> 319 <path>
255 <item name="ScienceCaptain" type="b2602c69:ProjectViewProjectNode" /> 320 <item name="ScienceCaptain" type="b2602c69:ProjectViewProjectNode" />
256 <item name="ScienceCaptain" type="462c0819:PsiDirectoryNode" /> 321 <item name="ScienceCaptain" type="462c0819:PsiDirectoryNode" />
257 - <item name="templates" type="462c0819:PsiDirectoryNode" /> 322 + <item name="utils" type="462c0819:PsiDirectoryNode" />
258 </path> 323 </path>
259 </expand> 324 </expand>
260 <select /> 325 <select />
@@ -264,11 +329,17 @@ @@ -264,11 +329,17 @@
264 </component> 329 </component>
265 <component name="PropertiesComponent"> 330 <component name="PropertiesComponent">
266 <property name="WebServerToolWindowFactoryState" value="false" /> 331 <property name="WebServerToolWindowFactoryState" value="false" />
267 - <property name="last_opened_file_path" value="D:/project/life" /> 332 + <property name="last_opened_file_path" value="D:/info/informationPlatform" />
268 <property name="nodejs_interpreter_path.stuck_in_default_project" value="undefined stuck path" /> 333 <property name="nodejs_interpreter_path.stuck_in_default_project" value="undefined stuck path" />
269 <property name="nodejs_npm_path_reset_for_default_project" value="true" /> 334 <property name="nodejs_npm_path_reset_for_default_project" value="true" />
270 <property name="settings.editor.selected.configurable" value="project.propDebugger" /> 335 <property name="settings.editor.selected.configurable" value="project.propDebugger" />
271 </component> 336 </component>
  337 + <component name="RecentsManager">
  338 + <key name="CopyFile.RECENT_KEYS">
  339 + <recent name="D:\小程序\scienceCaptain\ScienceCaptain" />
  340 + <recent name="D:\小程序\scienceCaptain\ScienceCaptain\utils" />
  341 + </key>
  342 + </component>
272 <component name="RunDashboard"> 343 <component name="RunDashboard">
273 <option name="ruleStates"> 344 <option name="ruleStates">
274 <list> 345 <list>
@@ -295,7 +366,9 @@ @@ -295,7 +366,9 @@
295 <workItem from="1542249556873" duration="6568000" /> 366 <workItem from="1542249556873" duration="6568000" />
296 <workItem from="1542594514355" duration="2482000" /> 367 <workItem from="1542594514355" duration="2482000" />
297 <workItem from="1542676616270" duration="2495000" /> 368 <workItem from="1542676616270" duration="2495000" />
298 - <workItem from="1543297125303" duration="4846000" /> 369 + <workItem from="1543297125303" duration="6704000" />
  370 + <workItem from="1543303923342" duration="8358000" />
  371 + <workItem from="1543367358458" duration="25214000" />
299 </task> 372 </task>
300 <task id="LOCAL-00001" summary="modal 模板"> 373 <task id="LOCAL-00001" summary="modal 模板">
301 <created>1542274356914</created> 374 <created>1542274356914</created>
@@ -311,11 +384,25 @@ @@ -311,11 +384,25 @@
311 <option name="project" value="LOCAL" /> 384 <option name="project" value="LOCAL" />
312 <updated>1542275805566</updated> 385 <updated>1542275805566</updated>
313 </task> 386 </task>
314 - <option name="localTasksCounter" value="3" /> 387 + <task id="LOCAL-00003" summary="对接本周实验,往期实验,登录接口">
  388 + <created>1543302198978</created>
  389 + <option name="number" value="00003" />
  390 + <option name="presentableId" value="LOCAL-00003" />
  391 + <option name="project" value="LOCAL" />
  392 + <updated>1543302198978</updated>
  393 + </task>
  394 + <task id="LOCAL-00004" summary="合并master">
  395 + <created>1543303397807</created>
  396 + <option name="number" value="00004" />
  397 + <option name="presentableId" value="LOCAL-00004" />
  398 + <option name="project" value="LOCAL" />
  399 + <updated>1543303397807</updated>
  400 + </task>
  401 + <option name="localTasksCounter" value="5" />
315 <servers /> 402 <servers />
316 </component> 403 </component>
317 <component name="TimeTrackingManager"> 404 <component name="TimeTrackingManager">
318 - <option name="totallyTimeSpent" value="16627000" /> 405 + <option name="totallyTimeSpent" value="52057000" />
319 </component> 406 </component>
320 <component name="ToolWindowManager"> 407 <component name="ToolWindowManager">
321 <frame x="-8" y="-8" width="1936" height="1056" extended-state="6" /> 408 <frame x="-8" y="-8" width="1936" height="1056" extended-state="6" />
@@ -375,7 +462,9 @@ @@ -375,7 +462,9 @@
375 <component name="VcsManagerConfiguration"> 462 <component name="VcsManagerConfiguration">
376 <MESSAGE value="modal 模板" /> 463 <MESSAGE value="modal 模板" />
377 <MESSAGE value="上传到远程wumengyu" /> 464 <MESSAGE value="上传到远程wumengyu" />
378 - <option name="LAST_COMMIT_MESSAGE" value="上传到远程wumengyu" /> 465 + <MESSAGE value="对接本周实验,往期实验,登录接口" />
  466 + <MESSAGE value="合并master" />
  467 + <option name="LAST_COMMIT_MESSAGE" value="合并master" />
379 </component> 468 </component>
380 <component name="editorHistoryManager"> 469 <component name="editorHistoryManager">
381 <entry file="file://$PROJECT_DIR$/pages/myProfile/myProfile.json" /> 470 <entry file="file://$PROJECT_DIR$/pages/myProfile/myProfile.json" />
@@ -390,160 +479,255 @@ @@ -390,160 +479,255 @@
390 <entry file="file://$PROJECT_DIR$/pages/my/all-appointment/all-appointment.wxml"> 479 <entry file="file://$PROJECT_DIR$/pages/my/all-appointment/all-appointment.wxml">
391 <provider selected="true" editor-type-id="text-editor" /> 480 <provider selected="true" editor-type-id="text-editor" />
392 </entry> 481 </entry>
393 - <entry file="file://$PROJECT_DIR$/pages/my/protocol/protocol.wxml"> 482 + <entry file="file://$PROJECT_DIR$/templates/templates.wxss">
  483 + <provider selected="true" editor-type-id="text-editor">
  484 + <state>
  485 + <caret column="34" selection-start-column="34" selection-end-column="34" />
  486 + </state>
  487 + </provider>
  488 + </entry>
  489 + <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxss">
  490 + <provider selected="true" editor-type-id="text-editor">
  491 + <state relative-caret-position="462">
  492 + <caret line="21" column="6" selection-start-line="21" selection-start-column="6" selection-end-line="21" selection-end-column="6" />
  493 + </state>
  494 + </provider>
  495 + </entry>
  496 + <entry file="file://$PROJECT_DIR$/pages/my/myCard/myCard.wxml">
394 <provider selected="true" editor-type-id="text-editor" /> 497 <provider selected="true" editor-type-id="text-editor" />
395 </entry> 498 </entry>
396 - <entry file="file://$PROJECT_DIR$/project.config.json"> 499 + <entry file="file://$PROJECT_DIR$/pages/start/start.wxml">
397 <provider selected="true" editor-type-id="text-editor"> 500 <provider selected="true" editor-type-id="text-editor">
398 - <state relative-caret-position="352">  
399 - <caret line="16" column="21" selection-start-line="16" selection-start-column="21" selection-end-line="16" selection-end-column="21" /> 501 + <state relative-caret-position="132">
  502 + <caret line="6" column="15" selection-start-line="6" selection-start-column="15" selection-end-line="6" selection-end-column="15" />
400 </state> 503 </state>
401 </provider> 504 </provider>
402 </entry> 505 </entry>
403 - <entry file="file://$PROJECT_DIR$/templates/templates.wxss"> 506 + <entry file="file://$PROJECT_DIR$/templates/templates.wxml">
404 <provider selected="true" editor-type-id="text-editor"> 507 <provider selected="true" editor-type-id="text-editor">
405 - <state>  
406 - <caret column="34" selection-start-column="34" selection-end-column="34" /> 508 + <state relative-caret-position="242">
  509 + <caret line="11" column="57" selection-start-line="11" selection-start-column="32" selection-end-line="11" selection-end-column="57" />
407 </state> 510 </state>
408 </provider> 511 </provider>
409 </entry> 512 </entry>
410 - <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.js"> 513 + <entry file="file://$PROJECT_DIR$/pages/my/groupBuy/groupBuy.wxml">
411 <provider selected="true" editor-type-id="text-editor"> 514 <provider selected="true" editor-type-id="text-editor">
412 - <state relative-caret-position="550">  
413 - <caret line="25" column="28" selection-start-line="25" selection-start-column="28" selection-end-line="25" selection-end-column="28" /> 515 + <state relative-caret-position="2684">
  516 + <caret line="122" column="59" selection-start-line="122" selection-start-column="44" selection-end-line="122" selection-end-column="59" />
414 </state> 517 </state>
415 </provider> 518 </provider>
416 </entry> 519 </entry>
417 - <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.wxml"> 520 + <entry file="file://$PROJECT_DIR$/pages/my/groupBuy/groupBuy.js">
418 <provider selected="true" editor-type-id="text-editor"> 521 <provider selected="true" editor-type-id="text-editor">
419 - <state relative-caret-position="220">  
420 - <caret line="10" selection-start-line="10" selection-end-line="12" selection-end-column="21" /> 522 + <state relative-caret-position="528">
  523 + <caret line="24" selection-start-line="24" selection-end-line="29" selection-end-column="4" />
  524 + </state>
  525 + </provider>
  526 + </entry>
  527 + <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.json">
  528 + <provider selected="true" editor-type-id="text-editor" />
  529 + </entry>
  530 + <entry file="file://$PROJECT_DIR$/pages/my/protocol/protocol.wxml">
  531 + <provider selected="true" editor-type-id="text-editor">
  532 + <state relative-caret-position="66">
  533 + <caret line="3" column="27" selection-start-line="3" selection-start-column="27" selection-end-line="3" selection-end-column="27" />
421 </state> 534 </state>
422 </provider> 535 </provider>
423 </entry> 536 </entry>
424 <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.wxss"> 537 <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.wxss">
425 <provider selected="true" editor-type-id="text-editor"> 538 <provider selected="true" editor-type-id="text-editor">
426 - <state relative-caret-position="924"> 539 + <state relative-caret-position="828">
427 <caret line="42" selection-start-line="42" selection-end-line="59" selection-end-column="1" /> 540 <caret line="42" selection-start-line="42" selection-end-line="59" selection-end-column="1" />
428 </state> 541 </state>
429 </provider> 542 </provider>
430 </entry> 543 </entry>
431 - <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxss"> 544 + <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.json">
432 <provider selected="true" editor-type-id="text-editor"> 545 <provider selected="true" editor-type-id="text-editor">
433 - <state relative-caret-position="462">  
434 - <caret line="21" column="6" selection-start-line="21" selection-start-column="6" selection-end-line="21" selection-end-column="6" /> 546 + <state relative-caret-position="44">
  547 + <caret line="2" column="1" selection-start-line="2" selection-start-column="1" selection-end-line="2" selection-end-column="1" />
435 </state> 548 </state>
436 </provider> 549 </provider>
437 </entry> 550 </entry>
438 - <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxml"> 551 + <entry file="file://$PROJECT_DIR$/pages/my/protocol/protocol.js">
439 <provider selected="true" editor-type-id="text-editor"> 552 <provider selected="true" editor-type-id="text-editor">
440 - <state relative-caret-position="1100">  
441 - <caret line="50" column="42" selection-start-line="50" selection-start-column="42" selection-end-line="50" selection-end-column="42" /> 553 + <state relative-caret-position="418">
  554 + <caret line="19" column="7" lean-forward="true" selection-start-line="19" selection-start-column="7" selection-end-line="19" selection-end-column="7" />
442 </state> 555 </state>
443 </provider> 556 </provider>
444 </entry> 557 </entry>
445 - <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js"> 558 + <entry file="file://$PROJECT_DIR$/pages/my/my.wxml">
446 <provider selected="true" editor-type-id="text-editor"> 559 <provider selected="true" editor-type-id="text-editor">
447 - <state relative-caret-position="924">  
448 - <caret line="42" column="10" lean-forward="true" selection-start-line="42" selection-start-column="10" selection-end-line="42" selection-end-column="10" /> 560 + <state relative-caret-position="66">
  561 + <caret line="3" column="2" selection-start-line="3" selection-start-column="2" selection-end-line="3" selection-end-column="2" />
449 </state> 562 </state>
450 </provider> 563 </provider>
451 </entry> 564 </entry>
452 - <entry file="file://$PROJECT_DIR$/pages/my/protocol/protocol.js"> 565 + <entry file="file://$PROJECT_DIR$/pages/index/index.wxss">
  566 + <provider selected="true" editor-type-id="text-editor">
  567 + <state relative-caret-position="198">
  568 + <caret line="9" selection-start-line="9" selection-end-line="18" selection-end-column="1" />
  569 + </state>
  570 + </provider>
  571 + </entry>
  572 + <entry file="file://$PROJECT_DIR$/pages/clock/clock.wxss">
  573 + <provider selected="true" editor-type-id="text-editor">
  574 + <state relative-caret-position="330">
  575 + <caret line="15" column="1" selection-start-line="15" selection-start-column="1" selection-end-line="15" selection-end-column="1" />
  576 + </state>
  577 + </provider>
  578 + </entry>
  579 + <entry file="file://$PROJECT_DIR$/pages/my/my.wxss">
  580 + <provider selected="true" editor-type-id="text-editor">
  581 + <state relative-caret-position="330">
  582 + <caret line="15" column="19" lean-forward="true" selection-start-line="15" selection-start-column="19" selection-end-line="15" selection-end-column="19" />
  583 + </state>
  584 + </provider>
  585 + </entry>
  586 + <entry file="file://$PROJECT_DIR$/pages/clock/clock.wxml">
  587 + <provider selected="true" editor-type-id="text-editor">
  588 + <state relative-caret-position="66">
  589 + <caret line="3" column="40" selection-start-line="3" selection-start-column="32" selection-end-line="3" selection-end-column="40" />
  590 + </state>
  591 + </provider>
  592 + </entry>
  593 + <entry file="file://$PROJECT_DIR$/pages/clock/clock.js">
  594 + <provider selected="true" editor-type-id="text-editor">
  595 + <state relative-caret-position="814">
  596 + <caret line="37" selection-start-line="37" selection-end-line="37" />
  597 + </state>
  598 + </provider>
  599 + </entry>
  600 + <entry file="file://$PROJECT_DIR$/utils/qqmap-wx-jssdk.min.js">
453 <provider selected="true" editor-type-id="text-editor" /> 601 <provider selected="true" editor-type-id="text-editor" />
454 </entry> 602 </entry>
455 - <entry file="file://$PROJECT_DIR$/pages/my/myCard/myCard.wxml"> 603 + <entry file="file://$PROJECT_DIR$/pages/my/my.js">
  604 + <provider selected="true" editor-type-id="text-editor">
  605 + <state relative-caret-position="154">
  606 + <caret line="7" column="22" selection-start-line="7" selection-start-column="22" selection-end-line="7" selection-end-column="22" />
  607 + </state>
  608 + </provider>
  609 + </entry>
  610 + <entry file="file://$PROJECT_DIR$/app.wxss">
456 <provider selected="true" editor-type-id="text-editor" /> 611 <provider selected="true" editor-type-id="text-editor" />
457 </entry> 612 </entry>
458 - <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.json"> 613 + <entry file="file://$PROJECT_DIR$/project.config.json">
459 <provider selected="true" editor-type-id="text-editor"> 614 <provider selected="true" editor-type-id="text-editor">
460 - <state relative-caret-position="44">  
461 - <caret line="2" column="1" selection-start-line="2" selection-start-column="1" selection-end-line="2" selection-end-column="1" /> 615 + <state relative-caret-position="308">
  616 + <caret line="14" column="32" selection-start-line="14" selection-start-column="14" selection-end-line="14" selection-end-column="32" />
462 </state> 617 </state>
463 </provider> 618 </provider>
464 </entry> 619 </entry>
465 - <entry file="file://$PROJECT_DIR$/app.js"> 620 + <entry file="file://$PROJECT_DIR$/app.json">
466 <provider selected="true" editor-type-id="text-editor"> 621 <provider selected="true" editor-type-id="text-editor">
467 - <state relative-caret-position="47">  
468 - <caret line="58" column="67" lean-forward="true" selection-start-line="58" selection-start-column="67" selection-end-line="58" selection-end-column="67" />  
469 - <folding>  
470 - <element signature="n#!!doc" expanded="true" />  
471 - </folding> 622 + <state relative-caret-position="44">
  623 + <caret line="2" column="24" selection-start-line="2" selection-start-column="24" selection-end-line="2" selection-end-column="24" />
472 </state> 624 </state>
473 </provider> 625 </provider>
474 </entry> 626 </entry>
475 - <entry file="file://$PROJECT_DIR$/pages/index/index.wxss"> 627 + <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.wxml">
476 <provider selected="true" editor-type-id="text-editor"> 628 <provider selected="true" editor-type-id="text-editor">
477 - <state relative-caret-position="154">  
478 - <caret line="7" column="21" lean-forward="true" selection-start-line="7" selection-start-column="21" selection-end-line="7" selection-end-column="21" /> 629 + <state relative-caret-position="264">
  630 + <caret line="12" column="68" selection-start-line="12" selection-start-column="68" selection-end-line="12" selection-end-column="68" />
479 </state> 631 </state>
480 </provider> 632 </provider>
481 </entry> 633 </entry>
482 - <entry file="file://$PROJECT_DIR$/pages/start/start.js"> 634 + <entry file="file://$PROJECT_DIR$/pages/index/phone-code/phone-code.js">
483 <provider selected="true" editor-type-id="text-editor"> 635 <provider selected="true" editor-type-id="text-editor">
484 - <state relative-caret-position="374">  
485 - <caret line="29" column="49" lean-forward="true" selection-start-line="29" selection-start-column="12" selection-end-line="29" selection-end-column="49" /> 636 + <state relative-caret-position="1870">
  637 + <caret line="85" column="40" selection-start-line="85" selection-start-column="40" selection-end-line="85" selection-end-column="40" />
486 </state> 638 </state>
487 </provider> 639 </provider>
488 </entry> 640 </entry>
489 <entry file="file://$PROJECT_DIR$/pages/index/index.wxml"> 641 <entry file="file://$PROJECT_DIR$/pages/index/index.wxml">
490 <provider selected="true" editor-type-id="text-editor"> 642 <provider selected="true" editor-type-id="text-editor">
491 - <state relative-caret-position="880">  
492 - <caret line="40" lean-forward="true" selection-start-line="40" selection-end-line="40" /> 643 + <state relative-caret-position="542">
  644 + <caret line="74" column="70" selection-start-line="74" selection-start-column="70" selection-end-line="74" selection-end-column="70" />
493 </state> 645 </state>
494 </provider> 646 </provider>
495 </entry> 647 </entry>
496 - <entry file="file://$PROJECT_DIR$/templates/templates.wxml"> 648 + <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml">
497 <provider selected="true" editor-type-id="text-editor"> 649 <provider selected="true" editor-type-id="text-editor">
498 - <state relative-caret-position="374">  
499 - <caret line="29" column="65" selection-start-line="29" selection-start-column="65" selection-end-line="29" selection-end-column="65" /> 650 + <state relative-caret-position="220">
  651 + <caret line="10" column="68" lean-forward="true" selection-start-line="10" selection-start-column="68" selection-end-line="10" selection-end-column="68" />
500 </state> 652 </state>
501 </provider> 653 </provider>
502 </entry> 654 </entry>
503 <entry file="file://$PROJECT_DIR$/pages/index/index.js"> 655 <entry file="file://$PROJECT_DIR$/pages/index/index.js">
504 <provider selected="true" editor-type-id="text-editor"> 656 <provider selected="true" editor-type-id="text-editor">
505 - <state relative-caret-position="506">  
506 - <caret line="98" column="64" selection-start-line="98" selection-start-column="64" selection-end-line="98" selection-end-column="64" /> 657 + <state relative-caret-position="-418">
  658 + <caret line="8" column="23" selection-start-line="8" selection-start-column="23" selection-end-line="8" selection-end-column="23" />
507 <folding> 659 <folding>
508 <element signature="n#!!doc" expanded="true" /> 660 <element signature="n#!!doc" expanded="true" />
509 - <element signature="e#1018#1103#0" />  
510 - <element signature="e#1121#1191#0" />  
511 - <element signature="e#1219#1288#0" />  
512 - <element signature="e#1321#1468#0" />  
513 - <element signature="e#1499#1619#0" />  
514 - <element signature="e#1750#1766#0" />  
515 </folding> 661 </folding>
516 </state> 662 </state>
517 </provider> 663 </provider>
518 </entry> 664 </entry>
519 - <entry file="file://$PROJECT_DIR$/app.json"> 665 + <entry file="file://$PROJECT_DIR$/pages/index/go-appointment/go-appointment.json">
  666 + <provider selected="true" editor-type-id="text-editor" />
  667 + </entry>
  668 + <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js">
520 <provider selected="true" editor-type-id="text-editor"> 669 <provider selected="true" editor-type-id="text-editor">
521 - <state relative-caret-position="440">  
522 - <caret line="20" column="38" selection-start-line="20" selection-start-column="38" selection-end-line="20" selection-end-column="38" /> 670 + <state relative-caret-position="374">
  671 + <caret line="29" column="44" lean-forward="true" selection-start-line="29" selection-start-column="44" selection-end-line="29" selection-end-column="44" />
523 </state> 672 </state>
524 </provider> 673 </provider>
525 </entry> 674 </entry>
526 - <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.json">  
527 - <provider selected="true" editor-type-id="text-editor" /> 675 + <entry file="file://$PROJECT_DIR$/pages/start/start.js">
  676 + <provider selected="true" editor-type-id="text-editor">
  677 + <state relative-caret-position="676">
  678 + <caret line="37" column="25" lean-forward="true" selection-start-line="37" selection-start-column="25" selection-end-line="37" selection-end-column="25" />
  679 + </state>
  680 + </provider>
528 </entry> 681 </entry>
529 - <entry file="file://$PROJECT_DIR$/pages/start/start.wxml"> 682 + <entry file="file://$PROJECT_DIR$/app.js">
530 <provider selected="true" editor-type-id="text-editor"> 683 <provider selected="true" editor-type-id="text-editor">
531 - <state relative-caret-position="132">  
532 - <caret line="6" column="15" lean-forward="true" selection-start-line="6" selection-start-column="15" selection-end-line="6" selection-end-column="15" /> 684 + <state relative-caret-position="344">
  685 + <caret line="140" column="12" selection-start-line="140" selection-start-column="4" selection-end-line="140" selection-end-column="12" />
  686 + <folding>
  687 + <element signature="n#!!doc" expanded="true" />
  688 + </folding>
533 </state> 689 </state>
534 </provider> 690 </provider>
535 </entry> 691 </entry>
536 - <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.js"> 692 + <entry file="file://$PROJECT_DIR$/utils/util.js">
537 <provider selected="true" editor-type-id="text-editor"> 693 <provider selected="true" editor-type-id="text-editor">
538 - <state relative-caret-position="550">  
539 - <caret line="25" column="32" selection-start-line="25" selection-start-column="32" selection-end-line="25" selection-end-column="32" /> 694 + <state relative-caret-position="946">
  695 + <caret line="43" column="15" selection-start-line="43" selection-start-column="2" selection-end-line="43" selection-end-column="15" />
540 </state> 696 </state>
541 </provider> 697 </provider>
542 </entry> 698 </entry>
543 - <entry file="file://$PROJECT_DIR$/pages/index/activity-detail/activity-detail.wxml"> 699 + <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.wxml">
  700 + <provider selected="true" editor-type-id="text-editor">
  701 + <state relative-caret-position="852">
  702 + <caret line="50" column="42" selection-start-line="50" selection-start-column="42" selection-end-line="50" selection-end-column="42" />
  703 + </state>
  704 + </provider>
  705 + </entry>
  706 + <entry file="file://$PROJECT_DIR$/pages/index/confirm-appointment-info/confirm-appointment-info.js">
  707 + <provider selected="true" editor-type-id="text-editor">
  708 + <state relative-caret-position="302">
  709 + <caret line="44" column="20" lean-forward="true" selection-start-line="44" selection-start-column="20" selection-end-line="44" selection-end-column="20" />
  710 + </state>
  711 + </provider>
  712 + </entry>
  713 + <entry file="file://$PROJECT_DIR$/pages/index/go-appointment/go-appointment.wxml">
  714 + <provider selected="true" editor-type-id="text-editor">
  715 + <state relative-caret-position="638">
  716 + <caret line="41" column="90" selection-start-line="41" selection-start-column="72" selection-end-line="41" selection-end-column="90" />
  717 + </state>
  718 + </provider>
  719 + </entry>
  720 + <entry file="file://$PROJECT_DIR$/pages/index/go-appointment/go-appointment.js">
544 <provider selected="true" editor-type-id="text-editor"> 721 <provider selected="true" editor-type-id="text-editor">
545 <state relative-caret-position="198"> 722 <state relative-caret-position="198">
546 - <caret line="9" column="35" selection-start-line="9" selection-start-column="35" selection-end-line="9" selection-end-column="35" /> 723 + <caret line="125" column="20" lean-forward="true" selection-start-line="125" selection-start-column="20" selection-end-line="125" selection-end-column="20" />
  724 + <folding>
  725 + <element signature="e#243#454#0" />
  726 + <element signature="e#462#673#0" />
  727 + <element signature="e#681#893#0" />
  728 + <element signature="e#901#1114#0" />
  729 + <element signature="e#1177#2298#0" />
  730 + </folding>
547 </state> 731 </state>
548 </provider> 732 </provider>
549 </entry> 733 </entry>
@@ -89,6 +89,7 @@ App({ @@ -89,6 +89,7 @@ App({
89 success: function (res) { } 89 success: function (res) { }
90 }) 90 })
91 } else { 91 } else {
  92 + console.log(res.data);
92 wx.showModal({ 93 wx.showModal({
93 title: '提示', 94 title: '提示',
94 content: res.data.msg, 95 content: res.data.msg,
@@ -123,6 +124,7 @@ App({ @@ -123,6 +124,7 @@ App({
123 124
124 }) 125 })
125 }, 126 },
  127 + //获取当前时间
126 nowDate() { 128 nowDate() {
127 let date = new Date(); 129 let date = new Date();
128 let month = date.getMonth() + 1; 130 let month = date.getMonth() + 1;
@@ -6,7 +6,12 @@ Page({ @@ -6,7 +6,12 @@ Page({
6 */ 6 */
7 data: { 7 data: {
8 punchState: true, 8 punchState: true,
9 - address: '' 9 + address: '',
  10 + hasPhone: true,
  11 + },
  12 + //首次登录小程序,跳转到认证手机页面
  13 + goPhoneCode() {
  14 + wx.navigateTo({url: '/pages/index/phone-code/phone-code'})
10 }, 15 },
11 //打卡申诉 16 //打卡申诉
12 punchApply() { 17 punchApply() {
@@ -30,6 +35,7 @@ Page({ @@ -30,6 +35,7 @@ Page({
30 }) 35 })
31 }, 36 },
32 }) 37 })
  38 +
33 // wx.chooseLocation({ 39 // wx.chooseLocation({
34 // success: function(res) { 40 // success: function(res) {
35 // console.log(res) 41 // console.log(res)
1 <!--pages/clock/clock.wxml--> 1 <!--pages/clock/clock.wxml-->
2 <view class='content_box'> 2 <view class='content_box'>
  3 + <!--蒙层-->
  4 + <view class="modal" wx:if="{{!hasPhone}}" bindtap="goPhoneCode"></view>
3 <view class='circle_large'> 5 <view class='circle_large'>
4 <view class='circle_middle'> 6 <view class='circle_middle'>
5 <view class='circle_small'> 7 <view class='circle_small'>
@@ -13,7 +13,16 @@ @@ -13,7 +13,16 @@
13 align-items: center; 13 align-items: center;
14 padding-top: 115rpx; 14 padding-top: 115rpx;
15 } 15 }
16 - 16 +.modal {
  17 + width: 100%;
  18 + height: 100%;
  19 + position: absolute;
  20 + left: 0;
  21 + top: 0;
  22 + background-color: #000000;
  23 + opacity: 0.8;
  24 + z-index: 10;
  25 +}
17 .circle_large, .circle_middle, .circle_small { 26 .circle_large, .circle_middle, .circle_small {
18 width: 327rpx; 27 width: 327rpx;
19 height: 327rpx; 28 height: 327rpx;
1 // pages/index/activity-detail/activity-detail.js 1 // pages/index/activity-detail/activity-detail.js
  2 +var WxParse = require('../../../wxParse/wxParse.js');
2 const app = getApp(); 3 const app = getApp();
3 Page({ 4 Page({
4 5
@@ -18,21 +19,39 @@ Page({ @@ -18,21 +19,39 @@ Page({
18 let params = { 19 let params = {
19 id: this.data.test_id,//当前实验的id 20 id: this.data.test_id,//当前实验的id
20 }; 21 };
21 - let headers= { 22 + let header = {
22 "XX-token": wx.getStorageSync('token') 23 "XX-token": wx.getStorageSync('token')
23 }; 24 };
24 - app.post(url, params,headers).then((res) => { 25 + app.post(url, params,header).then((res) => {
25 console.log(res); 26 console.log(res);
26 - this.setData({detail: res}) 27 + if(res.code === 0) {
  28 + wx.navigateTo({url: '/pages/index/phone-code/phone-code'})
  29 + }else {
  30 + this.setData({detail: res});
  31 + var article = res.content;
  32 + WxParse.wxParse('article', 'html', article, this, 5);
  33 + }
27 // console.log(this.data.this_week_test_info); 34 // console.log(this.data.this_week_test_info);
28 }) 35 })
29 }, 36 },
  37 + //服务协议
  38 + goServiceProtocol() {
  39 + wx.navigateTo({
  40 + url: '/pages/my/protocol/protocol',
  41 + })
  42 + },
  43 + goAppointment() {
  44 + wx.navigateTo({url: '/pages/index/go-appointment/go-appointment?experiment_id=' + this.data.detail.id})
  45 + },
30 /** 46 /**
31 * 生命周期函数--监听页面加载 47 * 生命周期函数--监听页面加载
32 */ 48 */
33 onLoad: function (options) { 49 onLoad: function (options) {
34 console.log(options); 50 console.log(options);
35 - this.setData({test_id: +options.id}); 51 + this.setData({
  52 + test_id: +options.id,
  53 + is_the_week: options.is_the_week
  54 + });
36 this.getDetail() 55 this.getDetail()
37 }, 56 },
38 57
@@ -2,24 +2,27 @@ @@ -2,24 +2,27 @@
2 <view class="content"> 2 <view class="content">
3 <view class="head"> 3 <view class="head">
4 <view class="img_box"> 4 <view class="img_box">
5 - <image src="../../../images/test_img.jpg" class="activity_img"></image> 5 + <image src="{{detail.thumb}}" class="activity_img" mode="aspectFill"></image>
6 </view> 6 </view>
7 <view class="title"> 7 <view class="title">
8 <view class="line"></view> 8 <view class="line"></view>
9 <view class="test_detail"> 9 <view class="test_detail">
10 - <text>{{detail.name}}科学队长实验室:雷电的产生!</text>  
11 - <view class="describe">云和云摩擦碰撞发出的声音是雷声产生火花是闪电。</view> 10 + <text>{{detail.name}}</text>
  11 + <view class="describe">{{detail.description}}</view>
12 </view> 12 </view>
13 </view> 13 </view>
14 </view> 14 </view>
15 15
16 - <view class="jiexi">解析长图</view>  
17 - <view class="footer" wx:if="{{is_the_week}}">  
18 - <view class="service" bindtap="clickAgreeBtn" bindtap='serviceProtocol'> 16 + <view class="jiexi">解析长图
  17 + <import src="/wxParse/wxParse.wxml"/>
  18 + <template is="wxParse" data="{{wxParseData:article.nodes}}"/>
  19 + </view>
  20 + <view class="footer" wx:if="{{detail.allow_res}}">
  21 + <view class="service" bindtap="clickAgreeBtn">
19 <view class="agree_box"> 22 <view class="agree_box">
20 <view class="agree_btn" wx:if="{{is_agree}}"></view> 23 <view class="agree_btn" wx:if="{{is_agree}}"></view>
21 </view> 24 </view>
22 - <text>服务协议</text> 25 + <text catchtap="goServiceProtocol">服务协议</text>
23 </view> 26 </view>
24 <view class="order_btn" bindtap='goAppointment'>去预约</view> 27 <view class="order_btn" bindtap='goAppointment'>去预约</view>
25 </view> 28 </view>
@@ -49,11 +49,14 @@ Page({ @@ -49,11 +49,14 @@ Page({
49 addStudent() { 49 addStudent() {
50 wx.navigateTo({url: './add-student/add-student'}) 50 wx.navigateTo({url: './add-student/add-student'})
51 }, 51 },
  52 + getAppointmentInfo() {
  53 +
  54 + },
52 /** 55 /**
53 * 生命周期函数--监听页面加载 56 * 生命周期函数--监听页面加载
54 */ 57 */
55 onLoad: function (options) { 58 onLoad: function (options) {
56 - 59 + console.log(options);
57 }, 60 },
58 61
59 /** 62 /**
1 // pages/index/go-appointment/go-appointment.js 1 // pages/index/go-appointment/go-appointment.js
  2 +const util = require('../../../utils/util.js');
  3 +const app = getApp();
2 Page({ 4 Page({
3 5
4 /** 6 /**
5 * 页面的初始数据 7 * 页面的初始数据
6 */ 8 */
7 data: { 9 data: {
8 - array: ['北京市', '天津市', '上海市', '河北省'], 10 + cityList: [],
  11 + // city_index: 0,
9 currentCity: '北京市', 12 currentCity: '北京市',
  13 + current_test_id: '',
10 is_change: false, 14 is_change: false,
  15 + is_choose: false,
  16 + description: '',
  17 + session_list: {},
11 show_submit_btn: false, 18 show_submit_btn: false,
12 areaTest: [{ 19 areaTest: [{
13 area: '朝阳区', 20 area: '朝阳区',
@@ -104,28 +111,49 @@ Page({ @@ -104,28 +111,49 @@ Page({
104 bindPickerChange: function(e) { 111 bindPickerChange: function(e) {
105 console.log('picker发送选择改变,携带值为', e.detail.value) 112 console.log('picker发送选择改变,携带值为', e.detail.value)
106 this.setData({ 113 this.setData({
107 - index: e.detail.value, 114 + city_index: e.detail.value,
108 is_change: true, 115 is_change: true,
109 - }) 116 + });
  117 + this.getChooseSession(this.data.cityList[e.detail.value])
110 }, 118 },
111 -  
112 //选择实验室 119 //选择实验室
113 chooseTest(e) { 120 chooseTest(e) {
114 - console.log(e.currentTarget.dataset.index); 121 + // console.log(e);
  122 + // console.log('实验',e.currentTarget.dataset.index);
115 const current = e.currentTarget.dataset.index; 123 const current = e.currentTarget.dataset.index;
116 const current_id = e.currentTarget.dataset.id; 124 const current_id = e.currentTarget.dataset.id;
  125 + // console.log(this.data.areaTest[0].lab_list[0].session_list[0].id);
117 this.setData({ 126 this.setData({
118 currentTest: current, 127 currentTest: current,
119 - currentId: current_id  
120 - }) 128 + current_test_id: current_id,
  129 + is_choose: false,//恢复默认值
  130 + // current_session_id: this.data.areaTest[current].lab_list ? this.data.areaTest[current].lab_list[0].session_list[0].id : '',
  131 + // description: this.data.areaTest[current].lab_list[current].description
  132 + });
  133 + // console.log(util.formatTimeTwo(1543375800,'h:m'),'1543375800');
  134 + return this.data.areaTest.forEach((item,index) => {
  135 + if(item.lab_list[current].id === current_id) {
  136 + // console.log(current_id, item.lab_list[current].id,item.lab_list[current].description);
  137 + this.setData({
  138 + description: item.lab_list[current].description ? item.lab_list[current].description : '',
  139 + session_list: item.lab_list[current].session_list ? item.lab_list[current].session_list: '',
  140 + });
  141 + // return
  142 + }
  143 + });
  144 + // console.log(a);
  145 + // console.log(this.data.areaTest[current].lab_list[current].description);
121 }, 146 },
122 147
123 - //选择时段 148 + //选择场次
124 chooseTime(e) { 149 chooseTime(e) {
125 console.log(e.currentTarget.dataset.index); 150 console.log(e.currentTarget.dataset.index);
126 const current = e.currentTarget.dataset.index; 151 const current = e.currentTarget.dataset.index;
  152 + const current_id = e.currentTarget.dataset.id;
127 this.setData({ 153 this.setData({
128 - currentTime: current 154 + currentTime: current,
  155 + current_session_id: current_id,
  156 + is_choose: true,
129 }) 157 })
130 }, 158 },
131 159
@@ -135,7 +163,7 @@ Page({ @@ -135,7 +163,7 @@ Page({
135 // show_submit_btn: true 163 // show_submit_btn: true
136 // }) 164 // })
137 wx.navigateTo({ 165 wx.navigateTo({
138 - url: '/pages/index/confirm-appointment-info/confirm-appointment-info' 166 + url: '/pages/index/confirm-appointment-info/confirm-appointment-info?num=' + 1
139 }) 167 })
140 }, 168 },
141 169
@@ -145,20 +173,93 @@ Page({ @@ -145,20 +173,93 @@ Page({
145 // show_submit_btn: true 173 // show_submit_btn: true
146 // }) 174 // })
147 wx.navigateTo({ 175 wx.navigateTo({
148 - url: '/pages/index/confirm-appointment-info/confirm-appointment-info' 176 + url: '/pages/index/confirm-appointment-info/confirm-appointment-info?num=' + 2
149 }) 177 })
150 }, 178 },
151 179
152 submit() { 180 submit() {
  181 + if(this.data.is_choose) {
153 wx.navigateTo({ 182 wx.navigateTo({
154 - url: '/pages/index/confirm-appointment-info/confirm-appointment-info' 183 + url: '/pages/index/confirm-appointment-info/confirm-appointment-info?experiment_id=' + this.data.current_test_id
  184 + + '&session_id=' + this.data.current_session_id
  185 + })
  186 + }else {
  187 + wx.showToast({title: '请选择场次!',icon: 'none'})
  188 + }
  189 + },
  190 + //获取城市列表
  191 + getCity() {
  192 + let url = '/portal/Experiment/cityList';
  193 + let header = {
  194 + "XX-token": wx.getStorageSync('token')
  195 + };
  196 + app.post(url, {},header).then((res) => {
  197 + console.log(res);
  198 + var city_list = [];
  199 + res.forEach((item,index) => {
  200 + city_list.push(item.city)
  201 + });
  202 + console.log(city_list,res);
  203 + this.setData({
  204 + cityList: city_list,
  205 + // description: res.list[0].lab_list[0].description,//默认显示第一个实验室的描述
  206 + // session_list: res.list[0].lab_list[0].session_list//默认显示第一个实验室的实验时间
  207 +
  208 + })
  209 + // console.log(this.data.this_week_test_info);
155 }) 210 })
156 }, 211 },
  212 + //获取预约城市、区、实验室、场次
  213 + getChooseSession(city) {
  214 + let url = '/portal/Experiment/choose_session';
  215 + let params = {
  216 + experiment_id: this.data.experiment_id,
  217 + city: city,
  218 + };
  219 + let header = {
  220 + "XX-token": wx.getStorageSync('token')
  221 + };
  222 + app.post(url, params,header).then((res) => {
  223 + console.log(res,res.list);
  224 + res.list.forEach((item,index) => {
  225 + if(item.lab_list) {
  226 + item.lab_list.forEach((i,k) => {
  227 + if(i.session_list) {
  228 + i.session_list.forEach((j,b) => {
  229 + j.start_time = util.formatTimeTwo(j.start_time, 'h:m');
  230 + j.end_time = util.formatTimeTwo(j.end_time, 'h:m');
  231 + })
  232 + }
  233 + })
  234 + }
  235 + });
  236 + console.log(res,res.list);
  237 + // console.log(res.list[0].lab_list[0].id);
  238 + this.setData({
  239 + areaTest: res.list,
  240 + // current_test_id: res.list[0].lab_list ? res.list[0].lab_list[0].id : '',//默认实验室id
  241 + // current_session_id: res.list[0].lab_list ? res.list[0].lab_list[0].session_list[0].id : '',//默认场次id
  242 + description: res.list[0].lab_list ? res.list[0].lab_list[0].description : '',//默认显示第一个实验室的描述
  243 + session_list: res.list[0].lab_list ? res.list[0].lab_list[0].session_list :''//默认显示第一个实验室的实验时间
157 244
  245 + })
  246 + // console.log(this.data.this_week_test_info);
  247 + })
  248 + },
158 /** 249 /**
159 * 生命周期函数--监听页面加载 250 * 生命周期函数--监听页面加载
160 */ 251 */
161 onLoad: function(options) { 252 onLoad: function(options) {
  253 + const self = this;
  254 + self.setData({experiment_id: +options.experiment_id});
  255 + if(wx.getStorageSync('city')) {
  256 + const city = wx.getStorageSync('city');
  257 + self.getChooseSession(city);
  258 + }else {
  259 + const city = '北京市';
  260 + self.getChooseSession(city);
  261 + }
  262 + self.getCity();
162 this.data.areaTest.forEach((v, i) => { 263 this.data.areaTest.forEach((v, i) => {
163 Object.keys(v).forEach(v => { 264 Object.keys(v).forEach(v => {
164 console.log(v) //取到了key 265 console.log(v) //取到了key
1 <!--pages/index/go-appointment/go-appointment.wxml--> 1 <!--pages/index/go-appointment/go-appointment.wxml-->
2 <view class="content"> 2 <view class="content">
3 <view class="city_box"> 3 <view class="city_box">
4 - <picker bindchange="bindPickerChange" value="{{index}}" range="{{array}}"> 4 + <picker bindchange="bindPickerChange" value="{{city_index}}" range="{{cityList}}">
5 <view class="select"> 5 <view class="select">
6 <view class="picker" wx:if="{{is_change}}"> 6 <view class="picker" wx:if="{{is_change}}">
7 - {{array[index]}} 7 + {{cityList[city_index]}}
8 </view> 8 </view>
9 <view wx:else>北京市</view> 9 <view wx:else>北京市</view>
10 <view class="iconfont icon-daosanjiao"></view> 10 <view class="iconfont icon-daosanjiao"></view>
@@ -15,16 +15,16 @@ @@ -15,16 +15,16 @@
15 <view class="section"> 15 <view class="section">
16 <scroll-view class="area_box" scroll-y> 16 <scroll-view class="area_box" scroll-y>
17 <block wx:for="{{areaTest}}" wx:key="index"> 17 <block wx:for="{{areaTest}}" wx:key="index">
18 - <view class="area-test">  
19 - <view class="area">{{item.area}}</view> 18 + <view class="area-test" data-index="{{index}}" wx:for-index="{{idx}}">
  19 + <view class="area">{{item.name}}</view>
20 <view class="test-list"> 20 <view class="test-list">
21 <view class="list-item-box"> 21 <view class="list-item-box">
22 - <block wx:for="{{item.list}}" wx:key="index"> 22 + <block wx:for="{{item.lab_list}}" wx:key="index">
23 <view class="list-item" bindtap="chooseTest" data-index="{{index}}" 23 <view class="list-item" bindtap="chooseTest" data-index="{{index}}"
24 data-id="{{item.id}}"> 24 data-id="{{item.id}}">
25 - <image src="../../../images/blue_line.png" wx:if="{{currentId === item.id}}" 25 + <image src="../../../images/blue_line.png" wx:if="{{current_test_id === item.id}}"
26 class="blue-line"></image> 26 class="blue-line"></image>
27 - <view class="test {{currentId === item.id?'change-color':''}}">{{item.name}}</view> 27 + <view class="test {{current_test_id === item.id?'change-color':''}}">{{item.name}}</view>
28 </view> 28 </view>
29 </block> 29 </block>
30 </view> 30 </view>
@@ -35,17 +35,17 @@ @@ -35,17 +35,17 @@
35 35
36 <view class="test_box"> 36 <view class="test_box">
37 <scroll-view scroll-y class="test_describe"> 37 <scroll-view scroll-y class="test_describe">
38 - {{test.title}} 38 + {{description}}
39 </scroll-view> 39 </scroll-view>
40 <scroll-view class="time-list-box" scroll-y> 40 <scroll-view class="time-list-box" scroll-y>
41 - <view wx:for="{{test.list}}" wx:key="index" class="time-list" bindtap="chooseTime" data-index="{{index}}">  
42 - <image src="../../../images/blue_line.png" wx:if="{{currentTime === index && item.rest !== 0}}" 41 + <view wx:for="{{session_list}}" wx:key="index" class="time-list" bindtap="chooseTime" data-index="{{index}}" data-id="{{item.id}}">
  42 + <image src="../../../images/blue_line.png" wx:if="{{current_session_id === item.id}}"
43 class="blue-line"></image> 43 class="blue-line"></image>
44 <view class="right-content {{item.rest === 0? 'change-gray':''}}"> 44 <view class="right-content {{item.rest === 0? 'change-gray':''}}">
45 - <view class="{{currentTime === index && item.rest !== 0? 'change-color':''}}">{{item.week}} {{item.date}}</view> 45 + <view class="{{current_session_id === item.id? 'change-color':''}}">{{item.weekday}} {{item.date}}</view>
46 <view class="bottom-time-box"> 46 <view class="bottom-time-box">
47 - <text class="time {{currentTime === index && item.rest !== 0? 'change-color':''}}">{{item.time}}</text>  
48 - <text class="rest {{item.rest === 0? 'change-gray':''}}">剩余: {{item.rest}}</text> 47 + <text class="time {{current_session_id === item.id? 'change-color':''}}">{{item.time}}</text>
  48 + <text class="rest {{item.rest === 0? 'change-gray':''}}">剩余: {{item.surplus_num}}</text>
49 </view> 49 </view>
50 </view> 50 </view>
51 </view> 51 </view>
@@ -54,13 +54,14 @@ @@ -54,13 +54,14 @@
54 </view> 54 </view>
55 55
56 <view class="footer"> 56 <view class="footer">
57 - <view class="confirm_btn" bindtap="submit" wx:if="{{show_submit_btn}}"> 57 + <!--<view class="confirm_btn" bindtap="submit" wx:if="{{show_submit_btn}}">-->
  58 + <view class="confirm_btn" bindtap="submit">
58 <text>提 交</text> 59 <text>提 交</text>
59 </view> 60 </view>
60 - <view class="appointment-box" wx:else>  
61 - <text class="single" bindtap="singleAppointment">单人预约</text>  
62 - <text class="double" bindtap="doubleAppointment">双人预约</text>  
63 - </view> 61 + <!--<view class="appointment-box" wx:else>-->
  62 + <!--<text class="single" bindtap="singleAppointment">单人预约</text>-->
  63 + <!--<text class="double" bindtap="doubleAppointment">双人预约</text>-->
  64 + <!--</view>-->
64 <!--<view>提 交</view>--> 65 <!--<view>提 交</view>-->
65 </view> 66 </view>
66 </view> 67 </view>
1 //index.js 1 //index.js
2 //获取应用实例 2 //获取应用实例
3 -const app = getApp() 3 +const app = getApp();
  4 +// 引入SDK核心类
  5 +var QQMapWX = require('../../utils/qqmap-wx-jssdk.min.js');
4 6
5 Page({ 7 Page({
6 data: { 8 data: {
7 - hasPhone: true, 9 + hasPhone: false,
8 imgUrls: [ 10 imgUrls: [
9 {'url': '../../images/banner.jpg'}, 11 {'url': '../../images/banner.jpg'},
10 {'url': '../../images/banner.jpg'}, 12 {'url': '../../images/banner.jpg'},
@@ -28,6 +30,21 @@ Page({ @@ -28,6 +30,21 @@ Page({
28 is_past: true, 30 is_past: true,
29 }, 31 },
30 }, 32 },
  33 + //判断手机号是否绑定
  34 + checkMobile() {
  35 + let url = '/portal/Index/check_mobile';
  36 + let header = {
  37 + "XX-token": wx.getStorageSync('token')
  38 + };
  39 + app.post(url, {},header).then((res) => {
  40 + console.log(res);
  41 + if(res.is_binding) {
  42 + this.setData({hasPhone: true})
  43 + }else {
  44 + this.setData({hasPhone: false})
  45 + }
  46 + })
  47 + },
31 swiperChange: function (e) { //切换轮播图 48 swiperChange: function (e) { //切换轮播图
32 this.setData({ 49 this.setData({
33 currentSwiper: e.detail.current 50 currentSwiper: e.detail.current
@@ -44,13 +61,17 @@ Page({ @@ -44,13 +61,17 @@ Page({
44 //点击进入活动详情 61 //点击进入活动详情
45 goTestDetail(e) { 62 goTestDetail(e) {
46 // console.log(e); 63 // console.log(e);
47 - const current = e.currentTarget.dataset.index;  
48 - wx.navigateTo({url: './activity-detail/activity-detail'}) 64 + const is_the_week = e.currentTarget.dataset.is_the_week;
  65 + const id = e.currentTarget.dataset.id;
  66 + wx.navigateTo({
  67 + url: './activity-detail/activity-detail?is_the_week=' + is_the_week +
  68 + '&id=' + id
  69 + })
49 }, 70 },
50 //前往预约 71 //前往预约
51 goAppointment(e) { 72 goAppointment(e) {
52 - const current = e.currentTarget.dataset.index;  
53 - wx.navigateTo({url: './go-appointment/go-appointment'}) 73 + const id = e.currentTarget.dataset.id;
  74 + wx.navigateTo({url: './go-appointment/go-appointment?experiment_id=' + id})
54 }, 75 },
55 //获取轮播图 76 //获取轮播图
56 getSlideImage() { 77 getSlideImage() {
@@ -58,15 +79,9 @@ Page({ @@ -58,15 +79,9 @@ Page({
58 let params = { 79 let params = {
59 city: this.data.city, 80 city: this.data.city,
60 }; 81 };
61 - app.post(url, params).then((ret) => {  
62 - wx.setStorageSync('token', ret.token)  
63 - app.globalData.userInfo = ret.user_type  
64 - // console.log('userInfo',app.globalData.userInfo,ret);  
65 - if (ret.user_type == 2) {  
66 - wx.switchTab({  
67 - url: '/pages/index/index',  
68 - })  
69 - } 82 + app.post(url, params).then((res) => {
  83 + console.log(res);
  84 + this.setData({imgUrls: res.list})
70 }) 85 })
71 }, 86 },
72 //获取本周实验 87 //获取本周实验
@@ -99,22 +114,52 @@ Page({ @@ -99,22 +114,52 @@ Page({
99 const id = e.currentTarget.dataset.id; 114 const id = e.currentTarget.dataset.id;
100 wx.navigateTo({url: '/pages/index/activity-detail/activity-detail?test_id=' + id}) 115 wx.navigateTo({url: '/pages/index/activity-detail/activity-detail?test_id=' + id})
101 }, 116 },
  117 + getUser() {
  118 + let url = '/user/Profile/getUserInfo';
  119 + let header = {
  120 + "XX-token": wx.getStorageSync('token')
  121 + };
  122 + app.post(url, {},header).then((res) => {
  123 + console.log('个人信息结果',res);
  124 + // self.setData({past_test_info: res})
  125 + // console.log(self.data.this_week_test_info);
  126 + });
  127 + },
102 onLoad: function () { 128 onLoad: function () {
103 - this.getTheWeekTest();  
104 - this.getPastWeekTest(); 129 + const self = this;
  130 + self.getTheWeekTest();
  131 + self.getPastWeekTest();
  132 + self.getUser();
  133 + self.checkMobile();
  134 + //获取当前城市
105 wx.getLocation({ 135 wx.getLocation({
106 type: 'wgs84', 136 type: 'wgs84',
107 success: function(res) { 137 success: function(res) {
108 console.log(res); 138 console.log(res);
109 - let url = '/public/getAddressInfoByCoordinate';  
110 - let params = { 139 + // 实例化API核心类
  140 + var demo = new QQMapWX({
  141 + key: 'CLVBZ-KHZKP-KWLDW-VDJTA-QVUST-XOBVO' // 必填
  142 + });
  143 +
  144 + // 调用接口
  145 + demo.reverseGeocoder({
  146 + location: {
111 latitude: res.latitude,//纬度 147 latitude: res.latitude,//纬度
112 longitude: res.longitude//经度 148 longitude: res.longitude//经度
113 - };  
114 - // app.post(url,params).then((res) => {  
115 - // // console.log('获取城市',res);  
116 - // self.setData({city: res.city})  
117 - // }) 149 + },
  150 + success: function (res) {
  151 + console.log(res);
  152 + self.setData({city: res.result.address_component.city});
  153 + wx.setStorageSync('city', res.result.address_component.city);
  154 + self.getSlideImage();
  155 + },
  156 + fail: function (res) {
  157 + console.log(res);
  158 + },
  159 + complete: function (res) {
  160 + // console.log(res);
  161 + }
  162 + });
118 } 163 }
119 }); 164 });
120 }, 165 },
@@ -5,12 +5,13 @@ @@ -5,12 +5,13 @@
5 5
6 <!--轮播图--> 6 <!--轮播图-->
7 <view class="wrap"> 7 <view class="wrap">
8 - <swiper autoplay="{{autoplay}}" interval="{{interval}}" duration="{{duration}}" circular="true" bindchange="swiperChange"> 8 + <swiper autoplay="{{autoplay}}" interval="{{interval}}" duration="{{duration}}" circular="true"
  9 + bindchange="swiperChange">
9 <block wx:for="{{imgUrls}}" wx:key="index"> 10 <block wx:for="{{imgUrls}}" wx:key="index">
10 <swiper-item> 11 <swiper-item>
11 <navigator url="{{item.url}}" hover-class="navigator-hover"> 12 <navigator url="{{item.url}}" hover-class="navigator-hover">
12 <view class="index_top"> 13 <view class="index_top">
13 - <image src="{{item.url}}" class="slide-image" mode="aspectFill" /> 14 + <image src="{{item.image}}" class="slide-image" mode="aspectFill"/>
14 </view> 15 </view>
15 </navigator> 16 </navigator>
16 </swiper-item> 17 </swiper-item>
@@ -31,9 +32,9 @@ @@ -31,9 +32,9 @@
31 </view> 32 </view>
32 33
33 <!--实验列表--> 34 <!--实验列表-->
34 - <!--<import src="/templates/templates.wxml" />--> 35 + <!--<import src="/templates/templates.wxml"/>-->
35 <!--<view class="past_template">--> 36 <!--<view class="past_template">-->
36 - <!--<template is="experiment" data="{{...this_week_test_info}}"></template>--> 37 + <!--<template is="experiment" data="{{this_week_test_info}}"></template>-->
37 <!--<template is="experiment" data="{{...past_test_info}}"></template>--> 38 <!--<template is="experiment" data="{{...past_test_info}}"></template>-->
38 <!--</view>--> 39 <!--</view>-->
39 <!-- <import src="/templates/templates.wxml" /> --> 40 <!-- <import src="/templates/templates.wxml" /> -->
@@ -46,15 +47,16 @@ @@ -46,15 +47,16 @@
46 <text>本周实验</text> 47 <text>本周实验</text>
47 <!-- <text>往期实验</text> --> 48 <!-- <text>往期实验</text> -->
48 </view> 49 </view>
49 - <view wx:for="{{info}}" wx:key="index" data-index="{{index}}" bindtap="goTestDetail" data-is_the_week='{{true}}'> 50 + <view bindtap="goTestDetail"
  51 + data-is_the_week='{{true}}' data-id="{{this_week_test_info.id}}">
50 <view class="test_box"> 52 <view class="test_box">
51 - <view class="time">{{item.time}}</view> 53 + <view class="time">{{this_week_test_info.start_time}} - {{this_week_test_info.end_time}}</view>
52 <view class="img_box"> 54 <view class="img_box">
53 - <image src="{{item.url}}"></image> 55 + <image src="{{this_week_test_info.thumb}}" mode="aspectFill"></image>
54 </view> 56 </view>
55 </view> 57 </view>
56 <view class="order_box"> 58 <view class="order_box">
57 - <text catchtap="goAppointment" data-index="{{index}}">前往预约</text> 59 + <text catchtap="goAppointment" data-index="{{index}}" data-id="{{this_week_test_info.id}}">前往预约</text>
58 </view> 60 </view>
59 </view> 61 </view>
60 </view> 62 </view>
@@ -65,11 +67,12 @@ @@ -65,11 +67,12 @@
65 <view class="line"></view> 67 <view class="line"></view>
66 <text>往期实验</text> 68 <text>往期实验</text>
67 </view> 69 </view>
68 - <view wx:for="{{info}}" wx:key="index" data-index="{{index}}" bindtap="goTestDetail" data-is_the_week='{{false}}'> 70 + <view wx:for="{{past_test_info.list}}" wx:key="index" data-index="{{index}}" data-id="{{item.id}}" bindtap="goTestDetail"
  71 + data-is_the_week='{{false}}'>
69 <view class="test_box"> 72 <view class="test_box">
70 - <view class="time">{{item.time}}</view> 73 + <view class="time">{{item.start_time}} - {{item.end_time}}</view>
71 <view class="img_box"> 74 <view class="img_box">
72 - <image src="{{item.url}}"></image> 75 + <image src="{{item.thumb}}" mode="aspectFill"></image>
73 </view> 76 </view>
74 </view> 77 </view>
75 <view class="order_box"> 78 <view class="order_box">
1 // pages/index/phone-code/phone-code.js 1 // pages/index/phone-code/phone-code.js
  2 +const app = getApp();
2 var interval = null //倒计时函数 3 var interval = null //倒计时函数
3 Page({ 4 Page({
4 5
@@ -8,12 +9,20 @@ Page({ @@ -8,12 +9,20 @@ Page({
8 data: { 9 data: {
9 fun_id: 2, 10 fun_id: 2,
10 time: '获取验证码', //倒计时 11 time: '获取验证码', //倒计时
11 - currentTime: 61 12 + currentTime: 61,
  13 + phone_number: '',
  14 + verification_code: '',
  15 + },
  16 + phoneInput(e) {
  17 + this.setData({phone_number: e.detail.value})
  18 + },
  19 + codeInput(e) {
  20 + this.setData({verification_code: e.detail.value})
12 }, 21 },
13 - getCode: function(options) { 22 + getCode: function (options) {
14 var that = this; 23 var that = this;
15 var currentTime = that.data.currentTime; 24 var currentTime = that.data.currentTime;
16 - interval = setInterval(function() { 25 + interval = setInterval(function () {
17 currentTime--; 26 currentTime--;
18 that.setData({ 27 that.setData({
19 time: currentTime + '秒' 28 time: currentTime + '秒'
@@ -26,74 +35,116 @@ Page({ @@ -26,74 +35,116 @@ Page({
26 disabled: false 35 disabled: false
27 }) 36 })
28 } 37 }
29 - }, 1000) 38 + }, 1000);
  39 + let url = '/user/Profile/getCode';
  40 + let params = {
  41 + mobile: that.data.phone_number,
  42 + };
  43 + let header = {
  44 + "XX-token": wx.getStorageSync('token')
  45 + };
  46 + app.post(url, params,header).then((res) => {
  47 + console.log(res);
  48 + that.setData({verification_code: res.code})
  49 + // console.log(this.data.this_week_test_info);
  50 + })
30 }, 51 },
31 getVerificationCode() { 52 getVerificationCode() {
32 - this.getCode();  
33 var that = this; 53 var that = this;
  54 + if(that.data.phone_number !== '') {
  55 + that.getCode();
34 that.setData({ 56 that.setData({
35 disabled: true 57 disabled: true
36 }) 58 })
  59 + }else {
  60 + that.setData({
  61 + disabled: false
  62 + });
  63 + wx.showToast({title: '请输入手机号!',icon: 'none'})
  64 + }
37 }, 65 },
38 - //确认  
39 - confirm() { 66 + //确认绑定
  67 + formSubmit(e) {
  68 + console.log(e.detail);
  69 + const self = this;
  70 + if(self.data.phone_number === '') {
  71 + wx.showToast({title: '请输入手机号!',icon: 'none'})
  72 + }else if(self.data.verification_code === '') {
  73 + wx.showToast({title: '请输入验证码!',icon: 'none'})
  74 + }else {
  75 + let url = '/user/Profile/bindingMobile';
  76 + let params = {
  77 + _type: 1,
  78 + formId: e.detail.formId,
  79 + mobile: e.detail.value.phone_number,
  80 + verification_code: e.detail.value.verification_code,
  81 + };
  82 + let header = {
  83 + "XX-token": wx.getStorageSync('token')
  84 + };
  85 + app.post(url, params,header).then((res) => {
  86 + console.log('绑定结果',res);
  87 + // self.setData({past_test_info: res})
  88 + // console.log(self.data.this_week_test_info);
  89 + });
40 wx.switchTab({ 90 wx.switchTab({
41 url: '../../index/index' 91 url: '../../index/index'
42 }) 92 })
  93 + }
43 }, 94 },
  95 +
44 /** 96 /**
45 * 生命周期函数--监听页面加载 97 * 生命周期函数--监听页面加载
46 */ 98 */
47 - onLoad: function(options) {  
48 - 99 + onLoad: function (options) {
49 }, 100 },
50 101
51 /** 102 /**
52 * 生命周期函数--监听页面初次渲染完成 103 * 生命周期函数--监听页面初次渲染完成
53 */ 104 */
54 - onReady: function() { 105 + onReady: function () {
55 106
56 }, 107 },
57 108
58 /** 109 /**
59 * 生命周期函数--监听页面显示 110 * 生命周期函数--监听页面显示
60 */ 111 */
61 - onShow: function() { 112 + onShow: function () {
62 113
63 }, 114 },
64 115
65 /** 116 /**
66 * 生命周期函数--监听页面隐藏 117 * 生命周期函数--监听页面隐藏
67 */ 118 */
68 - onHide: function() { 119 + onHide: function () {
69 120
70 }, 121 },
71 122
72 /** 123 /**
73 * 生命周期函数--监听页面卸载 124 * 生命周期函数--监听页面卸载
74 */ 125 */
75 - onUnload: function() { 126 + onUnload: function () {
76 127
77 }, 128 },
78 129
79 /** 130 /**
80 * 页面相关事件处理函数--监听用户下拉动作 131 * 页面相关事件处理函数--监听用户下拉动作
81 */ 132 */
82 - onPullDownRefresh: function() { 133 + onPullDownRefresh: function () {
83 134
84 }, 135 },
85 136
86 /** 137 /**
87 * 页面上拉触底事件的处理函数 138 * 页面上拉触底事件的处理函数
88 */ 139 */
89 - onReachBottom: function() { 140 + onReachBottom: function () {
90 141
91 }, 142 },
92 143
93 /** 144 /**
94 * 用户点击右上角分享 145 * 用户点击右上角分享
95 */ 146 */
96 - onShareAppMessage: function() { 147 + onShareAppMessage: function () {
97 148
98 } 149 }
99 }) 150 })
@@ -4,19 +4,23 @@ @@ -4,19 +4,23 @@
4 <image src="../../../images/logo.png"></image> 4 <image src="../../../images/logo.png"></image>
5 </view> 5 </view>
6 6
  7 + <form bindsubmit="formSubmit" bindreset="formReset" report-submit="true">
7 <view class="phone_box"> 8 <view class="phone_box">
8 - <input type="number" placeholder="请输入您的手机号" placeholder-class="placeholder" class="phone_num"/> 9 + <input type="number" placeholder="请输入您的手机号" placeholder-class="placeholder" class="phone_num"
  10 + value="{{phone_number}}" bindinput="phoneInput" name="phone_number"/>
9 <view class="code_box"> 11 <view class="code_box">
10 - <input type="number" placeholder="输入验证码" placeholder-class="placeholder"/>  
11 - <button class="button" disabled="{{disabled}}" data-id="2" bindtap="getVerificationCode" > 12 + <input type="number" placeholder="输入验证码" placeholder-class="placeholder" value="{{verification_code}}"
  13 + bindinput="codeInput" name="verification_code"/>
  14 + <button class="button" disabled="{{disabled}}" data-id="2" bindtap="getVerificationCode">
12 {{time}} 15 {{time}}
13 </button> 16 </button>
14 </view> 17 </view>
15 </view> 18 </view>
16 19
17 <view class="confirm_box"> 20 <view class="confirm_box">
18 - <view class="confirm_btn" bindtap='confirm'> 21 + <button class="confirm_btn" form-type="submit">
19 <text>确 定</text> 22 <text>确 定</text>
  23 + </button>
20 </view> 24 </view>
21 - </view> 25 + </form>
22 </view> 26 </view>
@@ -5,7 +5,11 @@ @@ -5,7 +5,11 @@
5 * 页面的初始数据 5 * 页面的初始数据
6 */ 6 */
7 data: { 7 data: {
8 - 8 + hasPhone: true,
  9 + },
  10 + //首次登录小程序,跳转到认证手机页面
  11 + goPhoneCode() {
  12 + wx.navigateTo({url: '/pages/index/phone-code/phone-code'})
9 }, 13 },
10 //兑换 14 //兑换
11 charge() { 15 charge() {
1 <!--pages/my/my.wxml--> 1 <!--pages/my/my.wxml-->
2 <view class='box'> 2 <view class='box'>
  3 + <!--蒙层-->
  4 + <view class="modal" wx:if="{{!hasPhone}}" bindtap="goPhoneCode"></view>
3 <view class='header_box '> 5 <view class='header_box '>
4 <view class='head_box'> 6 <view class='head_box'>
5 <view class='head_img'> 7 <view class='head_img'>
@@ -15,6 +15,16 @@ @@ -15,6 +15,16 @@
15 padding: 0 24rpx; 15 padding: 0 24rpx;
16 box-sizing: border-box; 16 box-sizing: border-box;
17 } 17 }
  18 +.modal {
  19 + width: 100%;
  20 + height: 100%;
  21 + position: absolute;
  22 + left: 0;
  23 + top: 0;
  24 + background-color: #000000;
  25 + opacity: 0.8;
  26 + z-index: 10;
  27 +}
18 .icon-bianji2{ 28 .icon-bianji2{
19 font-size: 50rpx; 29 font-size: 50rpx;
20 text-align: center; 30 text-align: center;
1 // pages/protocol/protocol.js 1 // pages/protocol/protocol.js
  2 +const app = getApp();
2 Page({ 3 Page({
3 4
4 /** 5 /**
5 * 页面的初始数据 6 * 页面的初始数据
6 */ 7 */
7 data: { 8 data: {
8 - 9 + service_protocol: {},
  10 + },
  11 + getServiceProtocol() {
  12 + let url = '/wxapp/public/arc_service';
  13 + app.post(url, {},{}).then((res) => {
  14 + console.log(res);
  15 + this.setData({service_protocol: res})
  16 + })
9 }, 17 },
10 -  
11 /** 18 /**
12 * 生命周期函数--监听页面加载 19 * 生命周期函数--监听页面加载
13 */ 20 */
14 onLoad: function (options) { 21 onLoad: function (options) {
15 - 22 + this.getServiceProtocol();
16 }, 23 },
17 24
18 /** 25 /**
1 <!--pages/protocol/protocol.wxml--> 1 <!--pages/protocol/protocol.wxml-->
2 <view class='content_box'> 2 <view class='content_box'>
3 - 还有简短的汉字以内的文字介绍还有简短的汉字以内 的文字介绍还有简短的汉字以内的文字介绍还有简短 的汉字以 还有简短的汉字以内的文字介绍还有简短的汉字以内 的文字介绍还有简短的汉字以内的文字介绍还有简短 的汉字以 3 + <view>{{service_protocol.title}}</view>
  4 + <view>{{service_protocol.description}}</view>
4 </view> 5 </view>
@@ -12,12 +12,7 @@ Page({ @@ -12,12 +12,7 @@ Page({
12 // 获取用户信息 12 // 获取用户信息
13 start(e) { 13 start(e) {
14 let that = this; 14 let that = this;
15 - console.log('点击');  
16 - // wx.redirectTo({  
17 - // url: '/pages/index/index',  
18 - // })  
19 let url = '/wxapp/public/login'; 15 let url = '/wxapp/public/login';
20 - // console.log(that.data.openid)  
21 let params = { 16 let params = {
22 encrypted_data: e.detail.encryptedData, 17 encrypted_data: e.detail.encryptedData,
23 iv: e.detail.iv, 18 iv: e.detail.iv,
@@ -30,11 +25,9 @@ Page({ @@ -30,11 +25,9 @@ Page({
30 wx.setStorageSync('token', ret.token) 25 wx.setStorageSync('token', ret.token)
31 app.globalData.userInfo = ret.user_type 26 app.globalData.userInfo = ret.user_type
32 // console.log('userInfo',app.globalData.userInfo,ret); 27 // console.log('userInfo',app.globalData.userInfo,ret);
33 - if (ret.user_type == 2) {  
34 wx.switchTab({ 28 wx.switchTab({
35 url: '/pages/index/index', 29 url: '/pages/index/index',
36 }) 30 })
37 - }  
38 }) 31 })
39 }, 32 },
40 example() { 33 example() {
@@ -55,11 +48,10 @@ Page({ @@ -55,11 +48,10 @@ Page({
55 let url = '/wxapp/public/getSessionKey' 48 let url = '/wxapp/public/getSessionKey'
56 wx.login({ 49 wx.login({
57 success: function (res) { 50 success: function (res) {
58 - console.log(res) 51 + // console.log(res)
59 app.post(url, { 52 app.post(url, {
60 code: res.code 53 code: res.code
61 }, {}).then((ret) => { 54 }, {}).then((ret) => {
62 - console.log('res',ret);  
63 that.setData({ 55 that.setData({
64 openid: ret.openid, 56 openid: ret.openid,
65 session_key: ret.session_key 57 session_key: ret.session_key
@@ -12,7 +12,7 @@ @@ -12,7 +12,7 @@
12 }, 12 },
13 "compileType": "miniprogram", 13 "compileType": "miniprogram",
14 "libVersion": "2.4.0", 14 "libVersion": "2.4.0",
15 - "appid": "wxaab44bbcfc0e5bc5", 15 + "appid": "wx9053b434e50ee2dd",
16 "projectname": "science", 16 "projectname": "science",
17 "debugOptions": { 17 "debugOptions": {
18 "hidedInDevtools": [] 18 "hidedInDevtools": []
  1 +var _createClass=function(){function a(e,c){for(var b=0;b<c.length;b++){var d=c[b];d.enumerable=d.enumerable||false;d.configurable=true;if("value" in d){d.writable=true}Object.defineProperty(e,d.key,d)}}return function(d,b,c){if(b){a(d.prototype,b)}if(c){a(d,c)}return d}}();function _classCallCheck(a,b){if(!(a instanceof b)){throw new TypeError("Cannot call a class as a function")}}var ERROR_CONF={KEY_ERR:311,KEY_ERR_MSG:"key格式错误",PARAM_ERR:310,PARAM_ERR_MSG:"请求参数信息有误",SYSTEM_ERR:600,SYSTEM_ERR_MSG:"系统错误",WX_ERR_CODE:1000,WX_OK_CODE:200};var BASE_URL="https://apis.map.qq.com/ws/";var URL_SEARCH=BASE_URL+"place/v1/search";var URL_SUGGESTION=BASE_URL+"place/v1/suggestion";var URL_GET_GEOCODER=BASE_URL+"geocoder/v1/";var URL_CITY_LIST=BASE_URL+"district/v1/list";var URL_AREA_LIST=BASE_URL+"district/v1/getchildren";var URL_DISTANCE=BASE_URL+"distance/v1/";var Utils={location2query:function location2query(c){if(typeof c=="string"){return c}var b="";for(var a=0;a<c.length;a++){var e=c[a];if(!!b){b+=";"}if(e.location){b=b+e.location.lat+","+e.location.lng}if(e.latitude&&e.longitude){b=b+e.latitude+","+e.longitude}}return b},getWXLocation:function getWXLocation(c,b,a){wx.getLocation({type:"gcj02",success:c,fail:b,complete:a})},getLocationParam:function getLocationParam(b){if(typeof b=="string"){var a=b.split(",");if(a.length===2){b={latitude:b.split(",")[0],longitude:b.split(",")[1]}}else{b={}}}return b},polyfillParam:function polyfillParam(a){a.success=a.success||function(){};a.fail=a.fail||function(){};a.complete=a.complete||function(){}},checkParamKeyEmpty:function checkParamKeyEmpty(c,b){if(!c[b]){var a=this.buildErrorConfig(ERROR_CONF.PARAM_ERR,ERROR_CONF.PARAM_ERR_MSG+b+"参数格式有误");c.fail(a);c.complete(a);return true}return false},checkKeyword:function checkKeyword(a){return !this.checkParamKeyEmpty(a,"keyword")},checkLocation:function checkLocation(c){var a=this.getLocationParam(c.location);if(!a||!a.latitude||!a.longitude){var b=this.buildErrorConfig(ERROR_CONF.PARAM_ERR,ERROR_CONF.PARAM_ERR_MSG+" location参数格式有误");c.fail(b);c.complete(b);return false}return true},buildErrorConfig:function buildErrorConfig(a,b){return{status:a,message:b}},buildWxRequestConfig:function buildWxRequestConfig(c,a){var b=this;a.header={"content-type":"application/json"};a.method="GET";a.success=function(d){var e=d.data;if(e.status===0){c.success(e)}else{c.fail(e)}};a.fail=function(d){d.statusCode=ERROR_CONF.WX_ERR_CODE;c.fail(b.buildErrorConfig(ERROR_CONF.WX_ERR_CODE,result.errMsg))};a.complete=function(d){var e=+d.statusCode;switch(e){case ERROR_CONF.WX_ERR_CODE:c.complete(b.buildErrorConfig(ERROR_CONF.WX_ERR_CODE,d.errMsg));break;case ERROR_CONF.WX_OK_CODE:var f=d.data;if(f.status===0){c.complete(f)}else{c.complete(b.buildErrorConfig(f.status,f.message))}break;default:c.complete(b.buildErrorConfig(ERROR_CONF.SYSTEM_ERR,ERROR_CONF.SYSTEM_ERR_MSG))}};return a},locationProcess:function locationProcess(f,e,c,a){var d=this;c=c||function(g){g.statusCode=ERROR_CONF.WX_ERR_CODE;f.fail(d.buildErrorConfig(ERROR_CONF.WX_ERR_CODE,g.errMsg))};a=a||function(g){if(g.statusCode==ERROR_CONF.WX_ERR_CODE){f.complete(d.buildErrorConfig(ERROR_CONF.WX_ERR_CODE,g.errMsg))}};if(!f.location){d.getWXLocation(e,c,a)}else{if(d.checkLocation(f)){var b=Utils.getLocationParam(f.location);e(b)}}}};var QQMapWX=function(){function b(i){_classCallCheck(this,b);if(!i.key){throw Error("key值不能为空")}this.key=i.key}_createClass(b,[{key:"search",value:function f(i){var l=this;i=i||{};Utils.polyfillParam(i);if(!Utils.checkKeyword(i)){return}var k={keyword:i.keyword,orderby:i.orderby||"_distance",page_size:i.page_size||10,page_index:i.page_index||1,output:"json",key:l.key};if(i.address_format){k.address_format=i.address_format}if(i.filter){k.filter=i.filter}var n=i.distance||"1000";var j=i.auto_extend||1;var m=function m(o){k.boundary="nearby("+o.latitude+","+o.longitude+","+n+","+j+")";wx.request(Utils.buildWxRequestConfig(i,{url:URL_SEARCH,data:k}))};Utils.locationProcess(i,m)}},{key:"getSuggestion",value:function h(i){var k=this;i=i||{};Utils.polyfillParam(i);if(!Utils.checkKeyword(i)){return}var j={keyword:i.keyword,region:i.region||"全国",region_fix:i.region_fix||0,policy:i.policy||0,output:"json",key:k.key};wx.request(Utils.buildWxRequestConfig(i,{url:URL_SUGGESTION,data:j}))}},{key:"reverseGeocoder",value:function a(i){var k=this;i=i||{};Utils.polyfillParam(i);var j={coord_type:i.coord_type||5,get_poi:i.get_poi||0,output:"json",key:k.key};if(i.poi_options){j.poi_options=i.poi_options}var l=function l(m){j.location=m.latitude+","+m.longitude;wx.request(Utils.buildWxRequestConfig(i,{url:URL_GET_GEOCODER,data:j}))};Utils.locationProcess(i,l)}},{key:"geocoder",value:function g(i){var k=this;i=i||{};Utils.polyfillParam(i);if(Utils.checkParamKeyEmpty(i,"address")){return}var j={address:i.address,output:"json",key:k.key};wx.request(Utils.buildWxRequestConfig(i,{url:URL_GET_GEOCODER,data:j}))}},{key:"getCityList",value:function c(i){var k=this;i=i||{};Utils.polyfillParam(i);var j={output:"json",key:k.key};
  2 +wx.request(Utils.buildWxRequestConfig(i,{url:URL_CITY_LIST,data:j}))}},{key:"getDistrictByCityId",value:function d(i){var k=this;i=i||{};Utils.polyfillParam(i);if(Utils.checkParamKeyEmpty(i,"id")){return}var j={id:i.id||"",output:"json",key:k.key};wx.request(Utils.buildWxRequestConfig(i,{url:URL_AREA_LIST,data:j}))}},{key:"calculateDistance",value:function e(i){var k=this;i=i||{};Utils.polyfillParam(i);if(Utils.checkParamKeyEmpty(i,"to")){return}var j={mode:i.mode||"walking",to:Utils.location2query(i.to),output:"json",key:k.key};var l=function l(m){j.from=m.latitude+","+m.longitude;wx.request(Utils.buildWxRequestConfig(i,{url:URL_DISTANCE,data:j}))};if(i.from){i.location=i.from}Utils.locationProcess(i,l)}}]);return b}();module.exports=QQMapWX;
@@ -14,6 +14,32 @@ const formatNumber = n => { @@ -14,6 +14,32 @@ const formatNumber = n => {
14 return n[1] ? n : '0' + n 14 return n[1] ? n : '0' + n
15 } 15 }
16 16
  17 +/**
  18 + * 时间戳转化为年 月 日 时 分 秒
  19 + * number: 传入时间戳
  20 + * format:返回格式,支持自定义,但参数必须与formateArr里保持一致
  21 + */
  22 +function formatTimeTwo(number, format) {
  23 +
  24 + var formateArr = ['Y', 'M', 'D', 'h', 'm', 's'];
  25 + var returnArr = [];
  26 +
  27 + var date = new Date(number * 1000);
  28 + returnArr.push(date.getFullYear());
  29 + returnArr.push(formatNumber(date.getMonth() + 1));
  30 + returnArr.push(formatNumber(date.getDate()));
  31 +
  32 + returnArr.push(formatNumber(date.getHours()));
  33 + returnArr.push(formatNumber(date.getMinutes()));
  34 + returnArr.push(formatNumber(date.getSeconds()));
  35 +
  36 + for (var i in returnArr) {
  37 + format = format.replace(formateArr[i], returnArr[i]);
  38 + }
  39 + return format;
  40 +}
  41 +
17 module.exports = { 42 module.exports = {
18 - formatTime: formatTime 43 + formatTime: formatTime,
  44 + formatTimeTwo: formatTimeTwo
19 } 45 }
  1 +/**
  2 + * html2Json 改造来自: https://github.com/Jxck/html2json
  3 + *
  4 + *
  5 + * author: Di (微信小程序开发工程师)
  6 + * organization: WeAppDev(微信小程序开发论坛)(http://weappdev.com)
  7 + * 垂直微信小程序开发交流社区
  8 + *
  9 + * github地址: https://github.com/icindy/wxParse
  10 + *
  11 + * for: 微信小程序富文本解析
  12 + * detail : http://weappdev.com/t/wxparse-alpha0-1-html-markdown/184
  13 + */
  14 +
  15 +var __placeImgeUrlHttps = "https";
  16 +var __emojisReg = '';
  17 +var __emojisBaseSrc = '';
  18 +var __emojis = {};
  19 +var wxDiscode = require('./wxDiscode.js');
  20 +var HTMLParser = require('./htmlparser.js');
  21 +// Empty Elements - HTML 5
  22 +var empty = makeMap("area,base,basefont,br,col,frame,hr,img,input,link,meta,param,embed,command,keygen,source,track,wbr");
  23 +// Block Elements - HTML 5
  24 +var block = makeMap("br,a,code,address,article,applet,aside,audio,blockquote,button,canvas,center,dd,del,dir,div,dl,dt,fieldset,figcaption,figure,footer,form,frameset,h1,h2,h3,h4,h5,h6,header,hgroup,hr,iframe,ins,isindex,li,map,menu,noframes,noscript,object,ol,output,p,pre,section,script,table,tbody,td,tfoot,th,thead,tr,ul,video");
  25 +
  26 +// Inline Elements - HTML 5
  27 +var inline = makeMap("abbr,acronym,applet,b,basefont,bdo,big,button,cite,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,object,q,s,samp,script,select,small,span,strike,strong,sub,sup,textarea,tt,u,var");
  28 +
  29 +// Elements that you can, intentionally, leave open
  30 +// (and which close themselves)
  31 +var closeSelf = makeMap("colgroup,dd,dt,li,options,p,td,tfoot,th,thead,tr");
  32 +
  33 +// Attributes that have their values filled in disabled="disabled"
  34 +var fillAttrs = makeMap("checked,compact,declare,defer,disabled,ismap,multiple,nohref,noresize,noshade,nowrap,readonly,selected");
  35 +
  36 +// Special Elements (can contain anything)
  37 +var special = makeMap("wxxxcode-style,script,style,view,scroll-view,block");
  38 +function makeMap(str) {
  39 + var obj = {}, items = str.split(",");
  40 + for (var i = 0; i < items.length; i++)
  41 + obj[items[i]] = true;
  42 + return obj;
  43 +}
  44 +
  45 +function q(v) {
  46 + return '"' + v + '"';
  47 +}
  48 +
  49 +function removeDOCTYPE(html) {
  50 + return html
  51 + .replace(/<\?xml.*\?>\n/, '')
  52 + .replace(/<.*!doctype.*\>\n/, '')
  53 + .replace(/<.*!DOCTYPE.*\>\n/, '');
  54 +}
  55 +
  56 +function trimHtml(html) {
  57 + return html
  58 + .replace(/\r?\n+/g, '')
  59 + .replace(/<!--.*?-->/ig, '')
  60 + .replace(/\/\*.*?\*\//ig, '')
  61 + .replace(/[ ]+</ig, '<')
  62 +}
  63 +
  64 +
  65 +function html2json(html, bindName) {
  66 + //处理字符串
  67 + html = removeDOCTYPE(html);
  68 + html = trimHtml(html);
  69 + html = wxDiscode.strDiscode(html);
  70 + //生成node节点
  71 + var bufArray = [];
  72 + var results = {
  73 + node: bindName,
  74 + nodes: [],
  75 + images:[],
  76 + imageUrls:[]
  77 + };
  78 + var index = 0;
  79 + HTMLParser(html, {
  80 + start: function (tag, attrs, unary) {
  81 + //debug(tag, attrs, unary);
  82 + // node for this element
  83 + var node = {
  84 + node: 'element',
  85 + tag: tag,
  86 + };
  87 +
  88 + if (bufArray.length === 0) {
  89 + node.index = index.toString()
  90 + index += 1
  91 + } else {
  92 + var parent = bufArray[0];
  93 + if (parent.nodes === undefined) {
  94 + parent.nodes = [];
  95 + }
  96 + node.index = parent.index + '.' + parent.nodes.length
  97 + }
  98 +
  99 + if (block[tag]) {
  100 + node.tagType = "block";
  101 + } else if (inline[tag]) {
  102 + node.tagType = "inline";
  103 + } else if (closeSelf[tag]) {
  104 + node.tagType = "closeSelf";
  105 + }
  106 +
  107 + if (attrs.length !== 0) {
  108 + node.attr = attrs.reduce(function (pre, attr) {
  109 + var name = attr.name;
  110 + var value = attr.value;
  111 + if (name == 'class') {
  112 + // console.dir(value);
  113 + // value = value.join("")
  114 + node.classStr = value;
  115 + }
  116 + // has multi attibutes
  117 + // make it array of attribute
  118 + if (name == 'style') {
  119 + console.dir(value);
  120 + // value = value.join("")
  121 + node.styleStr = value;
  122 + }
  123 + if (value.match(/ /)) {
  124 + value = value.split(' ');
  125 + }
  126 +
  127 +
  128 + // if attr already exists
  129 + // merge it
  130 + if (pre[name]) {
  131 + if (Array.isArray(pre[name])) {
  132 + // already array, push to last
  133 + pre[name].push(value);
  134 + } else {
  135 + // single value, make it array
  136 + pre[name] = [pre[name], value];
  137 + }
  138 + } else {
  139 + // not exist, put it
  140 + pre[name] = value;
  141 + }
  142 +
  143 + return pre;
  144 + }, {});
  145 + }
  146 +
  147 + //对img添加额外数据
  148 + if (node.tag === 'img') {
  149 + node.imgIndex = results.images.length;
  150 + var imgUrl = node.attr.src;
  151 + if (imgUrl[0] == '') {
  152 + imgUrl.splice(0, 1);
  153 + }
  154 + imgUrl = wxDiscode.urlToHttpUrl(imgUrl, __placeImgeUrlHttps);
  155 + node.attr.src = imgUrl;
  156 + node.from = bindName;
  157 + results.images.push(node);
  158 + results.imageUrls.push(imgUrl);
  159 + }
  160 +
  161 + // 处理font标签样式属性
  162 + if (node.tag === 'font') {
  163 + var fontSize = ['x-small', 'small', 'medium', 'large', 'x-large', 'xx-large', '-webkit-xxx-large'];
  164 + var styleAttrs = {
  165 + 'color': 'color',
  166 + 'face': 'font-family',
  167 + 'size': 'font-size'
  168 + };
  169 + if (!node.attr.style) node.attr.style = [];
  170 + if (!node.styleStr) node.styleStr = '';
  171 + for (var key in styleAttrs) {
  172 + if (node.attr[key]) {
  173 + var value = key === 'size' ? fontSize[node.attr[key]-1] : node.attr[key];
  174 + node.attr.style.push(styleAttrs[key]);
  175 + node.attr.style.push(value);
  176 + node.styleStr += styleAttrs[key] + ': ' + value + ';';
  177 + }
  178 + }
  179 + }
  180 +
  181 + //临时记录source资源
  182 + if(node.tag === 'source'){
  183 + results.source = node.attr.src;
  184 + }
  185 +
  186 + if (unary) {
  187 + // if this tag doesn't have end tag
  188 + // like <img src="hoge.png"/>
  189 + // add to parents
  190 + var parent = bufArray[0] || results;
  191 + if (parent.nodes === undefined) {
  192 + parent.nodes = [];
  193 + }
  194 + parent.nodes.push(node);
  195 + } else {
  196 + bufArray.unshift(node);
  197 + }
  198 + },
  199 + end: function (tag) {
  200 + //debug(tag);
  201 + // merge into parent tag
  202 + var node = bufArray.shift();
  203 + if (node.tag !== tag) console.error('invalid state: mismatch end tag');
  204 +
  205 + //当有缓存source资源时于于video补上src资源
  206 + if(node.tag === 'video' && results.source){
  207 + node.attr.src = results.source;
  208 + delete results.source;
  209 + }
  210 +
  211 + if (bufArray.length === 0) {
  212 + results.nodes.push(node);
  213 + } else {
  214 + var parent = bufArray[0];
  215 + if (parent.nodes === undefined) {
  216 + parent.nodes = [];
  217 + }
  218 + parent.nodes.push(node);
  219 + }
  220 + },
  221 + chars: function (text) {
  222 + //debug(text);
  223 + var node = {
  224 + node: 'text',
  225 + text: text,
  226 + textArray:transEmojiStr(text)
  227 + };
  228 +
  229 + if (bufArray.length === 0) {
  230 + node.index = index.toString()
  231 + index += 1
  232 + results.nodes.push(node);
  233 + } else {
  234 + var parent = bufArray[0];
  235 + if (parent.nodes === undefined) {
  236 + parent.nodes = [];
  237 + }
  238 + node.index = parent.index + '.' + parent.nodes.length
  239 + parent.nodes.push(node);
  240 + }
  241 + },
  242 + comment: function (text) {
  243 + //debug(text);
  244 + // var node = {
  245 + // node: 'comment',
  246 + // text: text,
  247 + // };
  248 + // var parent = bufArray[0];
  249 + // if (parent.nodes === undefined) {
  250 + // parent.nodes = [];
  251 + // }
  252 + // parent.nodes.push(node);
  253 + },
  254 + });
  255 + return results;
  256 +};
  257 +
  258 +function transEmojiStr(str){
  259 + // var eReg = new RegExp("["+__reg+' '+"]");
  260 +// str = str.replace(/\[([^\[\]]+)\]/g,':$1:')
  261 +
  262 + var emojiObjs = [];
  263 + //如果正则表达式为空
  264 + if(__emojisReg.length == 0 || !__emojis){
  265 + var emojiObj = {}
  266 + emojiObj.node = "text";
  267 + emojiObj.text = str;
  268 + array = [emojiObj];
  269 + return array;
  270 + }
  271 + //这个地方需要调整
  272 + str = str.replace(/\[([^\[\]]+)\]/g,':$1:')
  273 + var eReg = new RegExp("[:]");
  274 + var array = str.split(eReg);
  275 + for(var i = 0; i < array.length; i++){
  276 + var ele = array[i];
  277 + var emojiObj = {};
  278 + if(__emojis[ele]){
  279 + emojiObj.node = "element";
  280 + emojiObj.tag = "emoji";
  281 + emojiObj.text = __emojis[ele];
  282 + emojiObj.baseSrc= __emojisBaseSrc;
  283 + }else{
  284 + emojiObj.node = "text";
  285 + emojiObj.text = ele;
  286 + }
  287 + emojiObjs.push(emojiObj);
  288 + }
  289 +
  290 + return emojiObjs;
  291 +}
  292 +
  293 +function emojisInit(reg='',baseSrc="/wxParse/emojis/",emojis){
  294 + __emojisReg = reg;
  295 + __emojisBaseSrc=baseSrc;
  296 + __emojis=emojis;
  297 +}
  298 +
  299 +module.exports = {
  300 + html2json: html2json,
  301 + emojisInit:emojisInit
  302 +};
  303 +
  1 +/**
  2 + *
  3 + * htmlParser改造自: https://github.com/blowsie/Pure-JavaScript-HTML5-Parser
  4 + *
  5 + * author: Di (微信小程序开发工程师)
  6 + * organization: WeAppDev(微信小程序开发论坛)(http://weappdev.com)
  7 + * 垂直微信小程序开发交流社区
  8 + *
  9 + * github地址: https://github.com/icindy/wxParse
  10 + *
  11 + * for: 微信小程序富文本解析
  12 + * detail : http://weappdev.com/t/wxparse-alpha0-1-html-markdown/184
  13 + */
  14 +// Regular Expressions for parsing tags and attributes
  15 +var startTag = /^<([-A-Za-z0-9_]+)((?:\s+[a-zA-Z_:][-a-zA-Z0-9_:.]*(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)>/,
  16 + endTag = /^<\/([-A-Za-z0-9_]+)[^>]*>/,
  17 + attr = /([a-zA-Z_:][-a-zA-Z0-9_:.]*)(?:\s*=\s*(?:(?:"((?:\\.|[^"])*)")|(?:'((?:\\.|[^'])*)')|([^>\s]+)))?/g;
  18 +
  19 +// Empty Elements - HTML 5
  20 +var empty = makeMap("area,base,basefont,br,col,frame,hr,img,input,link,meta,param,embed,command,keygen,source,track,wbr");
  21 +
  22 +// Block Elements - HTML 5
  23 +var block = makeMap("a,address,code,article,applet,aside,audio,blockquote,button,canvas,center,dd,del,dir,div,dl,dt,fieldset,figcaption,figure,footer,form,frameset,h1,h2,h3,h4,h5,h6,header,hgroup,hr,iframe,ins,isindex,li,map,menu,noframes,noscript,object,ol,output,p,pre,section,script,table,tbody,td,tfoot,th,thead,tr,ul,video");
  24 +
  25 +// Inline Elements - HTML 5
  26 +var inline = makeMap("abbr,acronym,applet,b,basefont,bdo,big,br,button,cite,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,object,q,s,samp,script,select,small,span,strike,strong,sub,sup,textarea,tt,u,var");
  27 +
  28 +// Elements that you can, intentionally, leave open
  29 +// (and which close themselves)
  30 +var closeSelf = makeMap("colgroup,dd,dt,li,options,p,td,tfoot,th,thead,tr");
  31 +
  32 +// Attributes that have their values filled in disabled="disabled"
  33 +var fillAttrs = makeMap("checked,compact,declare,defer,disabled,ismap,multiple,nohref,noresize,noshade,nowrap,readonly,selected");
  34 +
  35 +// Special Elements (can contain anything)
  36 +var special = makeMap("wxxxcode-style,script,style,view,scroll-view,block");
  37 +
  38 +function HTMLParser(html, handler) {
  39 + var index, chars, match, stack = [], last = html;
  40 + stack.last = function () {
  41 + return this[this.length - 1];
  42 + };
  43 +
  44 + while (html) {
  45 + chars = true;
  46 +
  47 + // Make sure we're not in a script or style element
  48 + if (!stack.last() || !special[stack.last()]) {
  49 +
  50 + // Comment
  51 + if (html.indexOf("<!--") == 0) {
  52 + index = html.indexOf("-->");
  53 +
  54 + if (index >= 0) {
  55 + if (handler.comment)
  56 + handler.comment(html.substring(4, index));
  57 + html = html.substring(index + 3);
  58 + chars = false;
  59 + }
  60 +
  61 + // end tag
  62 + } else if (html.indexOf("</") == 0) {
  63 + match = html.match(endTag);
  64 +
  65 + if (match) {
  66 + html = html.substring(match[0].length);
  67 + match[0].replace(endTag, parseEndTag);
  68 + chars = false;
  69 + }
  70 +
  71 + // start tag
  72 + } else if (html.indexOf("<") == 0) {
  73 + match = html.match(startTag);
  74 +
  75 + if (match) {
  76 + html = html.substring(match[0].length);
  77 + match[0].replace(startTag, parseStartTag);
  78 + chars = false;
  79 + }
  80 + }
  81 +
  82 + if (chars) {
  83 + index = html.indexOf("<");
  84 + var text = ''
  85 + while (index === 0) {
  86 + text += "<";
  87 + html = html.substring(1);
  88 + index = html.indexOf("<");
  89 + }
  90 + text += index < 0 ? html : html.substring(0, index);
  91 + html = index < 0 ? "" : html.substring(index);
  92 +
  93 + if (handler.chars)
  94 + handler.chars(text);
  95 + }
  96 +
  97 + } else {
  98 +
  99 + html = html.replace(new RegExp("([\\s\\S]*?)<\/" + stack.last() + "[^>]*>"), function (all, text) {
  100 + text = text.replace(/<!--([\s\S]*?)-->|<!\[CDATA\[([\s\S]*?)]]>/g, "$1$2");
  101 + if (handler.chars)
  102 + handler.chars(text);
  103 +
  104 + return "";
  105 + });
  106 +
  107 +
  108 + parseEndTag("", stack.last());
  109 + }
  110 +
  111 + if (html == last)
  112 + throw "Parse Error: " + html;
  113 + last = html;
  114 + }
  115 +
  116 + // Clean up any remaining tags
  117 + parseEndTag();
  118 +
  119 + function parseStartTag(tag, tagName, rest, unary) {
  120 + tagName = tagName.toLowerCase();
  121 +
  122 + if (block[tagName]) {
  123 + while (stack.last() && inline[stack.last()]) {
  124 + parseEndTag("", stack.last());
  125 + }
  126 + }
  127 +
  128 + if (closeSelf[tagName] && stack.last() == tagName) {
  129 + parseEndTag("", tagName);
  130 + }
  131 +
  132 + unary = empty[tagName] || !!unary;
  133 +
  134 + if (!unary)
  135 + stack.push(tagName);
  136 +
  137 + if (handler.start) {
  138 + var attrs = [];
  139 +
  140 + rest.replace(attr, function (match, name) {
  141 + var value = arguments[2] ? arguments[2] :
  142 + arguments[3] ? arguments[3] :
  143 + arguments[4] ? arguments[4] :
  144 + fillAttrs[name] ? name : "";
  145 +
  146 + attrs.push({
  147 + name: name,
  148 + value: value,
  149 + escaped: value.replace(/(^|[^\\])"/g, '$1\\\"') //"
  150 + });
  151 + });
  152 +
  153 + if (handler.start) {
  154 + handler.start(tagName, attrs, unary);
  155 + }
  156 +
  157 + }
  158 + }
  159 +
  160 + function parseEndTag(tag, tagName) {
  161 + // If no tag name is provided, clean shop
  162 + if (!tagName)
  163 + var pos = 0;
  164 +
  165 + // Find the closest opened tag of the same type
  166 + else {
  167 + tagName = tagName.toLowerCase();
  168 + for (var pos = stack.length - 1; pos >= 0; pos--)
  169 + if (stack[pos] == tagName)
  170 + break;
  171 + }
  172 + if (pos >= 0) {
  173 + // Close all the open elements, up the stack
  174 + for (var i = stack.length - 1; i >= pos; i--)
  175 + if (handler.end)
  176 + handler.end(stack[i]);
  177 +
  178 + // Remove the open elements from the stack
  179 + stack.length = pos;
  180 + }
  181 + }
  182 +};
  183 +
  184 +
  185 +function makeMap(str) {
  186 + var obj = {}, items = str.split(",");
  187 + for (var i = 0; i < items.length; i++)
  188 + obj[items[i]] = true;
  189 + return obj;
  190 +}
  191 +
  192 +module.exports = HTMLParser;
  1 +/**
  2 + *
  3 + * showdown: https://github.com/showdownjs/showdown
  4 + *
  5 + * author: Di (微信小程序开发工程师)
  6 + * organization: WeAppDev(微信小程序开发论坛)(http://weappdev.com)
  7 + * 垂直微信小程序开发交流社区
  8 + *
  9 + * github地址: https://github.com/icindy/wxParse
  10 + *
  11 + * for: 微信小程序富文本解析
  12 + * detail : http://weappdev.com/t/wxparse-alpha0-1-html-markdown/184
  13 + */
  14 +
  15 +function getDefaultOpts(simple) {
  16 + 'use strict';
  17 +
  18 + var defaultOptions = {
  19 + omitExtraWLInCodeBlocks: {
  20 + defaultValue: false,
  21 + describe: 'Omit the default extra whiteline added to code blocks',
  22 + type: 'boolean'
  23 + },
  24 + noHeaderId: {
  25 + defaultValue: false,
  26 + describe: 'Turn on/off generated header id',
  27 + type: 'boolean'
  28 + },
  29 + prefixHeaderId: {
  30 + defaultValue: false,
  31 + describe: 'Specify a prefix to generated header ids',
  32 + type: 'string'
  33 + },
  34 + headerLevelStart: {
  35 + defaultValue: false,
  36 + describe: 'The header blocks level start',
  37 + type: 'integer'
  38 + },
  39 + parseImgDimensions: {
  40 + defaultValue: false,
  41 + describe: 'Turn on/off image dimension parsing',
  42 + type: 'boolean'
  43 + },
  44 + simplifiedAutoLink: {
  45 + defaultValue: false,
  46 + describe: 'Turn on/off GFM autolink style',
  47 + type: 'boolean'
  48 + },
  49 + literalMidWordUnderscores: {
  50 + defaultValue: false,
  51 + describe: 'Parse midword underscores as literal underscores',
  52 + type: 'boolean'
  53 + },
  54 + strikethrough: {
  55 + defaultValue: false,
  56 + describe: 'Turn on/off strikethrough support',
  57 + type: 'boolean'
  58 + },
  59 + tables: {
  60 + defaultValue: false,
  61 + describe: 'Turn on/off tables support',
  62 + type: 'boolean'
  63 + },
  64 + tablesHeaderId: {
  65 + defaultValue: false,
  66 + describe: 'Add an id to table headers',
  67 + type: 'boolean'
  68 + },
  69 + ghCodeBlocks: {
  70 + defaultValue: true,
  71 + describe: 'Turn on/off GFM fenced code blocks support',
  72 + type: 'boolean'
  73 + },
  74 + tasklists: {
  75 + defaultValue: false,
  76 + describe: 'Turn on/off GFM tasklist support',
  77 + type: 'boolean'
  78 + },
  79 + smoothLivePreview: {
  80 + defaultValue: false,
  81 + describe: 'Prevents weird effects in live previews due to incomplete input',
  82 + type: 'boolean'
  83 + },
  84 + smartIndentationFix: {
  85 + defaultValue: false,
  86 + description: 'Tries to smartly fix identation in es6 strings',
  87 + type: 'boolean'
  88 + }
  89 + };
  90 + if (simple === false) {
  91 + return JSON.parse(JSON.stringify(defaultOptions));
  92 + }
  93 + var ret = {};
  94 + for (var opt in defaultOptions) {
  95 + if (defaultOptions.hasOwnProperty(opt)) {
  96 + ret[opt] = defaultOptions[opt].defaultValue;
  97 + }
  98 + }
  99 + return ret;
  100 +}
  101 +
  102 +/**
  103 + * Created by Tivie on 06-01-2015.
  104 + */
  105 +
  106 +// Private properties
  107 +var showdown = {},
  108 + parsers = {},
  109 + extensions = {},
  110 + globalOptions = getDefaultOpts(true),
  111 + flavor = {
  112 + github: {
  113 + omitExtraWLInCodeBlocks: true,
  114 + prefixHeaderId: 'user-content-',
  115 + simplifiedAutoLink: true,
  116 + literalMidWordUnderscores: true,
  117 + strikethrough: true,
  118 + tables: true,
  119 + tablesHeaderId: true,
  120 + ghCodeBlocks: true,
  121 + tasklists: true
  122 + },
  123 + vanilla: getDefaultOpts(true)
  124 + };
  125 +
  126 +/**
  127 + * helper namespace
  128 + * @type {{}}
  129 + */
  130 +showdown.helper = {};
  131 +
  132 +/**
  133 + * TODO LEGACY SUPPORT CODE
  134 + * @type {{}}
  135 + */
  136 +showdown.extensions = {};
  137 +
  138 +/**
  139 + * Set a global option
  140 + * @static
  141 + * @param {string} key
  142 + * @param {*} value
  143 + * @returns {showdown}
  144 + */
  145 +showdown.setOption = function (key, value) {
  146 + 'use strict';
  147 + globalOptions[key] = value;
  148 + return this;
  149 +};
  150 +
  151 +/**
  152 + * Get a global option
  153 + * @static
  154 + * @param {string} key
  155 + * @returns {*}
  156 + */
  157 +showdown.getOption = function (key) {
  158 + 'use strict';
  159 + return globalOptions[key];
  160 +};
  161 +
  162 +/**
  163 + * Get the global options
  164 + * @static
  165 + * @returns {{}}
  166 + */
  167 +showdown.getOptions = function () {
  168 + 'use strict';
  169 + return globalOptions;
  170 +};
  171 +
  172 +/**
  173 + * Reset global options to the default values
  174 + * @static
  175 + */
  176 +showdown.resetOptions = function () {
  177 + 'use strict';
  178 + globalOptions = getDefaultOpts(true);
  179 +};
  180 +
  181 +/**
  182 + * Set the flavor showdown should use as default
  183 + * @param {string} name
  184 + */
  185 +showdown.setFlavor = function (name) {
  186 + 'use strict';
  187 + if (flavor.hasOwnProperty(name)) {
  188 + var preset = flavor[name];
  189 + for (var option in preset) {
  190 + if (preset.hasOwnProperty(option)) {
  191 + globalOptions[option] = preset[option];
  192 + }
  193 + }
  194 + }
  195 +};
  196 +
  197 +/**
  198 + * Get the default options
  199 + * @static
  200 + * @param {boolean} [simple=true]
  201 + * @returns {{}}
  202 + */
  203 +showdown.getDefaultOptions = function (simple) {
  204 + 'use strict';
  205 + return getDefaultOpts(simple);
  206 +};
  207 +
  208 +/**
  209 + * Get or set a subParser
  210 + *
  211 + * subParser(name) - Get a registered subParser
  212 + * subParser(name, func) - Register a subParser
  213 + * @static
  214 + * @param {string} name
  215 + * @param {function} [func]
  216 + * @returns {*}
  217 + */
  218 +showdown.subParser = function (name, func) {
  219 + 'use strict';
  220 + if (showdown.helper.isString(name)) {
  221 + if (typeof func !== 'undefined') {
  222 + parsers[name] = func;
  223 + } else {
  224 + if (parsers.hasOwnProperty(name)) {
  225 + return parsers[name];
  226 + } else {
  227 + throw Error('SubParser named ' + name + ' not registered!');
  228 + }
  229 + }
  230 + }
  231 +};
  232 +
  233 +/**
  234 + * Gets or registers an extension
  235 + * @static
  236 + * @param {string} name
  237 + * @param {object|function=} ext
  238 + * @returns {*}
  239 + */
  240 +showdown.extension = function (name, ext) {
  241 + 'use strict';
  242 +
  243 + if (!showdown.helper.isString(name)) {
  244 + throw Error('Extension \'name\' must be a string');
  245 + }
  246 +
  247 + name = showdown.helper.stdExtName(name);
  248 +
  249 + // Getter
  250 + if (showdown.helper.isUndefined(ext)) {
  251 + if (!extensions.hasOwnProperty(name)) {
  252 + throw Error('Extension named ' + name + ' is not registered!');
  253 + }
  254 + return extensions[name];
  255 +
  256 + // Setter
  257 + } else {
  258 + // Expand extension if it's wrapped in a function
  259 + if (typeof ext === 'function') {
  260 + ext = ext();
  261 + }
  262 +
  263 + // Ensure extension is an array
  264 + if (!showdown.helper.isArray(ext)) {
  265 + ext = [ext];
  266 + }
  267 +
  268 + var validExtension = validate(ext, name);
  269 +
  270 + if (validExtension.valid) {
  271 + extensions[name] = ext;
  272 + } else {
  273 + throw Error(validExtension.error);
  274 + }
  275 + }
  276 +};
  277 +
  278 +/**
  279 + * Gets all extensions registered
  280 + * @returns {{}}
  281 + */
  282 +showdown.getAllExtensions = function () {
  283 + 'use strict';
  284 + return extensions;
  285 +};
  286 +
  287 +/**
  288 + * Remove an extension
  289 + * @param {string} name
  290 + */
  291 +showdown.removeExtension = function (name) {
  292 + 'use strict';
  293 + delete extensions[name];
  294 +};
  295 +
  296 +/**
  297 + * Removes all extensions
  298 + */
  299 +showdown.resetExtensions = function () {
  300 + 'use strict';
  301 + extensions = {};
  302 +};
  303 +
  304 +/**
  305 + * Validate extension
  306 + * @param {array} extension
  307 + * @param {string} name
  308 + * @returns {{valid: boolean, error: string}}
  309 + */
  310 +function validate(extension, name) {
  311 + 'use strict';
  312 +
  313 + var errMsg = (name) ? 'Error in ' + name + ' extension->' : 'Error in unnamed extension',
  314 + ret = {
  315 + valid: true,
  316 + error: ''
  317 + };
  318 +
  319 + if (!showdown.helper.isArray(extension)) {
  320 + extension = [extension];
  321 + }
  322 +
  323 + for (var i = 0; i < extension.length; ++i) {
  324 + var baseMsg = errMsg + ' sub-extension ' + i + ': ',
  325 + ext = extension[i];
  326 + if (typeof ext !== 'object') {
  327 + ret.valid = false;
  328 + ret.error = baseMsg + 'must be an object, but ' + typeof ext + ' given';
  329 + return ret;
  330 + }
  331 +
  332 + if (!showdown.helper.isString(ext.type)) {
  333 + ret.valid = false;
  334 + ret.error = baseMsg + 'property "type" must be a string, but ' + typeof ext.type + ' given';
  335 + return ret;
  336 + }
  337 +
  338 + var type = ext.type = ext.type.toLowerCase();
  339 +
  340 + // normalize extension type
  341 + if (type === 'language') {
  342 + type = ext.type = 'lang';
  343 + }
  344 +
  345 + if (type === 'html') {
  346 + type = ext.type = 'output';
  347 + }
  348 +
  349 + if (type !== 'lang' && type !== 'output' && type !== 'listener') {
  350 + ret.valid = false;
  351 + ret.error = baseMsg + 'type ' + type + ' is not recognized. Valid values: "lang/language", "output/html" or "listener"';
  352 + return ret;
  353 + }
  354 +
  355 + if (type === 'listener') {
  356 + if (showdown.helper.isUndefined(ext.listeners)) {
  357 + ret.valid = false;
  358 + ret.error = baseMsg + '. Extensions of type "listener" must have a property called "listeners"';
  359 + return ret;
  360 + }
  361 + } else {
  362 + if (showdown.helper.isUndefined(ext.filter) && showdown.helper.isUndefined(ext.regex)) {
  363 + ret.valid = false;
  364 + ret.error = baseMsg + type + ' extensions must define either a "regex" property or a "filter" method';
  365 + return ret;
  366 + }
  367 + }
  368 +
  369 + if (ext.listeners) {
  370 + if (typeof ext.listeners !== 'object') {
  371 + ret.valid = false;
  372 + ret.error = baseMsg + '"listeners" property must be an object but ' + typeof ext.listeners + ' given';
  373 + return ret;
  374 + }
  375 + for (var ln in ext.listeners) {
  376 + if (ext.listeners.hasOwnProperty(ln)) {
  377 + if (typeof ext.listeners[ln] !== 'function') {
  378 + ret.valid = false;
  379 + ret.error = baseMsg + '"listeners" property must be an hash of [event name]: [callback]. listeners.' + ln +
  380 + ' must be a function but ' + typeof ext.listeners[ln] + ' given';
  381 + return ret;
  382 + }
  383 + }
  384 + }
  385 + }
  386 +
  387 + if (ext.filter) {
  388 + if (typeof ext.filter !== 'function') {
  389 + ret.valid = false;
  390 + ret.error = baseMsg + '"filter" must be a function, but ' + typeof ext.filter + ' given';
  391 + return ret;
  392 + }
  393 + } else if (ext.regex) {
  394 + if (showdown.helper.isString(ext.regex)) {
  395 + ext.regex = new RegExp(ext.regex, 'g');
  396 + }
  397 + if (!ext.regex instanceof RegExp) {
  398 + ret.valid = false;
  399 + ret.error = baseMsg + '"regex" property must either be a string or a RegExp object, but ' + typeof ext.regex + ' given';
  400 + return ret;
  401 + }
  402 + if (showdown.helper.isUndefined(ext.replace)) {
  403 + ret.valid = false;
  404 + ret.error = baseMsg + '"regex" extensions must implement a replace string or function';
  405 + return ret;
  406 + }
  407 + }
  408 + }
  409 + return ret;
  410 +}
  411 +
  412 +/**
  413 + * Validate extension
  414 + * @param {object} ext
  415 + * @returns {boolean}
  416 + */
  417 +showdown.validateExtension = function (ext) {
  418 + 'use strict';
  419 +
  420 + var validateExtension = validate(ext, null);
  421 + if (!validateExtension.valid) {
  422 + console.warn(validateExtension.error);
  423 + return false;
  424 + }
  425 + return true;
  426 +};
  427 +
  428 +/**
  429 + * showdownjs helper functions
  430 + */
  431 +
  432 +if (!showdown.hasOwnProperty('helper')) {
  433 + showdown.helper = {};
  434 +}
  435 +
  436 +/**
  437 + * Check if var is string
  438 + * @static
  439 + * @param {string} a
  440 + * @returns {boolean}
  441 + */
  442 +showdown.helper.isString = function isString(a) {
  443 + 'use strict';
  444 + return (typeof a === 'string' || a instanceof String);
  445 +};
  446 +
  447 +/**
  448 + * Check if var is a function
  449 + * @static
  450 + * @param {string} a
  451 + * @returns {boolean}
  452 + */
  453 +showdown.helper.isFunction = function isFunction(a) {
  454 + 'use strict';
  455 + var getType = {};
  456 + return a && getType.toString.call(a) === '[object Function]';
  457 +};
  458 +
  459 +/**
  460 + * ForEach helper function
  461 + * @static
  462 + * @param {*} obj
  463 + * @param {function} callback
  464 + */
  465 +showdown.helper.forEach = function forEach(obj, callback) {
  466 + 'use strict';
  467 + if (typeof obj.forEach === 'function') {
  468 + obj.forEach(callback);
  469 + } else {
  470 + for (var i = 0; i < obj.length; i++) {
  471 + callback(obj[i], i, obj);
  472 + }
  473 + }
  474 +};
  475 +
  476 +/**
  477 + * isArray helper function
  478 + * @static
  479 + * @param {*} a
  480 + * @returns {boolean}
  481 + */
  482 +showdown.helper.isArray = function isArray(a) {
  483 + 'use strict';
  484 + return a.constructor === Array;
  485 +};
  486 +
  487 +/**
  488 + * Check if value is undefined
  489 + * @static
  490 + * @param {*} value The value to check.
  491 + * @returns {boolean} Returns `true` if `value` is `undefined`, else `false`.
  492 + */
  493 +showdown.helper.isUndefined = function isUndefined(value) {
  494 + 'use strict';
  495 + return typeof value === 'undefined';
  496 +};
  497 +
  498 +/**
  499 + * Standardidize extension name
  500 + * @static
  501 + * @param {string} s extension name
  502 + * @returns {string}
  503 + */
  504 +showdown.helper.stdExtName = function (s) {
  505 + 'use strict';
  506 + return s.replace(/[_-]||\s/g, '').toLowerCase();
  507 +};
  508 +
  509 +function escapeCharactersCallback(wholeMatch, m1) {
  510 + 'use strict';
  511 + var charCodeToEscape = m1.charCodeAt(0);
  512 + return '~E' + charCodeToEscape + 'E';
  513 +}
  514 +
  515 +/**
  516 + * Callback used to escape characters when passing through String.replace
  517 + * @static
  518 + * @param {string} wholeMatch
  519 + * @param {string} m1
  520 + * @returns {string}
  521 + */
  522 +showdown.helper.escapeCharactersCallback = escapeCharactersCallback;
  523 +
  524 +/**
  525 + * Escape characters in a string
  526 + * @static
  527 + * @param {string} text
  528 + * @param {string} charsToEscape
  529 + * @param {boolean} afterBackslash
  530 + * @returns {XML|string|void|*}
  531 + */
  532 +showdown.helper.escapeCharacters = function escapeCharacters(text, charsToEscape, afterBackslash) {
  533 + 'use strict';
  534 + // First we have to escape the escape characters so that
  535 + // we can build a character class out of them
  536 + var regexString = '([' + charsToEscape.replace(/([\[\]\\])/g, '\\$1') + '])';
  537 +
  538 + if (afterBackslash) {
  539 + regexString = '\\\\' + regexString;
  540 + }
  541 +
  542 + var regex = new RegExp(regexString, 'g');
  543 + text = text.replace(regex, escapeCharactersCallback);
  544 +
  545 + return text;
  546 +};
  547 +
  548 +var rgxFindMatchPos = function (str, left, right, flags) {
  549 + 'use strict';
  550 + var f = flags || '',
  551 + g = f.indexOf('g') > -1,
  552 + x = new RegExp(left + '|' + right, 'g' + f.replace(/g/g, '')),
  553 + l = new RegExp(left, f.replace(/g/g, '')),
  554 + pos = [],
  555 + t, s, m, start, end;
  556 +
  557 + do {
  558 + t = 0;
  559 + while ((m = x.exec(str))) {
  560 + if (l.test(m[0])) {
  561 + if (!(t++)) {
  562 + s = x.lastIndex;
  563 + start = s - m[0].length;
  564 + }
  565 + } else if (t) {
  566 + if (!--t) {
  567 + end = m.index + m[0].length;
  568 + var obj = {
  569 + left: {start: start, end: s},
  570 + match: {start: s, end: m.index},
  571 + right: {start: m.index, end: end},
  572 + wholeMatch: {start: start, end: end}
  573 + };
  574 + pos.push(obj);
  575 + if (!g) {
  576 + return pos;
  577 + }
  578 + }
  579 + }
  580 + }
  581 + } while (t && (x.lastIndex = s));
  582 +
  583 + return pos;
  584 +};
  585 +
  586 +/**
  587 + * matchRecursiveRegExp
  588 + *
  589 + * (c) 2007 Steven Levithan <stevenlevithan.com>
  590 + * MIT License
  591 + *
  592 + * Accepts a string to search, a left and right format delimiter
  593 + * as regex patterns, and optional regex flags. Returns an array
  594 + * of matches, allowing nested instances of left/right delimiters.
  595 + * Use the "g" flag to return all matches, otherwise only the
  596 + * first is returned. Be careful to ensure that the left and
  597 + * right format delimiters produce mutually exclusive matches.
  598 + * Backreferences are not supported within the right delimiter
  599 + * due to how it is internally combined with the left delimiter.
  600 + * When matching strings whose format delimiters are unbalanced
  601 + * to the left or right, the output is intentionally as a
  602 + * conventional regex library with recursion support would
  603 + * produce, e.g. "<<x>" and "<x>>" both produce ["x"] when using
  604 + * "<" and ">" as the delimiters (both strings contain a single,
  605 + * balanced instance of "<x>").
  606 + *
  607 + * examples:
  608 + * matchRecursiveRegExp("test", "\\(", "\\)")
  609 + * returns: []
  610 + * matchRecursiveRegExp("<t<<e>><s>>t<>", "<", ">", "g")
  611 + * returns: ["t<<e>><s>", ""]
  612 + * matchRecursiveRegExp("<div id=\"x\">test</div>", "<div\\b[^>]*>", "</div>", "gi")
  613 + * returns: ["test"]
  614 + */
  615 +showdown.helper.matchRecursiveRegExp = function (str, left, right, flags) {
  616 + 'use strict';
  617 +
  618 + var matchPos = rgxFindMatchPos (str, left, right, flags),
  619 + results = [];
  620 +
  621 + for (var i = 0; i < matchPos.length; ++i) {
  622 + results.push([
  623 + str.slice(matchPos[i].wholeMatch.start, matchPos[i].wholeMatch.end),
  624 + str.slice(matchPos[i].match.start, matchPos[i].match.end),
  625 + str.slice(matchPos[i].left.start, matchPos[i].left.end),
  626 + str.slice(matchPos[i].right.start, matchPos[i].right.end)
  627 + ]);
  628 + }
  629 + return results;
  630 +};
  631 +
  632 +/**
  633 + *
  634 + * @param {string} str
  635 + * @param {string|function} replacement
  636 + * @param {string} left
  637 + * @param {string} right
  638 + * @param {string} flags
  639 + * @returns {string}
  640 + */
  641 +showdown.helper.replaceRecursiveRegExp = function (str, replacement, left, right, flags) {
  642 + 'use strict';
  643 +
  644 + if (!showdown.helper.isFunction(replacement)) {
  645 + var repStr = replacement;
  646 + replacement = function () {
  647 + return repStr;
  648 + };
  649 + }
  650 +
  651 + var matchPos = rgxFindMatchPos(str, left, right, flags),
  652 + finalStr = str,
  653 + lng = matchPos.length;
  654 +
  655 + if (lng > 0) {
  656 + var bits = [];
  657 + if (matchPos[0].wholeMatch.start !== 0) {
  658 + bits.push(str.slice(0, matchPos[0].wholeMatch.start));
  659 + }
  660 + for (var i = 0; i < lng; ++i) {
  661 + bits.push(
  662 + replacement(
  663 + str.slice(matchPos[i].wholeMatch.start, matchPos[i].wholeMatch.end),
  664 + str.slice(matchPos[i].match.start, matchPos[i].match.end),
  665 + str.slice(matchPos[i].left.start, matchPos[i].left.end),
  666 + str.slice(matchPos[i].right.start, matchPos[i].right.end)
  667 + )
  668 + );
  669 + if (i < lng - 1) {
  670 + bits.push(str.slice(matchPos[i].wholeMatch.end, matchPos[i + 1].wholeMatch.start));
  671 + }
  672 + }
  673 + if (matchPos[lng - 1].wholeMatch.end < str.length) {
  674 + bits.push(str.slice(matchPos[lng - 1].wholeMatch.end));
  675 + }
  676 + finalStr = bits.join('');
  677 + }
  678 + return finalStr;
  679 +};
  680 +
  681 +/**
  682 + * POLYFILLS
  683 + */
  684 +if (showdown.helper.isUndefined(console)) {
  685 + console = {
  686 + warn: function (msg) {
  687 + 'use strict';
  688 + alert(msg);
  689 + },
  690 + log: function (msg) {
  691 + 'use strict';
  692 + alert(msg);
  693 + },
  694 + error: function (msg) {
  695 + 'use strict';
  696 + throw msg;
  697 + }
  698 + };
  699 +}
  700 +
  701 +/**
  702 + * Created by Estevao on 31-05-2015.
  703 + */
  704 +
  705 +/**
  706 + * Showdown Converter class
  707 + * @class
  708 + * @param {object} [converterOptions]
  709 + * @returns {Converter}
  710 + */
  711 +showdown.Converter = function (converterOptions) {
  712 + 'use strict';
  713 +
  714 + var
  715 + /**
  716 + * Options used by this converter
  717 + * @private
  718 + * @type {{}}
  719 + */
  720 + options = {},
  721 +
  722 + /**
  723 + * Language extensions used by this converter
  724 + * @private
  725 + * @type {Array}
  726 + */
  727 + langExtensions = [],
  728 +
  729 + /**
  730 + * Output modifiers extensions used by this converter
  731 + * @private
  732 + * @type {Array}
  733 + */
  734 + outputModifiers = [],
  735 +
  736 + /**
  737 + * Event listeners
  738 + * @private
  739 + * @type {{}}
  740 + */
  741 + listeners = {};
  742 +
  743 + _constructor();
  744 +
  745 + /**
  746 + * Converter constructor
  747 + * @private
  748 + */
  749 + function _constructor() {
  750 + converterOptions = converterOptions || {};
  751 +
  752 + for (var gOpt in globalOptions) {
  753 + if (globalOptions.hasOwnProperty(gOpt)) {
  754 + options[gOpt] = globalOptions[gOpt];
  755 + }
  756 + }
  757 +
  758 + // Merge options
  759 + if (typeof converterOptions === 'object') {
  760 + for (var opt in converterOptions) {
  761 + if (converterOptions.hasOwnProperty(opt)) {
  762 + options[opt] = converterOptions[opt];
  763 + }
  764 + }
  765 + } else {
  766 + throw Error('Converter expects the passed parameter to be an object, but ' + typeof converterOptions +
  767 + ' was passed instead.');
  768 + }
  769 +
  770 + if (options.extensions) {
  771 + showdown.helper.forEach(options.extensions, _parseExtension);
  772 + }
  773 + }
  774 +
  775 + /**
  776 + * Parse extension
  777 + * @param {*} ext
  778 + * @param {string} [name='']
  779 + * @private
  780 + */
  781 + function _parseExtension(ext, name) {
  782 +
  783 + name = name || null;
  784 + // If it's a string, the extension was previously loaded
  785 + if (showdown.helper.isString(ext)) {
  786 + ext = showdown.helper.stdExtName(ext);
  787 + name = ext;
  788 +
  789 + // LEGACY_SUPPORT CODE
  790 + if (showdown.extensions[ext]) {
  791 + console.warn('DEPRECATION WARNING: ' + ext + ' is an old extension that uses a deprecated loading method.' +
  792 + 'Please inform the developer that the extension should be updated!');
  793 + legacyExtensionLoading(showdown.extensions[ext], ext);
  794 + return;
  795 + // END LEGACY SUPPORT CODE
  796 +
  797 + } else if (!showdown.helper.isUndefined(extensions[ext])) {
  798 + ext = extensions[ext];
  799 +
  800 + } else {
  801 + throw Error('Extension "' + ext + '" could not be loaded. It was either not found or is not a valid extension.');
  802 + }
  803 + }
  804 +
  805 + if (typeof ext === 'function') {
  806 + ext = ext();
  807 + }
  808 +
  809 + if (!showdown.helper.isArray(ext)) {
  810 + ext = [ext];
  811 + }
  812 +
  813 + var validExt = validate(ext, name);
  814 + if (!validExt.valid) {
  815 + throw Error(validExt.error);
  816 + }
  817 +
  818 + for (var i = 0; i < ext.length; ++i) {
  819 + switch (ext[i].type) {
  820 +
  821 + case 'lang':
  822 + langExtensions.push(ext[i]);
  823 + break;
  824 +
  825 + case 'output':
  826 + outputModifiers.push(ext[i]);
  827 + break;
  828 + }
  829 + if (ext[i].hasOwnProperty(listeners)) {
  830 + for (var ln in ext[i].listeners) {
  831 + if (ext[i].listeners.hasOwnProperty(ln)) {
  832 + listen(ln, ext[i].listeners[ln]);
  833 + }
  834 + }
  835 + }
  836 + }
  837 +
  838 + }
  839 +
  840 + /**
  841 + * LEGACY_SUPPORT
  842 + * @param {*} ext
  843 + * @param {string} name
  844 + */
  845 + function legacyExtensionLoading(ext, name) {
  846 + if (typeof ext === 'function') {
  847 + ext = ext(new showdown.Converter());
  848 + }
  849 + if (!showdown.helper.isArray(ext)) {
  850 + ext = [ext];
  851 + }
  852 + var valid = validate(ext, name);
  853 +
  854 + if (!valid.valid) {
  855 + throw Error(valid.error);
  856 + }
  857 +
  858 + for (var i = 0; i < ext.length; ++i) {
  859 + switch (ext[i].type) {
  860 + case 'lang':
  861 + langExtensions.push(ext[i]);
  862 + break;
  863 + case 'output':
  864 + outputModifiers.push(ext[i]);
  865 + break;
  866 + default:// should never reach here
  867 + throw Error('Extension loader error: Type unrecognized!!!');
  868 + }
  869 + }
  870 + }
  871 +
  872 + /**
  873 + * Listen to an event
  874 + * @param {string} name
  875 + * @param {function} callback
  876 + */
  877 + function listen(name, callback) {
  878 + if (!showdown.helper.isString(name)) {
  879 + throw Error('Invalid argument in converter.listen() method: name must be a string, but ' + typeof name + ' given');
  880 + }
  881 +
  882 + if (typeof callback !== 'function') {
  883 + throw Error('Invalid argument in converter.listen() method: callback must be a function, but ' + typeof callback + ' given');
  884 + }
  885 +
  886 + if (!listeners.hasOwnProperty(name)) {
  887 + listeners[name] = [];
  888 + }
  889 + listeners[name].push(callback);
  890 + }
  891 +
  892 + function rTrimInputText(text) {
  893 + var rsp = text.match(/^\s*/)[0].length,
  894 + rgx = new RegExp('^\\s{0,' + rsp + '}', 'gm');
  895 + return text.replace(rgx, '');
  896 + }
  897 +
  898 + /**
  899 + * Dispatch an event
  900 + * @private
  901 + * @param {string} evtName Event name
  902 + * @param {string} text Text
  903 + * @param {{}} options Converter Options
  904 + * @param {{}} globals
  905 + * @returns {string}
  906 + */
  907 + this._dispatch = function dispatch (evtName, text, options, globals) {
  908 + if (listeners.hasOwnProperty(evtName)) {
  909 + for (var ei = 0; ei < listeners[evtName].length; ++ei) {
  910 + var nText = listeners[evtName][ei](evtName, text, this, options, globals);
  911 + if (nText && typeof nText !== 'undefined') {
  912 + text = nText;
  913 + }
  914 + }
  915 + }
  916 + return text;
  917 + };
  918 +
  919 + /**
  920 + * Listen to an event
  921 + * @param {string} name
  922 + * @param {function} callback
  923 + * @returns {showdown.Converter}
  924 + */
  925 + this.listen = function (name, callback) {
  926 + listen(name, callback);
  927 + return this;
  928 + };
  929 +
  930 + /**
  931 + * Converts a markdown string into HTML
  932 + * @param {string} text
  933 + * @returns {*}
  934 + */
  935 + this.makeHtml = function (text) {
  936 + //check if text is not falsy
  937 + if (!text) {
  938 + return text;
  939 + }
  940 +
  941 + var globals = {
  942 + gHtmlBlocks: [],
  943 + gHtmlMdBlocks: [],
  944 + gHtmlSpans: [],
  945 + gUrls: {},
  946 + gTitles: {},
  947 + gDimensions: {},
  948 + gListLevel: 0,
  949 + hashLinkCounts: {},
  950 + langExtensions: langExtensions,
  951 + outputModifiers: outputModifiers,
  952 + converter: this,
  953 + ghCodeBlocks: []
  954 + };
  955 +
  956 + // attacklab: Replace ~ with ~T
  957 + // This lets us use tilde as an escape char to avoid md5 hashes
  958 + // The choice of character is arbitrary; anything that isn't
  959 + // magic in Markdown will work.
  960 + text = text.replace(/~/g, '~T');
  961 +
  962 + // attacklab: Replace $ with ~D
  963 + // RegExp interprets $ as a special character
  964 + // when it's in a replacement string
  965 + text = text.replace(/\$/g, '~D');
  966 +
  967 + // Standardize line endings
  968 + text = text.replace(/\r\n/g, '\n'); // DOS to Unix
  969 + text = text.replace(/\r/g, '\n'); // Mac to Unix
  970 +
  971 + if (options.smartIndentationFix) {
  972 + text = rTrimInputText(text);
  973 + }
  974 +
  975 + // Make sure text begins and ends with a couple of newlines:
  976 + //text = '\n\n' + text + '\n\n';
  977 + text = text;
  978 + // detab
  979 + text = showdown.subParser('detab')(text, options, globals);
  980 +
  981 + // stripBlankLines
  982 + text = showdown.subParser('stripBlankLines')(text, options, globals);
  983 +
  984 + //run languageExtensions
  985 + showdown.helper.forEach(langExtensions, function (ext) {
  986 + text = showdown.subParser('runExtension')(ext, text, options, globals);
  987 + });
  988 +
  989 + // run the sub parsers
  990 + text = showdown.subParser('hashPreCodeTags')(text, options, globals);
  991 + text = showdown.subParser('githubCodeBlocks')(text, options, globals);
  992 + text = showdown.subParser('hashHTMLBlocks')(text, options, globals);
  993 + text = showdown.subParser('hashHTMLSpans')(text, options, globals);
  994 + text = showdown.subParser('stripLinkDefinitions')(text, options, globals);
  995 + text = showdown.subParser('blockGamut')(text, options, globals);
  996 + text = showdown.subParser('unhashHTMLSpans')(text, options, globals);
  997 + text = showdown.subParser('unescapeSpecialChars')(text, options, globals);
  998 +
  999 + // attacklab: Restore dollar signs
  1000 + text = text.replace(/~D/g, '$$');
  1001 +
  1002 + // attacklab: Restore tildes
  1003 + text = text.replace(/~T/g, '~');
  1004 +
  1005 + // Run output modifiers
  1006 + showdown.helper.forEach(outputModifiers, function (ext) {
  1007 + text = showdown.subParser('runExtension')(ext, text, options, globals);
  1008 + });
  1009 + return text;
  1010 + };
  1011 +
  1012 + /**
  1013 + * Set an option of this Converter instance
  1014 + * @param {string} key
  1015 + * @param {*} value
  1016 + */
  1017 + this.setOption = function (key, value) {
  1018 + options[key] = value;
  1019 + };
  1020 +
  1021 + /**
  1022 + * Get the option of this Converter instance
  1023 + * @param {string} key
  1024 + * @returns {*}
  1025 + */
  1026 + this.getOption = function (key) {
  1027 + return options[key];
  1028 + };
  1029 +
  1030 + /**
  1031 + * Get the options of this Converter instance
  1032 + * @returns {{}}
  1033 + */
  1034 + this.getOptions = function () {
  1035 + return options;
  1036 + };
  1037 +
  1038 + /**
  1039 + * Add extension to THIS converter
  1040 + * @param {{}} extension
  1041 + * @param {string} [name=null]
  1042 + */
  1043 + this.addExtension = function (extension, name) {
  1044 + name = name || null;
  1045 + _parseExtension(extension, name);
  1046 + };
  1047 +
  1048 + /**
  1049 + * Use a global registered extension with THIS converter
  1050 + * @param {string} extensionName Name of the previously registered extension
  1051 + */
  1052 + this.useExtension = function (extensionName) {
  1053 + _parseExtension(extensionName);
  1054 + };
  1055 +
  1056 + /**
  1057 + * Set the flavor THIS converter should use
  1058 + * @param {string} name
  1059 + */
  1060 + this.setFlavor = function (name) {
  1061 + if (flavor.hasOwnProperty(name)) {
  1062 + var preset = flavor[name];
  1063 + for (var option in preset) {
  1064 + if (preset.hasOwnProperty(option)) {
  1065 + options[option] = preset[option];
  1066 + }
  1067 + }
  1068 + }
  1069 + };
  1070 +
  1071 + /**
  1072 + * Remove an extension from THIS converter.
  1073 + * Note: This is a costly operation. It's better to initialize a new converter
  1074 + * and specify the extensions you wish to use
  1075 + * @param {Array} extension
  1076 + */
  1077 + this.removeExtension = function (extension) {
  1078 + if (!showdown.helper.isArray(extension)) {
  1079 + extension = [extension];
  1080 + }
  1081 + for (var a = 0; a < extension.length; ++a) {
  1082 + var ext = extension[a];
  1083 + for (var i = 0; i < langExtensions.length; ++i) {
  1084 + if (langExtensions[i] === ext) {
  1085 + langExtensions[i].splice(i, 1);
  1086 + }
  1087 + }
  1088 + for (var ii = 0; ii < outputModifiers.length; ++i) {
  1089 + if (outputModifiers[ii] === ext) {
  1090 + outputModifiers[ii].splice(i, 1);
  1091 + }
  1092 + }
  1093 + }
  1094 + };
  1095 +
  1096 + /**
  1097 + * Get all extension of THIS converter
  1098 + * @returns {{language: Array, output: Array}}
  1099 + */
  1100 + this.getAllExtensions = function () {
  1101 + return {
  1102 + language: langExtensions,
  1103 + output: outputModifiers
  1104 + };
  1105 + };
  1106 +};
  1107 +
  1108 +/**
  1109 + * Turn Markdown link shortcuts into XHTML <a> tags.
  1110 + */
  1111 +showdown.subParser('anchors', function (text, options, globals) {
  1112 + 'use strict';
  1113 +
  1114 + text = globals.converter._dispatch('anchors.before', text, options, globals);
  1115 +
  1116 + var writeAnchorTag = function (wholeMatch, m1, m2, m3, m4, m5, m6, m7) {
  1117 + if (showdown.helper.isUndefined(m7)) {
  1118 + m7 = '';
  1119 + }
  1120 + wholeMatch = m1;
  1121 + var linkText = m2,
  1122 + linkId = m3.toLowerCase(),
  1123 + url = m4,
  1124 + title = m7;
  1125 +
  1126 + if (!url) {
  1127 + if (!linkId) {
  1128 + // lower-case and turn embedded newlines into spaces
  1129 + linkId = linkText.toLowerCase().replace(/ ?\n/g, ' ');
  1130 + }
  1131 + url = '#' + linkId;
  1132 +
  1133 + if (!showdown.helper.isUndefined(globals.gUrls[linkId])) {
  1134 + url = globals.gUrls[linkId];
  1135 + if (!showdown.helper.isUndefined(globals.gTitles[linkId])) {
  1136 + title = globals.gTitles[linkId];
  1137 + }
  1138 + } else {
  1139 + if (wholeMatch.search(/\(\s*\)$/m) > -1) {
  1140 + // Special case for explicit empty url
  1141 + url = '';
  1142 + } else {
  1143 + return wholeMatch;
  1144 + }
  1145 + }
  1146 + }
  1147 +
  1148 + url = showdown.helper.escapeCharacters(url, '*_', false);
  1149 + var result = '<a href="' + url + '"';
  1150 +
  1151 + if (title !== '' && title !== null) {
  1152 + title = title.replace(/"/g, '&quot;');
  1153 + title = showdown.helper.escapeCharacters(title, '*_', false);
  1154 + result += ' title="' + title + '"';
  1155 + }
  1156 +
  1157 + result += '>' + linkText + '</a>';
  1158 +
  1159 + return result;
  1160 + };
  1161 +
  1162 + // First, handle reference-style links: [link text] [id]
  1163 + /*
  1164 + text = text.replace(/
  1165 + ( // wrap whole match in $1
  1166 + \[
  1167 + (
  1168 + (?:
  1169 + \[[^\]]*\] // allow brackets nested one level
  1170 + |
  1171 + [^\[] // or anything else
  1172 + )*
  1173 + )
  1174 + \]
  1175 +
  1176 + [ ]? // one optional space
  1177 + (?:\n[ ]*)? // one optional newline followed by spaces
  1178 +
  1179 + \[
  1180 + (.*?) // id = $3
  1181 + \]
  1182 + )()()()() // pad remaining backreferences
  1183 + /g,_DoAnchors_callback);
  1184 + */
  1185 + text = text.replace(/(\[((?:\[[^\]]*]|[^\[\]])*)][ ]?(?:\n[ ]*)?\[(.*?)])()()()()/g, writeAnchorTag);
  1186 +
  1187 + //
  1188 + // Next, inline-style links: [link text](url "optional title")
  1189 + //
  1190 +
  1191 + /*
  1192 + text = text.replace(/
  1193 + ( // wrap whole match in $1
  1194 + \[
  1195 + (
  1196 + (?:
  1197 + \[[^\]]*\] // allow brackets nested one level
  1198 + |
  1199 + [^\[\]] // or anything else
  1200 + )
  1201 + )
  1202 + \]
  1203 + \( // literal paren
  1204 + [ \t]*
  1205 + () // no id, so leave $3 empty
  1206 + <?(.*?)>? // href = $4
  1207 + [ \t]*
  1208 + ( // $5
  1209 + (['"]) // quote char = $6
  1210 + (.*?) // Title = $7
  1211 + \6 // matching quote
  1212 + [ \t]* // ignore any spaces/tabs between closing quote and )
  1213 + )? // title is optional
  1214 + \)
  1215 + )
  1216 + /g,writeAnchorTag);
  1217 + */
  1218 + text = text.replace(/(\[((?:\[[^\]]*]|[^\[\]])*)]\([ \t]*()<?(.*?(?:\(.*?\).*?)?)>?[ \t]*((['"])(.*?)\6[ \t]*)?\))/g,
  1219 + writeAnchorTag);
  1220 +
  1221 + //
  1222 + // Last, handle reference-style shortcuts: [link text]
  1223 + // These must come last in case you've also got [link test][1]
  1224 + // or [link test](/foo)
  1225 + //
  1226 +
  1227 + /*
  1228 + text = text.replace(/
  1229 + ( // wrap whole match in $1
  1230 + \[
  1231 + ([^\[\]]+) // link text = $2; can't contain '[' or ']'
  1232 + \]
  1233 + )()()()()() // pad rest of backreferences
  1234 + /g, writeAnchorTag);
  1235 + */
  1236 + text = text.replace(/(\[([^\[\]]+)])()()()()()/g, writeAnchorTag);
  1237 +
  1238 + text = globals.converter._dispatch('anchors.after', text, options, globals);
  1239 + return text;
  1240 +});
  1241 +
  1242 +showdown.subParser('autoLinks', function (text, options, globals) {
  1243 + 'use strict';
  1244 +
  1245 + text = globals.converter._dispatch('autoLinks.before', text, options, globals);
  1246 +
  1247 + var simpleURLRegex = /\b(((https?|ftp|dict):\/\/|www\.)[^'">\s]+\.[^'">\s]+)(?=\s|$)(?!["<>])/gi,
  1248 + delimUrlRegex = /<(((https?|ftp|dict):\/\/|www\.)[^'">\s]+)>/gi,
  1249 + simpleMailRegex = /(?:^|[ \n\t])([A-Za-z0-9!#$%&'*+-/=?^_`\{|}~\.]+@[-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+)(?:$|[ \n\t])/gi,
  1250 + delimMailRegex = /<(?:mailto:)?([-.\w]+@[-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+)>/gi;
  1251 +
  1252 + text = text.replace(delimUrlRegex, replaceLink);
  1253 + text = text.replace(delimMailRegex, replaceMail);
  1254 + // simpleURLRegex = /\b(((https?|ftp|dict):\/\/|www\.)[-.+~:?#@!$&'()*,;=[\]\w]+)\b/gi,
  1255 + // Email addresses: <address@domain.foo>
  1256 +
  1257 + if (options.simplifiedAutoLink) {
  1258 + text = text.replace(simpleURLRegex, replaceLink);
  1259 + text = text.replace(simpleMailRegex, replaceMail);
  1260 + }
  1261 +
  1262 + function replaceLink(wm, link) {
  1263 + var lnkTxt = link;
  1264 + if (/^www\./i.test(link)) {
  1265 + link = link.replace(/^www\./i, 'http://www.');
  1266 + }
  1267 + return '<a href="' + link + '">' + lnkTxt + '</a>';
  1268 + }
  1269 +
  1270 + function replaceMail(wholeMatch, m1) {
  1271 + var unescapedStr = showdown.subParser('unescapeSpecialChars')(m1);
  1272 + return showdown.subParser('encodeEmailAddress')(unescapedStr);
  1273 + }
  1274 +
  1275 + text = globals.converter._dispatch('autoLinks.after', text, options, globals);
  1276 +
  1277 + return text;
  1278 +});
  1279 +
  1280 +/**
  1281 + * These are all the transformations that form block-level
  1282 + * tags like paragraphs, headers, and list items.
  1283 + */
  1284 +showdown.subParser('blockGamut', function (text, options, globals) {
  1285 + 'use strict';
  1286 +
  1287 + text = globals.converter._dispatch('blockGamut.before', text, options, globals);
  1288 +
  1289 + // we parse blockquotes first so that we can have headings and hrs
  1290 + // inside blockquotes
  1291 + text = showdown.subParser('blockQuotes')(text, options, globals);
  1292 + text = showdown.subParser('headers')(text, options, globals);
  1293 +
  1294 + // Do Horizontal Rules:
  1295 + var key = showdown.subParser('hashBlock')('<hr />', options, globals);
  1296 + text = text.replace(/^[ ]{0,2}([ ]?\*[ ]?){3,}[ \t]*$/gm, key);
  1297 + text = text.replace(/^[ ]{0,2}([ ]?\-[ ]?){3,}[ \t]*$/gm, key);
  1298 + text = text.replace(/^[ ]{0,2}([ ]?_[ ]?){3,}[ \t]*$/gm, key);
  1299 +
  1300 + text = showdown.subParser('lists')(text, options, globals);
  1301 + text = showdown.subParser('codeBlocks')(text, options, globals);
  1302 + text = showdown.subParser('tables')(text, options, globals);
  1303 +
  1304 + // We already ran _HashHTMLBlocks() before, in Markdown(), but that
  1305 + // was to escape raw HTML in the original Markdown source. This time,
  1306 + // we're escaping the markup we've just created, so that we don't wrap
  1307 + // <p> tags around block-level tags.
  1308 + text = showdown.subParser('hashHTMLBlocks')(text, options, globals);
  1309 + text = showdown.subParser('paragraphs')(text, options, globals);
  1310 +
  1311 + text = globals.converter._dispatch('blockGamut.after', text, options, globals);
  1312 +
  1313 + return text;
  1314 +});
  1315 +
  1316 +showdown.subParser('blockQuotes', function (text, options, globals) {
  1317 + 'use strict';
  1318 +
  1319 + text = globals.converter._dispatch('blockQuotes.before', text, options, globals);
  1320 + /*
  1321 + text = text.replace(/
  1322 + ( // Wrap whole match in $1
  1323 + (
  1324 + ^[ \t]*>[ \t]? // '>' at the start of a line
  1325 + .+\n // rest of the first line
  1326 + (.+\n)* // subsequent consecutive lines
  1327 + \n* // blanks
  1328 + )+
  1329 + )
  1330 + /gm, function(){...});
  1331 + */
  1332 +
  1333 + text = text.replace(/((^[ \t]{0,3}>[ \t]?.+\n(.+\n)*\n*)+)/gm, function (wholeMatch, m1) {
  1334 + var bq = m1;
  1335 +
  1336 + // attacklab: hack around Konqueror 3.5.4 bug:
  1337 + // "----------bug".replace(/^-/g,"") == "bug"
  1338 + bq = bq.replace(/^[ \t]*>[ \t]?/gm, '~0'); // trim one level of quoting
  1339 +
  1340 + // attacklab: clean up hack
  1341 + bq = bq.replace(/~0/g, '');
  1342 +
  1343 + bq = bq.replace(/^[ \t]+$/gm, ''); // trim whitespace-only lines
  1344 + bq = showdown.subParser('githubCodeBlocks')(bq, options, globals);
  1345 + bq = showdown.subParser('blockGamut')(bq, options, globals); // recurse
  1346 +
  1347 + bq = bq.replace(/(^|\n)/g, '$1 ');
  1348 + // These leading spaces screw with <pre> content, so we need to fix that:
  1349 + bq = bq.replace(/(\s*<pre>[^\r]+?<\/pre>)/gm, function (wholeMatch, m1) {
  1350 + var pre = m1;
  1351 + // attacklab: hack around Konqueror 3.5.4 bug:
  1352 + pre = pre.replace(/^ /mg, '~0');
  1353 + pre = pre.replace(/~0/g, '');
  1354 + return pre;
  1355 + });
  1356 +
  1357 + return showdown.subParser('hashBlock')('<blockquote>\n' + bq + '\n</blockquote>', options, globals);
  1358 + });
  1359 +
  1360 + text = globals.converter._dispatch('blockQuotes.after', text, options, globals);
  1361 + return text;
  1362 +});
  1363 +
  1364 +/**
  1365 + * Process Markdown `<pre><code>` blocks.
  1366 + */
  1367 +showdown.subParser('codeBlocks', function (text, options, globals) {
  1368 + 'use strict';
  1369 +
  1370 + text = globals.converter._dispatch('codeBlocks.before', text, options, globals);
  1371 + /*
  1372 + text = text.replace(text,
  1373 + /(?:\n\n|^)
  1374 + ( // $1 = the code block -- one or more lines, starting with a space/tab
  1375 + (?:
  1376 + (?:[ ]{4}|\t) // Lines must start with a tab or a tab-width of spaces - attacklab: g_tab_width
  1377 + .*\n+
  1378 + )+
  1379 + )
  1380 + (\n*[ ]{0,3}[^ \t\n]|(?=~0)) // attacklab: g_tab_width
  1381 + /g,function(){...});
  1382 + */
  1383 +
  1384 + // attacklab: sentinel workarounds for lack of \A and \Z, safari\khtml bug
  1385 + text += '~0';
  1386 +
  1387 + var pattern = /(?:\n\n|^)((?:(?:[ ]{4}|\t).*\n+)+)(\n*[ ]{0,3}[^ \t\n]|(?=~0))/g;
  1388 + text = text.replace(pattern, function (wholeMatch, m1, m2) {
  1389 + var codeblock = m1,
  1390 + nextChar = m2,
  1391 + end = '\n';
  1392 +
  1393 + codeblock = showdown.subParser('outdent')(codeblock);
  1394 + codeblock = showdown.subParser('encodeCode')(codeblock);
  1395 + codeblock = showdown.subParser('detab')(codeblock);
  1396 + codeblock = codeblock.replace(/^\n+/g, ''); // trim leading newlines
  1397 + codeblock = codeblock.replace(/\n+$/g, ''); // trim trailing newlines
  1398 +
  1399 + if (options.omitExtraWLInCodeBlocks) {
  1400 + end = '';
  1401 + }
  1402 +
  1403 + codeblock = '<pre><code>' + codeblock + end + '</code></pre>';
  1404 +
  1405 + return showdown.subParser('hashBlock')(codeblock, options, globals) + nextChar;
  1406 + });
  1407 +
  1408 + // attacklab: strip sentinel
  1409 + text = text.replace(/~0/, '');
  1410 +
  1411 + text = globals.converter._dispatch('codeBlocks.after', text, options, globals);
  1412 + return text;
  1413 +});
  1414 +
  1415 +/**
  1416 + *
  1417 + * * Backtick quotes are used for <code></code> spans.
  1418 + *
  1419 + * * You can use multiple backticks as the delimiters if you want to
  1420 + * include literal backticks in the code span. So, this input:
  1421 + *
  1422 + * Just type ``foo `bar` baz`` at the prompt.
  1423 + *
  1424 + * Will translate to:
  1425 + *
  1426 + * <p>Just type <code>foo `bar` baz</code> at the prompt.</p>
  1427 + *
  1428 + * There's no arbitrary limit to the number of backticks you
  1429 + * can use as delimters. If you need three consecutive backticks
  1430 + * in your code, use four for delimiters, etc.
  1431 + *
  1432 + * * You can use spaces to get literal backticks at the edges:
  1433 + *
  1434 + * ... type `` `bar` `` ...
  1435 + *
  1436 + * Turns to:
  1437 + *
  1438 + * ... type <code>`bar`</code> ...
  1439 + */
  1440 +showdown.subParser('codeSpans', function (text, options, globals) {
  1441 + 'use strict';
  1442 +
  1443 + text = globals.converter._dispatch('codeSpans.before', text, options, globals);
  1444 +
  1445 + /*
  1446 + text = text.replace(/
  1447 + (^|[^\\]) // Character before opening ` can't be a backslash
  1448 + (`+) // $2 = Opening run of `
  1449 + ( // $3 = The code block
  1450 + [^\r]*?
  1451 + [^`] // attacklab: work around lack of lookbehind
  1452 + )
  1453 + \2 // Matching closer
  1454 + (?!`)
  1455 + /gm, function(){...});
  1456 + */
  1457 +
  1458 + if (typeof(text) === 'undefined') {
  1459 + text = '';
  1460 + }
  1461 + text = text.replace(/(^|[^\\])(`+)([^\r]*?[^`])\2(?!`)/gm,
  1462 + function (wholeMatch, m1, m2, m3) {
  1463 + var c = m3;
  1464 + c = c.replace(/^([ \t]*)/g, ''); // leading whitespace
  1465 + c = c.replace(/[ \t]*$/g, ''); // trailing whitespace
  1466 + c = showdown.subParser('encodeCode')(c);
  1467 + return m1 + '<code>' + c + '</code>';
  1468 + }
  1469 + );
  1470 +
  1471 + text = globals.converter._dispatch('codeSpans.after', text, options, globals);
  1472 + return text;
  1473 +});
  1474 +
  1475 +/**
  1476 + * Convert all tabs to spaces
  1477 + */
  1478 +showdown.subParser('detab', function (text) {
  1479 + 'use strict';
  1480 +
  1481 + // expand first n-1 tabs
  1482 + text = text.replace(/\t(?=\t)/g, ' '); // g_tab_width
  1483 +
  1484 + // replace the nth with two sentinels
  1485 + text = text.replace(/\t/g, '~A~B');
  1486 +
  1487 + // use the sentinel to anchor our regex so it doesn't explode
  1488 + text = text.replace(/~B(.+?)~A/g, function (wholeMatch, m1) {
  1489 + var leadingText = m1,
  1490 + numSpaces = 4 - leadingText.length % 4; // g_tab_width
  1491 +
  1492 + // there *must* be a better way to do this:
  1493 + for (var i = 0; i < numSpaces; i++) {
  1494 + leadingText += ' ';
  1495 + }
  1496 +
  1497 + return leadingText;
  1498 + });
  1499 +
  1500 + // clean up sentinels
  1501 + text = text.replace(/~A/g, ' '); // g_tab_width
  1502 + text = text.replace(/~B/g, '');
  1503 +
  1504 + return text;
  1505 +
  1506 +});
  1507 +
  1508 +/**
  1509 + * Smart processing for ampersands and angle brackets that need to be encoded.
  1510 + */
  1511 +showdown.subParser('encodeAmpsAndAngles', function (text) {
  1512 + 'use strict';
  1513 + // Ampersand-encoding based entirely on Nat Irons's Amputator MT plugin:
  1514 + // http://bumppo.net/projects/amputator/
  1515 + text = text.replace(/&(?!#?[xX]?(?:[0-9a-fA-F]+|\w+);)/g, '&amp;');
  1516 +
  1517 + // Encode naked <'s
  1518 + text = text.replace(/<(?![a-z\/?\$!])/gi, '&lt;');
  1519 +
  1520 + return text;
  1521 +});
  1522 +
  1523 +/**
  1524 + * Returns the string, with after processing the following backslash escape sequences.
  1525 + *
  1526 + * attacklab: The polite way to do this is with the new escapeCharacters() function:
  1527 + *
  1528 + * text = escapeCharacters(text,"\\",true);
  1529 + * text = escapeCharacters(text,"`*_{}[]()>#+-.!",true);
  1530 + *
  1531 + * ...but we're sidestepping its use of the (slow) RegExp constructor
  1532 + * as an optimization for Firefox. This function gets called a LOT.
  1533 + */
  1534 +showdown.subParser('encodeBackslashEscapes', function (text) {
  1535 + 'use strict';
  1536 + text = text.replace(/\\(\\)/g, showdown.helper.escapeCharactersCallback);
  1537 + text = text.replace(/\\([`*_{}\[\]()>#+-.!])/g, showdown.helper.escapeCharactersCallback);
  1538 + return text;
  1539 +});
  1540 +
  1541 +/**
  1542 + * Encode/escape certain characters inside Markdown code runs.
  1543 + * The point is that in code, these characters are literals,
  1544 + * and lose their special Markdown meanings.
  1545 + */
  1546 +showdown.subParser('encodeCode', function (text) {
  1547 + 'use strict';
  1548 +
  1549 + // Encode all ampersands; HTML entities are not
  1550 + // entities within a Markdown code span.
  1551 + text = text.replace(/&/g, '&amp;');
  1552 +
  1553 + // Do the angle bracket song and dance:
  1554 + text = text.replace(/</g, '&lt;');
  1555 + text = text.replace(/>/g, '&gt;');
  1556 +
  1557 + // Now, escape characters that are magic in Markdown:
  1558 + text = showdown.helper.escapeCharacters(text, '*_{}[]\\', false);
  1559 +
  1560 + // jj the line above breaks this:
  1561 + //---
  1562 + //* Item
  1563 + // 1. Subitem
  1564 + // special char: *
  1565 + // ---
  1566 +
  1567 + return text;
  1568 +});
  1569 +
  1570 +/**
  1571 + * Input: an email address, e.g. "foo@example.com"
  1572 + *
  1573 + * Output: the email address as a mailto link, with each character
  1574 + * of the address encoded as either a decimal or hex entity, in
  1575 + * the hopes of foiling most address harvesting spam bots. E.g.:
  1576 + *
  1577 + * <a href="&#x6D;&#97;&#105;&#108;&#x74;&#111;:&#102;&#111;&#111;&#64;&#101;
  1578 + * x&#x61;&#109;&#x70;&#108;&#x65;&#x2E;&#99;&#111;&#109;">&#102;&#111;&#111;
  1579 + * &#64;&#101;x&#x61;&#109;&#x70;&#108;&#x65;&#x2E;&#99;&#111;&#109;</a>
  1580 + *
  1581 + * Based on a filter by Matthew Wickline, posted to the BBEdit-Talk
  1582 + * mailing list: <http://tinyurl.com/yu7ue>
  1583 + *
  1584 + */
  1585 +showdown.subParser('encodeEmailAddress', function (addr) {
  1586 + 'use strict';
  1587 +
  1588 + var encode = [
  1589 + function (ch) {
  1590 + return '&#' + ch.charCodeAt(0) + ';';
  1591 + },
  1592 + function (ch) {
  1593 + return '&#x' + ch.charCodeAt(0).toString(16) + ';';
  1594 + },
  1595 + function (ch) {
  1596 + return ch;
  1597 + }
  1598 + ];
  1599 +
  1600 + addr = 'mailto:' + addr;
  1601 +
  1602 + addr = addr.replace(/./g, function (ch) {
  1603 + if (ch === '@') {
  1604 + // this *must* be encoded. I insist.
  1605 + ch = encode[Math.floor(Math.random() * 2)](ch);
  1606 + } else if (ch !== ':') {
  1607 + // leave ':' alone (to spot mailto: later)
  1608 + var r = Math.random();
  1609 + // roughly 10% raw, 45% hex, 45% dec
  1610 + ch = (
  1611 + r > 0.9 ? encode[2](ch) : r > 0.45 ? encode[1](ch) : encode[0](ch)
  1612 + );
  1613 + }
  1614 + return ch;
  1615 + });
  1616 +
  1617 + addr = '<a href="' + addr + '">' + addr + '</a>';
  1618 + addr = addr.replace(/">.+:/g, '">'); // strip the mailto: from the visible part
  1619 +
  1620 + return addr;
  1621 +});
  1622 +
  1623 +/**
  1624 + * Within tags -- meaning between < and > -- encode [\ ` * _] so they
  1625 + * don't conflict with their use in Markdown for code, italics and strong.
  1626 + */
  1627 +showdown.subParser('escapeSpecialCharsWithinTagAttributes', function (text) {
  1628 + 'use strict';
  1629 +
  1630 + // Build a regex to find HTML tags and comments. See Friedl's
  1631 + // "Mastering Regular Expressions", 2nd Ed., pp. 200-201.
  1632 + var regex = /(<[a-z\/!$]("[^"]*"|'[^']*'|[^'">])*>|<!(--.*?--\s*)+>)/gi;
  1633 +
  1634 + text = text.replace(regex, function (wholeMatch) {
  1635 + var tag = wholeMatch.replace(/(.)<\/?code>(?=.)/g, '$1`');
  1636 + tag = showdown.helper.escapeCharacters(tag, '\\`*_', false);
  1637 + return tag;
  1638 + });
  1639 +
  1640 + return text;
  1641 +});
  1642 +
  1643 +/**
  1644 + * Handle github codeblocks prior to running HashHTML so that
  1645 + * HTML contained within the codeblock gets escaped properly
  1646 + * Example:
  1647 + * ```ruby
  1648 + * def hello_world(x)
  1649 + * puts "Hello, #{x}"
  1650 + * end
  1651 + * ```
  1652 + */
  1653 +showdown.subParser('githubCodeBlocks', function (text, options, globals) {
  1654 + 'use strict';
  1655 +
  1656 + // early exit if option is not enabled
  1657 + if (!options.ghCodeBlocks) {
  1658 + return text;
  1659 + }
  1660 +
  1661 + text = globals.converter._dispatch('githubCodeBlocks.before', text, options, globals);
  1662 +
  1663 + text += '~0';
  1664 +
  1665 + text = text.replace(/(?:^|\n)```(.*)\n([\s\S]*?)\n```/g, function (wholeMatch, language, codeblock) {
  1666 + var end = (options.omitExtraWLInCodeBlocks) ? '' : '\n';
  1667 +
  1668 + // First parse the github code block
  1669 + codeblock = showdown.subParser('encodeCode')(codeblock);
  1670 + codeblock = showdown.subParser('detab')(codeblock);
  1671 + codeblock = codeblock.replace(/^\n+/g, ''); // trim leading newlines
  1672 + codeblock = codeblock.replace(/\n+$/g, ''); // trim trailing whitespace
  1673 +
  1674 + codeblock = '<pre><code' + (language ? ' class="' + language + ' language-' + language + '"' : '') + '>' + codeblock + end + '</code></pre>';
  1675 +
  1676 + codeblock = showdown.subParser('hashBlock')(codeblock, options, globals);
  1677 +
  1678 + // Since GHCodeblocks can be false positives, we need to
  1679 + // store the primitive text and the parsed text in a global var,
  1680 + // and then return a token
  1681 + return '\n\n~G' + (globals.ghCodeBlocks.push({text: wholeMatch, codeblock: codeblock}) - 1) + 'G\n\n';
  1682 + });
  1683 +
  1684 + // attacklab: strip sentinel
  1685 + text = text.replace(/~0/, '');
  1686 +
  1687 + return globals.converter._dispatch('githubCodeBlocks.after', text, options, globals);
  1688 +});
  1689 +
  1690 +showdown.subParser('hashBlock', function (text, options, globals) {
  1691 + 'use strict';
  1692 + text = text.replace(/(^\n+|\n+$)/g, '');
  1693 + return '\n\n~K' + (globals.gHtmlBlocks.push(text) - 1) + 'K\n\n';
  1694 +});
  1695 +
  1696 +showdown.subParser('hashElement', function (text, options, globals) {
  1697 + 'use strict';
  1698 +
  1699 + return function (wholeMatch, m1) {
  1700 + var blockText = m1;
  1701 +
  1702 + // Undo double lines
  1703 + blockText = blockText.replace(/\n\n/g, '\n');
  1704 + blockText = blockText.replace(/^\n/, '');
  1705 +
  1706 + // strip trailing blank lines
  1707 + blockText = blockText.replace(/\n+$/g, '');
  1708 +
  1709 + // Replace the element text with a marker ("~KxK" where x is its key)
  1710 + blockText = '\n\n~K' + (globals.gHtmlBlocks.push(blockText) - 1) + 'K\n\n';
  1711 +
  1712 + return blockText;
  1713 + };
  1714 +});
  1715 +
  1716 +showdown.subParser('hashHTMLBlocks', function (text, options, globals) {
  1717 + 'use strict';
  1718 +
  1719 + var blockTags = [
  1720 + 'pre',
  1721 + 'div',
  1722 + 'h1',
  1723 + 'h2',
  1724 + 'h3',
  1725 + 'h4',
  1726 + 'h5',
  1727 + 'h6',
  1728 + 'blockquote',
  1729 + 'table',
  1730 + 'dl',
  1731 + 'ol',
  1732 + 'ul',
  1733 + 'script',
  1734 + 'noscript',
  1735 + 'form',
  1736 + 'fieldset',
  1737 + 'iframe',
  1738 + 'math',
  1739 + 'style',
  1740 + 'section',
  1741 + 'header',
  1742 + 'footer',
  1743 + 'nav',
  1744 + 'article',
  1745 + 'aside',
  1746 + 'address',
  1747 + 'audio',
  1748 + 'canvas',
  1749 + 'figure',
  1750 + 'hgroup',
  1751 + 'output',
  1752 + 'video',
  1753 + 'p'
  1754 + ],
  1755 + repFunc = function (wholeMatch, match, left, right) {
  1756 + var txt = wholeMatch;
  1757 + // check if this html element is marked as markdown
  1758 + // if so, it's contents should be parsed as markdown
  1759 + if (left.search(/\bmarkdown\b/) !== -1) {
  1760 + txt = left + globals.converter.makeHtml(match) + right;
  1761 + }
  1762 + return '\n\n~K' + (globals.gHtmlBlocks.push(txt) - 1) + 'K\n\n';
  1763 + };
  1764 +
  1765 + for (var i = 0; i < blockTags.length; ++i) {
  1766 + text = showdown.helper.replaceRecursiveRegExp(text, repFunc, '^(?: |\\t){0,3}<' + blockTags[i] + '\\b[^>]*>', '</' + blockTags[i] + '>', 'gim');
  1767 + }
  1768 +
  1769 + // HR SPECIAL CASE
  1770 + text = text.replace(/(\n[ ]{0,3}(<(hr)\b([^<>])*?\/?>)[ \t]*(?=\n{2,}))/g,
  1771 + showdown.subParser('hashElement')(text, options, globals));
  1772 +
  1773 + // Special case for standalone HTML comments:
  1774 + text = text.replace(/(<!--[\s\S]*?-->)/g,
  1775 + showdown.subParser('hashElement')(text, options, globals));
  1776 +
  1777 + // PHP and ASP-style processor instructions (<?...?> and <%...%>)
  1778 + text = text.replace(/(?:\n\n)([ ]{0,3}(?:<([?%])[^\r]*?\2>)[ \t]*(?=\n{2,}))/g,
  1779 + showdown.subParser('hashElement')(text, options, globals));
  1780 + return text;
  1781 +});
  1782 +
  1783 +/**
  1784 + * Hash span elements that should not be parsed as markdown
  1785 + */
  1786 +showdown.subParser('hashHTMLSpans', function (text, config, globals) {
  1787 + 'use strict';
  1788 +
  1789 + var matches = showdown.helper.matchRecursiveRegExp(text, '<code\\b[^>]*>', '</code>', 'gi');
  1790 +
  1791 + for (var i = 0; i < matches.length; ++i) {
  1792 + text = text.replace(matches[i][0], '~L' + (globals.gHtmlSpans.push(matches[i][0]) - 1) + 'L');
  1793 + }
  1794 + return text;
  1795 +});
  1796 +
  1797 +/**
  1798 + * Unhash HTML spans
  1799 + */
  1800 +showdown.subParser('unhashHTMLSpans', function (text, config, globals) {
  1801 + 'use strict';
  1802 +
  1803 + for (var i = 0; i < globals.gHtmlSpans.length; ++i) {
  1804 + text = text.replace('~L' + i + 'L', globals.gHtmlSpans[i]);
  1805 + }
  1806 +
  1807 + return text;
  1808 +});
  1809 +
  1810 +/**
  1811 + * Hash span elements that should not be parsed as markdown
  1812 + */
  1813 +showdown.subParser('hashPreCodeTags', function (text, config, globals) {
  1814 + 'use strict';
  1815 +
  1816 + var repFunc = function (wholeMatch, match, left, right) {
  1817 + // encode html entities
  1818 + var codeblock = left + showdown.subParser('encodeCode')(match) + right;
  1819 + return '\n\n~G' + (globals.ghCodeBlocks.push({text: wholeMatch, codeblock: codeblock}) - 1) + 'G\n\n';
  1820 + };
  1821 +
  1822 + text = showdown.helper.replaceRecursiveRegExp(text, repFunc, '^(?: |\\t){0,3}<pre\\b[^>]*>\\s*<code\\b[^>]*>', '^(?: |\\t){0,3}</code>\\s*</pre>', 'gim');
  1823 + return text;
  1824 +});
  1825 +
  1826 +showdown.subParser('headers', function (text, options, globals) {
  1827 + 'use strict';
  1828 +
  1829 + text = globals.converter._dispatch('headers.before', text, options, globals);
  1830 +
  1831 + var prefixHeader = options.prefixHeaderId,
  1832 + headerLevelStart = (isNaN(parseInt(options.headerLevelStart))) ? 1 : parseInt(options.headerLevelStart),
  1833 +
  1834 + // Set text-style headers:
  1835 + // Header 1
  1836 + // ========
  1837 + //
  1838 + // Header 2
  1839 + // --------
  1840 + //
  1841 + setextRegexH1 = (options.smoothLivePreview) ? /^(.+)[ \t]*\n={2,}[ \t]*\n+/gm : /^(.+)[ \t]*\n=+[ \t]*\n+/gm,
  1842 + setextRegexH2 = (options.smoothLivePreview) ? /^(.+)[ \t]*\n-{2,}[ \t]*\n+/gm : /^(.+)[ \t]*\n-+[ \t]*\n+/gm;
  1843 +
  1844 + text = text.replace(setextRegexH1, function (wholeMatch, m1) {
  1845 +
  1846 + var spanGamut = showdown.subParser('spanGamut')(m1, options, globals),
  1847 + hID = (options.noHeaderId) ? '' : ' id="' + headerId(m1) + '"',
  1848 + hLevel = headerLevelStart,
  1849 + hashBlock = '<h' + hLevel + hID + '>' + spanGamut + '</h' + hLevel + '>';
  1850 + return showdown.subParser('hashBlock')(hashBlock, options, globals);
  1851 + });
  1852 +
  1853 + text = text.replace(setextRegexH2, function (matchFound, m1) {
  1854 + var spanGamut = showdown.subParser('spanGamut')(m1, options, globals),
  1855 + hID = (options.noHeaderId) ? '' : ' id="' + headerId(m1) + '"',
  1856 + hLevel = headerLevelStart + 1,
  1857 + hashBlock = '<h' + hLevel + hID + '>' + spanGamut + '</h' + hLevel + '>';
  1858 + return showdown.subParser('hashBlock')(hashBlock, options, globals);
  1859 + });
  1860 +
  1861 + // atx-style headers:
  1862 + // # Header 1
  1863 + // ## Header 2
  1864 + // ## Header 2 with closing hashes ##
  1865 + // ...
  1866 + // ###### Header 6
  1867 + //
  1868 + text = text.replace(/^(#{1,6})[ \t]*(.+?)[ \t]*#*\n+/gm, function (wholeMatch, m1, m2) {
  1869 + var span = showdown.subParser('spanGamut')(m2, options, globals),
  1870 + hID = (options.noHeaderId) ? '' : ' id="' + headerId(m2) + '"',
  1871 + hLevel = headerLevelStart - 1 + m1.length,
  1872 + header = '<h' + hLevel + hID + '>' + span + '</h' + hLevel + '>';
  1873 +
  1874 + return showdown.subParser('hashBlock')(header, options, globals);
  1875 + });
  1876 +
  1877 + function headerId(m) {
  1878 + var title, escapedId = m.replace(/[^\w]/g, '').toLowerCase();
  1879 +
  1880 + if (globals.hashLinkCounts[escapedId]) {
  1881 + title = escapedId + '-' + (globals.hashLinkCounts[escapedId]++);
  1882 + } else {
  1883 + title = escapedId;
  1884 + globals.hashLinkCounts[escapedId] = 1;
  1885 + }
  1886 +
  1887 + // Prefix id to prevent causing inadvertent pre-existing style matches.
  1888 + if (prefixHeader === true) {
  1889 + prefixHeader = 'section';
  1890 + }
  1891 +
  1892 + if (showdown.helper.isString(prefixHeader)) {
  1893 + return prefixHeader + title;
  1894 + }
  1895 + return title;
  1896 + }
  1897 +
  1898 + text = globals.converter._dispatch('headers.after', text, options, globals);
  1899 + return text;
  1900 +});
  1901 +
  1902 +/**
  1903 + * Turn Markdown image shortcuts into <img> tags.
  1904 + */
  1905 +showdown.subParser('images', function (text, options, globals) {
  1906 + 'use strict';
  1907 +
  1908 + text = globals.converter._dispatch('images.before', text, options, globals);
  1909 +
  1910 + var inlineRegExp = /!\[(.*?)]\s?\([ \t]*()<?(\S+?)>?(?: =([*\d]+[A-Za-z%]{0,4})x([*\d]+[A-Za-z%]{0,4}))?[ \t]*(?:(['"])(.*?)\6[ \t]*)?\)/g,
  1911 + referenceRegExp = /!\[([^\]]*?)] ?(?:\n *)?\[(.*?)]()()()()()/g;
  1912 +
  1913 + function writeImageTag (wholeMatch, altText, linkId, url, width, height, m5, title) {
  1914 +
  1915 + var gUrls = globals.gUrls,
  1916 + gTitles = globals.gTitles,
  1917 + gDims = globals.gDimensions;
  1918 +
  1919 + linkId = linkId.toLowerCase();
  1920 +
  1921 + if (!title) {
  1922 + title = '';
  1923 + }
  1924 +
  1925 + if (url === '' || url === null) {
  1926 + if (linkId === '' || linkId === null) {
  1927 + // lower-case and turn embedded newlines into spaces
  1928 + linkId = altText.toLowerCase().replace(/ ?\n/g, ' ');
  1929 + }
  1930 + url = '#' + linkId;
  1931 +
  1932 + if (!showdown.helper.isUndefined(gUrls[linkId])) {
  1933 + url = gUrls[linkId];
  1934 + if (!showdown.helper.isUndefined(gTitles[linkId])) {
  1935 + title = gTitles[linkId];
  1936 + }
  1937 + if (!showdown.helper.isUndefined(gDims[linkId])) {
  1938 + width = gDims[linkId].width;
  1939 + height = gDims[linkId].height;
  1940 + }
  1941 + } else {
  1942 + return wholeMatch;
  1943 + }
  1944 + }
  1945 +
  1946 + altText = altText.replace(/"/g, '&quot;');
  1947 + altText = showdown.helper.escapeCharacters(altText, '*_', false);
  1948 + url = showdown.helper.escapeCharacters(url, '*_', false);
  1949 + var result = '<img src="' + url + '" alt="' + altText + '"';
  1950 +
  1951 + if (title) {
  1952 + title = title.replace(/"/g, '&quot;');
  1953 + title = showdown.helper.escapeCharacters(title, '*_', false);
  1954 + result += ' title="' + title + '"';
  1955 + }
  1956 +
  1957 + if (width && height) {
  1958 + width = (width === '*') ? 'auto' : width;
  1959 + height = (height === '*') ? 'auto' : height;
  1960 +
  1961 + result += ' width="' + width + '"';
  1962 + result += ' height="' + height + '"';
  1963 + }
  1964 +
  1965 + result += ' />';
  1966 + return result;
  1967 + }
  1968 +
  1969 + // First, handle reference-style labeled images: ![alt text][id]
  1970 + text = text.replace(referenceRegExp, writeImageTag);
  1971 +
  1972 + // Next, handle inline images: ![alt text](url =<width>x<height> "optional title")
  1973 + text = text.replace(inlineRegExp, writeImageTag);
  1974 +
  1975 + text = globals.converter._dispatch('images.after', text, options, globals);
  1976 + return text;
  1977 +});
  1978 +
  1979 +showdown.subParser('italicsAndBold', function (text, options, globals) {
  1980 + 'use strict';
  1981 +
  1982 + text = globals.converter._dispatch('italicsAndBold.before', text, options, globals);
  1983 +
  1984 + if (options.literalMidWordUnderscores) {
  1985 + //underscores
  1986 + // Since we are consuming a \s character, we need to add it
  1987 + text = text.replace(/(^|\s|>|\b)__(?=\S)([\s\S]+?)__(?=\b|<|\s|$)/gm, '$1<strong>$2</strong>');
  1988 + text = text.replace(/(^|\s|>|\b)_(?=\S)([\s\S]+?)_(?=\b|<|\s|$)/gm, '$1<em>$2</em>');
  1989 + //asterisks
  1990 + text = text.replace(/(\*\*)(?=\S)([^\r]*?\S[*]*)\1/g, '<strong>$2</strong>');
  1991 + text = text.replace(/(\*)(?=\S)([^\r]*?\S)\1/g, '<em>$2</em>');
  1992 +
  1993 + } else {
  1994 + // <strong> must go first:
  1995 + text = text.replace(/(\*\*|__)(?=\S)([^\r]*?\S[*_]*)\1/g, '<strong>$2</strong>');
  1996 + text = text.replace(/(\*|_)(?=\S)([^\r]*?\S)\1/g, '<em>$2</em>');
  1997 + }
  1998 +
  1999 + text = globals.converter._dispatch('italicsAndBold.after', text, options, globals);
  2000 + return text;
  2001 +});
  2002 +
  2003 +/**
  2004 + * Form HTML ordered (numbered) and unordered (bulleted) lists.
  2005 + */
  2006 +showdown.subParser('lists', function (text, options, globals) {
  2007 + 'use strict';
  2008 +
  2009 + text = globals.converter._dispatch('lists.before', text, options, globals);
  2010 + /**
  2011 + * Process the contents of a single ordered or unordered list, splitting it
  2012 + * into individual list items.
  2013 + * @param {string} listStr
  2014 + * @param {boolean} trimTrailing
  2015 + * @returns {string}
  2016 + */
  2017 + function processListItems (listStr, trimTrailing) {
  2018 + // The $g_list_level global keeps track of when we're inside a list.
  2019 + // Each time we enter a list, we increment it; when we leave a list,
  2020 + // we decrement. If it's zero, we're not in a list anymore.
  2021 + //
  2022 + // We do this because when we're not inside a list, we want to treat
  2023 + // something like this:
  2024 + //
  2025 + // I recommend upgrading to version
  2026 + // 8. Oops, now this line is treated
  2027 + // as a sub-list.
  2028 + //
  2029 + // As a single paragraph, despite the fact that the second line starts
  2030 + // with a digit-period-space sequence.
  2031 + //
  2032 + // Whereas when we're inside a list (or sub-list), that line will be
  2033 + // treated as the start of a sub-list. What a kludge, huh? This is
  2034 + // an aspect of Markdown's syntax that's hard to parse perfectly
  2035 + // without resorting to mind-reading. Perhaps the solution is to
  2036 + // change the syntax rules such that sub-lists must start with a
  2037 + // starting cardinal number; e.g. "1." or "a.".
  2038 + globals.gListLevel++;
  2039 +
  2040 + // trim trailing blank lines:
  2041 + listStr = listStr.replace(/\n{2,}$/, '\n');
  2042 +
  2043 + // attacklab: add sentinel to emulate \z
  2044 + listStr += '~0';
  2045 +
  2046 + var rgx = /(\n)?(^[ \t]*)([*+-]|\d+[.])[ \t]+((\[(x|X| )?])?[ \t]*[^\r]+?(\n{1,2}))(?=\n*(~0|\2([*+-]|\d+[.])[ \t]+))/gm,
  2047 + isParagraphed = (/\n[ \t]*\n(?!~0)/.test(listStr));
  2048 +
  2049 + listStr = listStr.replace(rgx, function (wholeMatch, m1, m2, m3, m4, taskbtn, checked) {
  2050 + checked = (checked && checked.trim() !== '');
  2051 + var item = showdown.subParser('outdent')(m4, options, globals),
  2052 + bulletStyle = '';
  2053 +
  2054 + // Support for github tasklists
  2055 + if (taskbtn && options.tasklists) {
  2056 + bulletStyle = ' class="task-list-item" style="list-style-type: none;"';
  2057 + item = item.replace(/^[ \t]*\[(x|X| )?]/m, function () {
  2058 + var otp = '<input type="checkbox" disabled style="margin: 0px 0.35em 0.25em -1.6em; vertical-align: middle;"';
  2059 + if (checked) {
  2060 + otp += ' checked';
  2061 + }
  2062 + otp += '>';
  2063 + return otp;
  2064 + });
  2065 + }
  2066 + // m1 - Leading line or
  2067 + // Has a double return (multi paragraph) or
  2068 + // Has sublist
  2069 + if (m1 || (item.search(/\n{2,}/) > -1)) {
  2070 + item = showdown.subParser('githubCodeBlocks')(item, options, globals);
  2071 + item = showdown.subParser('blockGamut')(item, options, globals);
  2072 + } else {
  2073 + // Recursion for sub-lists:
  2074 + item = showdown.subParser('lists')(item, options, globals);
  2075 + item = item.replace(/\n$/, ''); // chomp(item)
  2076 + if (isParagraphed) {
  2077 + item = showdown.subParser('paragraphs')(item, options, globals);
  2078 + } else {
  2079 + item = showdown.subParser('spanGamut')(item, options, globals);
  2080 + }
  2081 + }
  2082 + item = '\n<li' + bulletStyle + '>' + item + '</li>\n';
  2083 + return item;
  2084 + });
  2085 +
  2086 + // attacklab: strip sentinel
  2087 + listStr = listStr.replace(/~0/g, '');
  2088 +
  2089 + globals.gListLevel--;
  2090 +
  2091 + if (trimTrailing) {
  2092 + listStr = listStr.replace(/\s+$/, '');
  2093 + }
  2094 +
  2095 + return listStr;
  2096 + }
  2097 +
  2098 + /**
  2099 + * Check and parse consecutive lists (better fix for issue #142)
  2100 + * @param {string} list
  2101 + * @param {string} listType
  2102 + * @param {boolean} trimTrailing
  2103 + * @returns {string}
  2104 + */
  2105 + function parseConsecutiveLists(list, listType, trimTrailing) {
  2106 + // check if we caught 2 or more consecutive lists by mistake
  2107 + // we use the counterRgx, meaning if listType is UL we look for UL and vice versa
  2108 + var counterRxg = (listType === 'ul') ? /^ {0,2}\d+\.[ \t]/gm : /^ {0,2}[*+-][ \t]/gm,
  2109 + subLists = [],
  2110 + result = '';
  2111 +
  2112 + if (list.search(counterRxg) !== -1) {
  2113 + (function parseCL(txt) {
  2114 + var pos = txt.search(counterRxg);
  2115 + if (pos !== -1) {
  2116 + // slice
  2117 + result += '\n\n<' + listType + '>' + processListItems(txt.slice(0, pos), !!trimTrailing) + '</' + listType + '>\n\n';
  2118 +
  2119 + // invert counterType and listType
  2120 + listType = (listType === 'ul') ? 'ol' : 'ul';
  2121 + counterRxg = (listType === 'ul') ? /^ {0,2}\d+\.[ \t]/gm : /^ {0,2}[*+-][ \t]/gm;
  2122 +
  2123 + //recurse
  2124 + parseCL(txt.slice(pos));
  2125 + } else {
  2126 + result += '\n\n<' + listType + '>' + processListItems(txt, !!trimTrailing) + '</' + listType + '>\n\n';
  2127 + }
  2128 + })(list);
  2129 + for (var i = 0; i < subLists.length; ++i) {
  2130 +
  2131 + }
  2132 + } else {
  2133 + result = '\n\n<' + listType + '>' + processListItems(list, !!trimTrailing) + '</' + listType + '>\n\n';
  2134 + }
  2135 +
  2136 + return result;
  2137 + }
  2138 +
  2139 + // attacklab: add sentinel to hack around khtml/safari bug:
  2140 + // http://bugs.webkit.org/show_bug.cgi?id=11231
  2141 + text += '~0';
  2142 +
  2143 + // Re-usable pattern to match any entire ul or ol list:
  2144 + var wholeList = /^(([ ]{0,3}([*+-]|\d+[.])[ \t]+)[^\r]+?(~0|\n{2,}(?=\S)(?![ \t]*(?:[*+-]|\d+[.])[ \t]+)))/gm;
  2145 +
  2146 + if (globals.gListLevel) {
  2147 + text = text.replace(wholeList, function (wholeMatch, list, m2) {
  2148 + var listType = (m2.search(/[*+-]/g) > -1) ? 'ul' : 'ol';
  2149 + return parseConsecutiveLists(list, listType, true);
  2150 + });
  2151 + } else {
  2152 + wholeList = /(\n\n|^\n?)(([ ]{0,3}([*+-]|\d+[.])[ \t]+)[^\r]+?(~0|\n{2,}(?=\S)(?![ \t]*(?:[*+-]|\d+[.])[ \t]+)))/gm;
  2153 + //wholeList = /(\n\n|^\n?)( {0,3}([*+-]|\d+\.)[ \t]+[\s\S]+?)(?=(~0)|(\n\n(?!\t| {2,}| {0,3}([*+-]|\d+\.)[ \t])))/g;
  2154 + text = text.replace(wholeList, function (wholeMatch, m1, list, m3) {
  2155 +
  2156 + var listType = (m3.search(/[*+-]/g) > -1) ? 'ul' : 'ol';
  2157 + return parseConsecutiveLists(list, listType);
  2158 + });
  2159 + }
  2160 +
  2161 + // attacklab: strip sentinel
  2162 + text = text.replace(/~0/, '');
  2163 +
  2164 + text = globals.converter._dispatch('lists.after', text, options, globals);
  2165 + return text;
  2166 +});
  2167 +
  2168 +/**
  2169 + * Remove one level of line-leading tabs or spaces
  2170 + */
  2171 +showdown.subParser('outdent', function (text) {
  2172 + 'use strict';
  2173 +
  2174 + // attacklab: hack around Konqueror 3.5.4 bug:
  2175 + // "----------bug".replace(/^-/g,"") == "bug"
  2176 + text = text.replace(/^(\t|[ ]{1,4})/gm, '~0'); // attacklab: g_tab_width
  2177 +
  2178 + // attacklab: clean up hack
  2179 + text = text.replace(/~0/g, '');
  2180 +
  2181 + return text;
  2182 +});
  2183 +
  2184 +/**
  2185 + *
  2186 + */
  2187 +showdown.subParser('paragraphs', function (text, options, globals) {
  2188 + 'use strict';
  2189 +
  2190 + text = globals.converter._dispatch('paragraphs.before', text, options, globals);
  2191 + // Strip leading and trailing lines:
  2192 + text = text.replace(/^\n+/g, '');
  2193 + text = text.replace(/\n+$/g, '');
  2194 +
  2195 + var grafs = text.split(/\n{2,}/g),
  2196 + grafsOut = [],
  2197 + end = grafs.length; // Wrap <p> tags
  2198 +
  2199 + for (var i = 0; i < end; i++) {
  2200 + var str = grafs[i];
  2201 + // if this is an HTML marker, copy it
  2202 + if (str.search(/~(K|G)(\d+)\1/g) >= 0) {
  2203 + grafsOut.push(str);
  2204 + } else {
  2205 + str = showdown.subParser('spanGamut')(str, options, globals);
  2206 + str = str.replace(/^([ \t]*)/g, '<p>');
  2207 + str += '</p>';
  2208 + grafsOut.push(str);
  2209 + }
  2210 + }
  2211 +
  2212 + /** Unhashify HTML blocks */
  2213 + end = grafsOut.length;
  2214 + for (i = 0; i < end; i++) {
  2215 + var blockText = '',
  2216 + grafsOutIt = grafsOut[i],
  2217 + codeFlag = false;
  2218 + // if this is a marker for an html block...
  2219 + while (grafsOutIt.search(/~(K|G)(\d+)\1/) >= 0) {
  2220 + var delim = RegExp.$1,
  2221 + num = RegExp.$2;
  2222 +
  2223 + if (delim === 'K') {
  2224 + blockText = globals.gHtmlBlocks[num];
  2225 + } else {
  2226 + // we need to check if ghBlock is a false positive
  2227 + if (codeFlag) {
  2228 + // use encoded version of all text
  2229 + blockText = showdown.subParser('encodeCode')(globals.ghCodeBlocks[num].text);
  2230 + } else {
  2231 + blockText = globals.ghCodeBlocks[num].codeblock;
  2232 + }
  2233 + }
  2234 + blockText = blockText.replace(/\$/g, '$$$$'); // Escape any dollar signs
  2235 +
  2236 + grafsOutIt = grafsOutIt.replace(/(\n\n)?~(K|G)\d+\2(\n\n)?/, blockText);
  2237 + // Check if grafsOutIt is a pre->code
  2238 + if (/^<pre\b[^>]*>\s*<code\b[^>]*>/.test(grafsOutIt)) {
  2239 + codeFlag = true;
  2240 + }
  2241 + }
  2242 + grafsOut[i] = grafsOutIt;
  2243 + }
  2244 + text = grafsOut.join('\n\n');
  2245 + // Strip leading and trailing lines:
  2246 + text = text.replace(/^\n+/g, '');
  2247 + text = text.replace(/\n+$/g, '');
  2248 + return globals.converter._dispatch('paragraphs.after', text, options, globals);
  2249 +});
  2250 +
  2251 +/**
  2252 + * Run extension
  2253 + */
  2254 +showdown.subParser('runExtension', function (ext, text, options, globals) {
  2255 + 'use strict';
  2256 +
  2257 + if (ext.filter) {
  2258 + text = ext.filter(text, globals.converter, options);
  2259 +
  2260 + } else if (ext.regex) {
  2261 + // TODO remove this when old extension loading mechanism is deprecated
  2262 + var re = ext.regex;
  2263 + if (!re instanceof RegExp) {
  2264 + re = new RegExp(re, 'g');
  2265 + }
  2266 + text = text.replace(re, ext.replace);
  2267 + }
  2268 +
  2269 + return text;
  2270 +});
  2271 +
  2272 +/**
  2273 + * These are all the transformations that occur *within* block-level
  2274 + * tags like paragraphs, headers, and list items.
  2275 + */
  2276 +showdown.subParser('spanGamut', function (text, options, globals) {
  2277 + 'use strict';
  2278 +
  2279 + text = globals.converter._dispatch('spanGamut.before', text, options, globals);
  2280 + text = showdown.subParser('codeSpans')(text, options, globals);
  2281 + text = showdown.subParser('escapeSpecialCharsWithinTagAttributes')(text, options, globals);
  2282 + text = showdown.subParser('encodeBackslashEscapes')(text, options, globals);
  2283 +
  2284 + // Process anchor and image tags. Images must come first,
  2285 + // because ![foo][f] looks like an anchor.
  2286 + text = showdown.subParser('images')(text, options, globals);
  2287 + text = showdown.subParser('anchors')(text, options, globals);
  2288 +
  2289 + // Make links out of things like `<http://example.com/>`
  2290 + // Must come after _DoAnchors(), because you can use < and >
  2291 + // delimiters in inline links like [this](<url>).
  2292 + text = showdown.subParser('autoLinks')(text, options, globals);
  2293 + text = showdown.subParser('encodeAmpsAndAngles')(text, options, globals);
  2294 + text = showdown.subParser('italicsAndBold')(text, options, globals);
  2295 + text = showdown.subParser('strikethrough')(text, options, globals);
  2296 +
  2297 + // Do hard breaks:
  2298 + text = text.replace(/ +\n/g, ' <br />\n');
  2299 +
  2300 + text = globals.converter._dispatch('spanGamut.after', text, options, globals);
  2301 + return text;
  2302 +});
  2303 +
  2304 +showdown.subParser('strikethrough', function (text, options, globals) {
  2305 + 'use strict';
  2306 +
  2307 + if (options.strikethrough) {
  2308 + text = globals.converter._dispatch('strikethrough.before', text, options, globals);
  2309 + text = text.replace(/(?:~T){2}([\s\S]+?)(?:~T){2}/g, '<del>$1</del>');
  2310 + text = globals.converter._dispatch('strikethrough.after', text, options, globals);
  2311 + }
  2312 +
  2313 + return text;
  2314 +});
  2315 +
  2316 +/**
  2317 + * Strip any lines consisting only of spaces and tabs.
  2318 + * This makes subsequent regexs easier to write, because we can
  2319 + * match consecutive blank lines with /\n+/ instead of something
  2320 + * contorted like /[ \t]*\n+/
  2321 + */
  2322 +showdown.subParser('stripBlankLines', function (text) {
  2323 + 'use strict';
  2324 + return text.replace(/^[ \t]+$/mg, '');
  2325 +});
  2326 +
  2327 +/**
  2328 + * Strips link definitions from text, stores the URLs and titles in
  2329 + * hash references.
  2330 + * Link defs are in the form: ^[id]: url "optional title"
  2331 + *
  2332 + * ^[ ]{0,3}\[(.+)\]: // id = $1 attacklab: g_tab_width - 1
  2333 + * [ \t]*
  2334 + * \n? // maybe *one* newline
  2335 + * [ \t]*
  2336 + * <?(\S+?)>? // url = $2
  2337 + * [ \t]*
  2338 + * \n? // maybe one newline
  2339 + * [ \t]*
  2340 + * (?:
  2341 + * (\n*) // any lines skipped = $3 attacklab: lookbehind removed
  2342 + * ["(]
  2343 + * (.+?) // title = $4
  2344 + * [")]
  2345 + * [ \t]*
  2346 + * )? // title is optional
  2347 + * (?:\n+|$)
  2348 + * /gm,
  2349 + * function(){...});
  2350 + *
  2351 + */
  2352 +showdown.subParser('stripLinkDefinitions', function (text, options, globals) {
  2353 + 'use strict';
  2354 +
  2355 + var regex = /^ {0,3}\[(.+)]:[ \t]*\n?[ \t]*<?(\S+?)>?(?: =([*\d]+[A-Za-z%]{0,4})x([*\d]+[A-Za-z%]{0,4}))?[ \t]*\n?[ \t]*(?:(\n*)["|'(](.+?)["|')][ \t]*)?(?:\n+|(?=~0))/gm;
  2356 +
  2357 + // attacklab: sentinel workarounds for lack of \A and \Z, safari\khtml bug
  2358 + text += '~0';
  2359 +
  2360 + text = text.replace(regex, function (wholeMatch, linkId, url, width, height, blankLines, title) {
  2361 + linkId = linkId.toLowerCase();
  2362 + globals.gUrls[linkId] = showdown.subParser('encodeAmpsAndAngles')(url); // Link IDs are case-insensitive
  2363 +
  2364 + if (blankLines) {
  2365 + // Oops, found blank lines, so it's not a title.
  2366 + // Put back the parenthetical statement we stole.
  2367 + return blankLines + title;
  2368 +
  2369 + } else {
  2370 + if (title) {
  2371 + globals.gTitles[linkId] = title.replace(/"|'/g, '&quot;');
  2372 + }
  2373 + if (options.parseImgDimensions && width && height) {
  2374 + globals.gDimensions[linkId] = {
  2375 + width: width,
  2376 + height: height
  2377 + };
  2378 + }
  2379 + }
  2380 + // Completely remove the definition from the text
  2381 + return '';
  2382 + });
  2383 +
  2384 + // attacklab: strip sentinel
  2385 + text = text.replace(/~0/, '');
  2386 +
  2387 + return text;
  2388 +});
  2389 +
  2390 +showdown.subParser('tables', function (text, options, globals) {
  2391 + 'use strict';
  2392 +
  2393 + if (!options.tables) {
  2394 + return text;
  2395 + }
  2396 +
  2397 + var tableRgx = /^[ \t]{0,3}\|?.+\|.+\n[ \t]{0,3}\|?[ \t]*:?[ \t]*(?:-|=){2,}[ \t]*:?[ \t]*\|[ \t]*:?[ \t]*(?:-|=){2,}[\s\S]+?(?:\n\n|~0)/gm;
  2398 +
  2399 + function parseStyles(sLine) {
  2400 + if (/^:[ \t]*--*$/.test(sLine)) {
  2401 + return ' style="text-align:left;"';
  2402 + } else if (/^--*[ \t]*:[ \t]*$/.test(sLine)) {
  2403 + return ' style="text-align:right;"';
  2404 + } else if (/^:[ \t]*--*[ \t]*:$/.test(sLine)) {
  2405 + return ' style="text-align:center;"';
  2406 + } else {
  2407 + return '';
  2408 + }
  2409 + }
  2410 +
  2411 + function parseHeaders(header, style) {
  2412 + var id = '';
  2413 + header = header.trim();
  2414 + if (options.tableHeaderId) {
  2415 + id = ' id="' + header.replace(/ /g, '_').toLowerCase() + '"';
  2416 + }
  2417 + header = showdown.subParser('spanGamut')(header, options, globals);
  2418 +
  2419 + return '<th' + id + style + '>' + header + '</th>\n';
  2420 + }
  2421 +
  2422 + function parseCells(cell, style) {
  2423 + var subText = showdown.subParser('spanGamut')(cell, options, globals);
  2424 + return '<td' + style + '>' + subText + '</td>\n';
  2425 + }
  2426 +
  2427 + function buildTable(headers, cells) {
  2428 + var tb = '<table>\n<thead>\n<tr>\n',
  2429 + tblLgn = headers.length;
  2430 +
  2431 + for (var i = 0; i < tblLgn; ++i) {
  2432 + tb += headers[i];
  2433 + }
  2434 + tb += '</tr>\n</thead>\n<tbody>\n';
  2435 +
  2436 + for (i = 0; i < cells.length; ++i) {
  2437 + tb += '<tr>\n';
  2438 + for (var ii = 0; ii < tblLgn; ++ii) {
  2439 + tb += cells[i][ii];
  2440 + }
  2441 + tb += '</tr>\n';
  2442 + }
  2443 + tb += '</tbody>\n</table>\n';
  2444 + return tb;
  2445 + }
  2446 +
  2447 + text = globals.converter._dispatch('tables.before', text, options, globals);
  2448 +
  2449 + text = text.replace(tableRgx, function (rawTable) {
  2450 +
  2451 + var i, tableLines = rawTable.split('\n');
  2452 +
  2453 + // strip wrong first and last column if wrapped tables are used
  2454 + for (i = 0; i < tableLines.length; ++i) {
  2455 + if (/^[ \t]{0,3}\|/.test(tableLines[i])) {
  2456 + tableLines[i] = tableLines[i].replace(/^[ \t]{0,3}\|/, '');
  2457 + }
  2458 + if (/\|[ \t]*$/.test(tableLines[i])) {
  2459 + tableLines[i] = tableLines[i].replace(/\|[ \t]*$/, '');
  2460 + }
  2461 + }
  2462 +
  2463 + var rawHeaders = tableLines[0].split('|').map(function (s) { return s.trim();}),
  2464 + rawStyles = tableLines[1].split('|').map(function (s) { return s.trim();}),
  2465 + rawCells = [],
  2466 + headers = [],
  2467 + styles = [],
  2468 + cells = [];
  2469 +
  2470 + tableLines.shift();
  2471 + tableLines.shift();
  2472 +
  2473 + for (i = 0; i < tableLines.length; ++i) {
  2474 + if (tableLines[i].trim() === '') {
  2475 + continue;
  2476 + }
  2477 + rawCells.push(
  2478 + tableLines[i]
  2479 + .split('|')
  2480 + .map(function (s) {
  2481 + return s.trim();
  2482 + })
  2483 + );
  2484 + }
  2485 +
  2486 + if (rawHeaders.length < rawStyles.length) {
  2487 + return rawTable;
  2488 + }
  2489 +
  2490 + for (i = 0; i < rawStyles.length; ++i) {
  2491 + styles.push(parseStyles(rawStyles[i]));
  2492 + }
  2493 +
  2494 + for (i = 0; i < rawHeaders.length; ++i) {
  2495 + if (showdown.helper.isUndefined(styles[i])) {
  2496 + styles[i] = '';
  2497 + }
  2498 + headers.push(parseHeaders(rawHeaders[i], styles[i]));
  2499 + }
  2500 +
  2501 + for (i = 0; i < rawCells.length; ++i) {
  2502 + var row = [];
  2503 + for (var ii = 0; ii < headers.length; ++ii) {
  2504 + if (showdown.helper.isUndefined(rawCells[i][ii])) {
  2505 +
  2506 + }
  2507 + row.push(parseCells(rawCells[i][ii], styles[ii]));
  2508 + }
  2509 + cells.push(row);
  2510 + }
  2511 +
  2512 + return buildTable(headers, cells);
  2513 + });
  2514 +
  2515 + text = globals.converter._dispatch('tables.after', text, options, globals);
  2516 +
  2517 + return text;
  2518 +});
  2519 +
  2520 +/**
  2521 + * Swap back in all the special characters we've hidden.
  2522 + */
  2523 +showdown.subParser('unescapeSpecialChars', function (text) {
  2524 + 'use strict';
  2525 +
  2526 + text = text.replace(/~E(\d+)E/g, function (wholeMatch, m1) {
  2527 + var charCodeToReplace = parseInt(m1);
  2528 + return String.fromCharCode(charCodeToReplace);
  2529 + });
  2530 + return text;
  2531 +});
  2532 +module.exports = showdown;
  1 +// HTML 支持的数学符号
  2 +function strNumDiscode(str){
  3 + str = str.replace(/&forall;/g, '∀');
  4 + str = str.replace(/&part;/g, '∂');
  5 + str = str.replace(/&exists;/g, '∃');
  6 + str = str.replace(/&empty;/g, '∅');
  7 + str = str.replace(/&nabla;/g, '∇');
  8 + str = str.replace(/&isin;/g, '∈');
  9 + str = str.replace(/&notin;/g, '∉');
  10 + str = str.replace(/&ni;/g, '∋');
  11 + str = str.replace(/&prod;/g, '∏');
  12 + str = str.replace(/&sum;/g, '∑');
  13 + str = str.replace(/&minus;/g, '−');
  14 + str = str.replace(/&lowast;/g, '∗');
  15 + str = str.replace(/&radic;/g, '√');
  16 + str = str.replace(/&prop;/g, '∝');
  17 + str = str.replace(/&infin;/g, '∞');
  18 + str = str.replace(/&ang;/g, '∠');
  19 + str = str.replace(/&and;/g, '∧');
  20 + str = str.replace(/&or;/g, '∨');
  21 + str = str.replace(/&cap;/g, '∩');
  22 + str = str.replace(/&cap;/g, '∪');
  23 + str = str.replace(/&int;/g, '∫');
  24 + str = str.replace(/&there4;/g, '∴');
  25 + str = str.replace(/&sim;/g, '∼');
  26 + str = str.replace(/&cong;/g, '≅');
  27 + str = str.replace(/&asymp;/g, '≈');
  28 + str = str.replace(/&ne;/g, '≠');
  29 + str = str.replace(/&le;/g, '≤');
  30 + str = str.replace(/&ge;/g, '≥');
  31 + str = str.replace(/&sub;/g, '⊂');
  32 + str = str.replace(/&sup;/g, '⊃');
  33 + str = str.replace(/&nsub;/g, '⊄');
  34 + str = str.replace(/&sube;/g, '⊆');
  35 + str = str.replace(/&supe;/g, '⊇');
  36 + str = str.replace(/&oplus;/g, '⊕');
  37 + str = str.replace(/&otimes;/g, '⊗');
  38 + str = str.replace(/&perp;/g, '⊥');
  39 + str = str.replace(/&sdot;/g, '⋅');
  40 + return str;
  41 +}
  42 +
  43 +//HTML 支持的希腊字母
  44 +function strGreeceDiscode(str){
  45 + str = str.replace(/&Alpha;/g, 'Α');
  46 + str = str.replace(/&Beta;/g, 'Β');
  47 + str = str.replace(/&Gamma;/g, 'Γ');
  48 + str = str.replace(/&Delta;/g, 'Δ');
  49 + str = str.replace(/&Epsilon;/g, 'Ε');
  50 + str = str.replace(/&Zeta;/g, 'Ζ');
  51 + str = str.replace(/&Eta;/g, 'Η');
  52 + str = str.replace(/&Theta;/g, 'Θ');
  53 + str = str.replace(/&Iota;/g, 'Ι');
  54 + str = str.replace(/&Kappa;/g, 'Κ');
  55 + str = str.replace(/&Lambda;/g, 'Λ');
  56 + str = str.replace(/&Mu;/g, 'Μ');
  57 + str = str.replace(/&Nu;/g, 'Ν');
  58 + str = str.replace(/&Xi;/g, 'Ν');
  59 + str = str.replace(/&Omicron;/g, 'Ο');
  60 + str = str.replace(/&Pi;/g, 'Π');
  61 + str = str.replace(/&Rho;/g, 'Ρ');
  62 + str = str.replace(/&Sigma;/g, 'Σ');
  63 + str = str.replace(/&Tau;/g, 'Τ');
  64 + str = str.replace(/&Upsilon;/g, 'Υ');
  65 + str = str.replace(/&Phi;/g, 'Φ');
  66 + str = str.replace(/&Chi;/g, 'Χ');
  67 + str = str.replace(/&Psi;/g, 'Ψ');
  68 + str = str.replace(/&Omega;/g, 'Ω');
  69 +
  70 + str = str.replace(/&alpha;/g, 'α');
  71 + str = str.replace(/&beta;/g, 'β');
  72 + str = str.replace(/&gamma;/g, 'γ');
  73 + str = str.replace(/&delta;/g, 'δ');
  74 + str = str.replace(/&epsilon;/g, 'ε');
  75 + str = str.replace(/&zeta;/g, 'ζ');
  76 + str = str.replace(/&eta;/g, 'η');
  77 + str = str.replace(/&theta;/g, 'θ');
  78 + str = str.replace(/&iota;/g, 'ι');
  79 + str = str.replace(/&kappa;/g, 'κ');
  80 + str = str.replace(/&lambda;/g, 'λ');
  81 + str = str.replace(/&mu;/g, 'μ');
  82 + str = str.replace(/&nu;/g, 'ν');
  83 + str = str.replace(/&xi;/g, 'ξ');
  84 + str = str.replace(/&omicron;/g, 'ο');
  85 + str = str.replace(/&pi;/g, 'π');
  86 + str = str.replace(/&rho;/g, 'ρ');
  87 + str = str.replace(/&sigmaf;/g, 'ς');
  88 + str = str.replace(/&sigma;/g, 'σ');
  89 + str = str.replace(/&tau;/g, 'τ');
  90 + str = str.replace(/&upsilon;/g, 'υ');
  91 + str = str.replace(/&phi;/g, 'φ');
  92 + str = str.replace(/&chi;/g, 'χ');
  93 + str = str.replace(/&psi;/g, 'ψ');
  94 + str = str.replace(/&omega;/g, 'ω');
  95 + str = str.replace(/&thetasym;/g, 'ϑ');
  96 + str = str.replace(/&upsih;/g, 'ϒ');
  97 + str = str.replace(/&piv;/g, 'ϖ');
  98 + str = str.replace(/&middot;/g, '·');
  99 + return str;
  100 +}
  101 +
  102 +//
  103 +
  104 +function strcharacterDiscode(str){
  105 + // 加入常用解析
  106 + str = str.replace(/&nbsp;/g, ' ');
  107 + str = str.replace(/&quot;/g, "'");
  108 + str = str.replace(/&amp;/g, '&');
  109 + // str = str.replace(/&lt;/g, '‹');
  110 + // str = str.replace(/&gt;/g, '›');
  111 +
  112 + str = str.replace(/&lt;/g, '<');
  113 + str = str.replace(/&gt;/g, '>');
  114 + str = str.replace(/&#8226;/g, '•');
  115 +
  116 + return str;
  117 +}
  118 +
  119 +// HTML 支持的其他实体
  120 +function strOtherDiscode(str){
  121 + str = str.replace(/&OElig;/g, 'Œ');
  122 + str = str.replace(/&oelig;/g, 'œ');
  123 + str = str.replace(/&Scaron;/g, 'Š');
  124 + str = str.replace(/&scaron;/g, 'š');
  125 + str = str.replace(/&Yuml;/g, 'Ÿ');
  126 + str = str.replace(/&fnof;/g, 'ƒ');
  127 + str = str.replace(/&circ;/g, 'ˆ');
  128 + str = str.replace(/&tilde;/g, '˜');
  129 + str = str.replace(/&ensp;/g, '');
  130 + str = str.replace(/&emsp;/g, '');
  131 + str = str.replace(/&thinsp;/g, '');
  132 + str = str.replace(/&zwnj;/g, '');
  133 + str = str.replace(/&zwj;/g, '');
  134 + str = str.replace(/&lrm;/g, '');
  135 + str = str.replace(/&rlm;/g, '');
  136 + str = str.replace(/&ndash;/g, '–');
  137 + str = str.replace(/&mdash;/g, '—');
  138 + str = str.replace(/&lsquo;/g, '‘');
  139 + str = str.replace(/&rsquo;/g, '’');
  140 + str = str.replace(/&sbquo;/g, '‚');
  141 + str = str.replace(/&ldquo;/g, '“');
  142 + str = str.replace(/&rdquo;/g, '”');
  143 + str = str.replace(/&bdquo;/g, '„');
  144 + str = str.replace(/&dagger;/g, '†');
  145 + str = str.replace(/&Dagger;/g, '‡');
  146 + str = str.replace(/&bull;/g, '•');
  147 + str = str.replace(/&hellip;/g, '…');
  148 + str = str.replace(/&permil;/g, '‰');
  149 + str = str.replace(/&prime;/g, '′');
  150 + str = str.replace(/&Prime;/g, '″');
  151 + str = str.replace(/&lsaquo;/g, '‹');
  152 + str = str.replace(/&rsaquo;/g, '›');
  153 + str = str.replace(/&oline;/g, '‾');
  154 + str = str.replace(/&euro;/g, '€');
  155 + str = str.replace(/&trade;/g, '™');
  156 +
  157 + str = str.replace(/&larr;/g, '←');
  158 + str = str.replace(/&uarr;/g, '↑');
  159 + str = str.replace(/&rarr;/g, '→');
  160 + str = str.replace(/&darr;/g, '↓');
  161 + str = str.replace(/&harr;/g, '↔');
  162 + str = str.replace(/&crarr;/g, '↵');
  163 + str = str.replace(/&lceil;/g, '⌈');
  164 + str = str.replace(/&rceil;/g, '⌉');
  165 +
  166 + str = str.replace(/&lfloor;/g, '⌊');
  167 + str = str.replace(/&rfloor;/g, '⌋');
  168 + str = str.replace(/&loz;/g, '◊');
  169 + str = str.replace(/&spades;/g, '♠');
  170 + str = str.replace(/&clubs;/g, '♣');
  171 + str = str.replace(/&hearts;/g, '♥');
  172 +
  173 + str = str.replace(/&diams;/g, '♦');
  174 + str = str.replace(/&#39;/g, '\'');
  175 + return str;
  176 +}
  177 +
  178 +function strMoreDiscode(str){
  179 + str = str.replace(/\r\n/g,"");
  180 + str = str.replace(/\n/g,"");
  181 +
  182 + str = str.replace(/code/g,"wxxxcode-style");
  183 + return str;
  184 +}
  185 +
  186 +function strDiscode(str){
  187 + str = strNumDiscode(str);
  188 + str = strGreeceDiscode(str);
  189 + str = strcharacterDiscode(str);
  190 + str = strOtherDiscode(str);
  191 + str = strMoreDiscode(str);
  192 + return str;
  193 +}
  194 +function urlToHttpUrl(url,rep){
  195 +
  196 + var patt1 = new RegExp("^//");
  197 + var result = patt1.test(url);
  198 + if(result){
  199 + url = rep+":"+url;
  200 + }
  201 + return url;
  202 +}
  203 +
  204 +module.exports = {
  205 + strDiscode:strDiscode,
  206 + urlToHttpUrl:urlToHttpUrl
  207 +}
  1 +/**
  2 + * author: Di (微信小程序开发工程师)
  3 + * organization: WeAppDev(微信小程序开发论坛)(http://weappdev.com)
  4 + * 垂直微信小程序开发交流社区
  5 + *
  6 + * github地址: https://github.com/icindy/wxParse
  7 + *
  8 + * for: 微信小程序富文本解析
  9 + * detail : http://weappdev.com/t/wxparse-alpha0-1-html-markdown/184
  10 + */
  11 +
  12 +/**
  13 + * utils函数引入
  14 + **/
  15 +import showdown from './showdown.js';
  16 +import HtmlToJson from './html2json.js';
  17 +/**
  18 + * 配置及公有属性
  19 + **/
  20 +var realWindowWidth = 0;
  21 +var realWindowHeight = 0;
  22 +wx.getSystemInfo({
  23 + success: function (res) {
  24 + realWindowWidth = res.windowWidth
  25 + realWindowHeight = res.windowHeight
  26 + }
  27 +})
  28 +/**
  29 + * 主函数入口区
  30 + **/
  31 +function wxParse(bindName = 'wxParseData', type='html', data='<div class="color:red;">数据不能为空</div>', target,imagePadding) {
  32 + var that = target;
  33 + var transData = {};//存放转化后的数据
  34 + if (type == 'html') {
  35 + transData = HtmlToJson.html2json(data, bindName);
  36 + // console.log(JSON.stringify(transData, ' ', ' '));
  37 + } else if (type == 'md' || type == 'markdown') {
  38 + var converter = new showdown.Converter();
  39 + var html = converter.makeHtml(data);
  40 + transData = HtmlToJson.html2json(html, bindName);
  41 + console.log(JSON.stringify(transData, ' ', ' '));
  42 + }
  43 + transData.view = {};
  44 + transData.view.imagePadding = 0;
  45 + if(typeof(imagePadding) != 'undefined'){
  46 + transData.view.imagePadding = imagePadding
  47 + }
  48 + var bindData = {};
  49 + bindData[bindName] = transData;
  50 + that.setData(bindData)
  51 + that.wxParseImgLoad = wxParseImgLoad;
  52 + that.wxParseImgTap = wxParseImgTap;
  53 +}
  54 +// 图片点击事件
  55 +function wxParseImgTap(e) {
  56 + var that = this;
  57 + var nowImgUrl = e.target.dataset.src;
  58 + var tagFrom = e.target.dataset.from;
  59 + if (typeof (tagFrom) != 'undefined' && tagFrom.length > 0) {
  60 + wx.previewImage({
  61 + current: nowImgUrl, // 当前显示图片的http链接
  62 + urls: that.data[tagFrom].imageUrls // 需要预览的图片http链接列表
  63 + })
  64 + }
  65 +}
  66 +
  67 +/**
  68 + * 图片视觉宽高计算函数区
  69 + **/
  70 +function wxParseImgLoad(e) {
  71 + var that = this;
  72 + var tagFrom = e.target.dataset.from;
  73 + var idx = e.target.dataset.idx;
  74 + if (typeof (tagFrom) != 'undefined' && tagFrom.length > 0) {
  75 + calMoreImageInfo(e, idx, that, tagFrom)
  76 + }
  77 +}
  78 +// 假循环获取计算图片视觉最佳宽高
  79 +function calMoreImageInfo(e, idx, that, bindName) {
  80 + var temData = that.data[bindName];
  81 + if (!temData || temData.images.length == 0) {
  82 + return;
  83 + }
  84 + var temImages = temData.images;
  85 + //因为无法获取view宽度 需要自定义padding进行计算,稍后处理
  86 + var recal = wxAutoImageCal(e.detail.width, e.detail.height,that,bindName);
  87 + // temImages[idx].width = recal.imageWidth;
  88 + // temImages[idx].height = recal.imageheight;
  89 + // temData.images = temImages;
  90 + // var bindData = {};
  91 + // bindData[bindName] = temData;
  92 + // that.setData(bindData);
  93 + var index = temImages[idx].index
  94 + var key = `${bindName}`
  95 + for (var i of index.split('.')) key+=`.nodes[${i}]`
  96 + var keyW = key + '.width'
  97 + var keyH = key + '.height'
  98 + that.setData({
  99 + [keyW]: recal.imageWidth,
  100 + [keyH]: recal.imageheight,
  101 + })
  102 +}
  103 +
  104 +// 计算视觉优先的图片宽高
  105 +function wxAutoImageCal(originalWidth, originalHeight,that,bindName) {
  106 + //获取图片的原始长宽
  107 + var windowWidth = 0, windowHeight = 0;
  108 + var autoWidth = 0, autoHeight = 0;
  109 + var results = {};
  110 + var padding = that.data[bindName].view.imagePadding;
  111 + windowWidth = realWindowWidth-2*padding;
  112 + windowHeight = realWindowHeight;
  113 + //判断按照那种方式进行缩放
  114 + // console.log("windowWidth" + windowWidth);
  115 + if (originalWidth > windowWidth) {//在图片width大于手机屏幕width时候
  116 + autoWidth = windowWidth;
  117 + // console.log("autoWidth" + autoWidth);
  118 + autoHeight = (autoWidth * originalHeight) / originalWidth;
  119 + // console.log("autoHeight" + autoHeight);
  120 + results.imageWidth = autoWidth;
  121 + results.imageheight = autoHeight;
  122 + } else {//否则展示原来的数据
  123 + results.imageWidth = originalWidth;
  124 + results.imageheight = originalHeight;
  125 + }
  126 + return results;
  127 +}
  128 +
  129 +function wxParseTemArray(temArrayName,bindNameReg,total,that){
  130 + var array = [];
  131 + var temData = that.data;
  132 + var obj = null;
  133 + for(var i = 0; i < total; i++){
  134 + var simArr = temData[bindNameReg+i].nodes;
  135 + array.push(simArr);
  136 + }
  137 +
  138 + temArrayName = temArrayName || 'wxParseTemArray';
  139 + obj = JSON.parse('{"'+ temArrayName +'":""}');
  140 + obj[temArrayName] = array;
  141 + that.setData(obj);
  142 +}
  143 +
  144 +/**
  145 + * 配置emojis
  146 + *
  147 + */
  148 +
  149 +function emojisInit(reg='',baseSrc="/wxParse/emojis/",emojis){
  150 + HtmlToJson.emojisInit(reg,baseSrc,emojis);
  151 +}
  152 +
  153 +module.exports = {
  154 + wxParse: wxParse,
  155 + wxParseTemArray:wxParseTemArray,
  156 + emojisInit:emojisInit
  157 +}
  158 +
  159 +
  1 +<!--**
  2 + * author: Di (微信小程序开发工程师)
  3 + * organization: WeAppDev(微信小程序开发论坛)(http://weappdev.com)
  4 + * 垂直微信小程序开发交流社区
  5 + *
  6 + * github地址: https://github.com/icindy/wxParse
  7 + *
  8 + * for: 微信小程序富文本解析
  9 + * detail : http://weappdev.com/t/wxparse-alpha0-1-html-markdown/184
  10 + */-->
  11 +
  12 +<!--基础元素-->
  13 +<template name="wxParseVideo">
  14 + <!--增加video标签支持,并循环添加-->
  15 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  16 + <video class="{{item.classStr}} wxParse-{{item.tag}}-video" src="{{item.attr.src}}"></video>
  17 + </view>
  18 +</template>
  19 +
  20 +<template name="wxParseImg">
  21 + <image class="{{item.classStr}} wxParse-{{item.tag}}" data-from="{{item.from}}" data-src="{{item.attr.src}}" data-idx="{{item.imgIndex}}" src="{{item.attr.src}}" mode="aspectFit" bindload="wxParseImgLoad" bindtap="wxParseImgTap" mode="widthFix" style="width:{{item.width}}px;"
  22 + />
  23 +</template>
  24 +
  25 +<template name="WxEmojiView">
  26 + <view class="WxEmojiView wxParse-inline" style="{{item.styleStr}}">
  27 + <block wx:for="{{item.textArray}}" wx:key="">
  28 + <block class="{{item.text == '\\n' ? 'wxParse-hide':''}}" wx:if="{{item.node == 'text'}}">{{item.text}}</block>
  29 + <block wx:elif="{{item.node == 'element'}}">
  30 + <image class="wxEmoji" src="{{item.baseSrc}}{{item.text}}" />
  31 + </block>
  32 + </block>
  33 + </view>
  34 +</template>
  35 +
  36 +<template name="WxParseBr">
  37 + <text>\n</text>
  38 +</template>
  39 +<!--入口模版-->
  40 +
  41 +<template name="wxParse">
  42 + <block wx:for="{{wxParseData}}" wx:key="">
  43 + <template is="wxParse0" data="{{item}}" />
  44 + </block>
  45 +</template>
  46 +
  47 +
  48 +<!--循环模版-->
  49 +<template name="wxParse0">
  50 + <!--<template is="wxParse1" data="{{item}}" />-->
  51 + <!--判断是否是标签节点-->
  52 + <block wx:if="{{item.node == 'element'}}">
  53 + <block wx:if="{{item.tag == 'button'}}">
  54 + <button type="default" size="mini">
  55 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  56 + <template is="wxParse1" data="{{item}}" />
  57 + </block>
  58 + </button>
  59 + </block>
  60 + <!--li类型-->
  61 + <block wx:elif="{{item.tag == 'li'}}">
  62 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  63 + <view class="{{item.classStr}} wxParse-li-inner">
  64 + <view class="{{item.classStr}} wxParse-li-text">
  65 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  66 + </view>
  67 + <view class="{{item.classStr}} wxParse-li-text">
  68 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  69 + <template is="wxParse1" data="{{item}}" />
  70 + </block>
  71 + </view>
  72 + </view>
  73 + </view>
  74 + </block>
  75 +
  76 + <!--video类型-->
  77 + <block wx:elif="{{item.tag == 'video'}}">
  78 + <template is="wxParseVideo" data="{{item}}" />
  79 + </block>
  80 +
  81 + <!--img类型-->
  82 + <block wx:elif="{{item.tag == 'img'}}">
  83 + <template is="wxParseImg" data="{{item}}" />
  84 + </block>
  85 +
  86 + <!--a类型-->
  87 + <block wx:elif="{{item.tag == 'a'}}">
  88 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  89 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  90 + <template is="wxParse1" data="{{item}}" />
  91 + </block>
  92 + </view>
  93 + </block>
  94 + <block wx:elif="{{item.tag == 'table'}}">
  95 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  96 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  97 + <template is="wxParse1" data="{{item}}" />
  98 + </block>
  99 + </view>
  100 + </block>
  101 +
  102 + <block wx:elif="{{item.tag == 'br'}}">
  103 + <template is="WxParseBr"></template>
  104 + </block>
  105 + <!--其他块级标签-->
  106 + <block wx:elif="{{item.tagType == 'block'}}">
  107 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  108 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  109 + <template is="wxParse1" data="{{item}}" />
  110 + </block>
  111 + </view>
  112 + </block>
  113 +
  114 + <!--内联标签-->
  115 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  116 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  117 + <template is="wxParse1" data="{{item}}" />
  118 + </block>
  119 + </view>
  120 +
  121 + </block>
  122 +
  123 + <!--判断是否是文本节点-->
  124 + <block wx:elif="{{item.node == 'text'}}">
  125 + <!--如果是,直接进行-->
  126 + <template is="WxEmojiView" data="{{item}}" />
  127 + </block>
  128 +
  129 +</template>
  130 +
  131 +
  132 +
  133 +<!--循环模版-->
  134 +<template name="wxParse1">
  135 + <!--<template is="wxParse2" data="{{item}}" />-->
  136 + <!--判断是否是标签节点-->
  137 + <block wx:if="{{item.node == 'element'}}">
  138 + <block wx:if="{{item.tag == 'button'}}">
  139 + <button type="default" size="mini">
  140 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  141 + <template is="wxParse2" data="{{item}}" />
  142 + </block>
  143 + </button>
  144 + </block>
  145 + <!--li类型-->
  146 + <block wx:elif="{{item.tag == 'li'}}">
  147 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  148 + <view class="{{item.classStr}} wxParse-li-inner">
  149 + <view class="{{item.classStr}} wxParse-li-text">
  150 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  151 + </view>
  152 + <view class="{{item.classStr}} wxParse-li-text">
  153 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  154 + <template is="wxParse2" data="{{item}}" />
  155 + </block>
  156 + </view>
  157 + </view>
  158 + </view>
  159 + </block>
  160 +
  161 + <!--video类型-->
  162 + <block wx:elif="{{item.tag == 'video'}}">
  163 + <template is="wxParseVideo" data="{{item}}" />
  164 + </block>
  165 +
  166 + <!--img类型-->
  167 + <block wx:elif="{{item.tag == 'img'}}">
  168 + <template is="wxParseImg" data="{{item}}" />
  169 + </block>
  170 +
  171 + <!--a类型-->
  172 + <block wx:elif="{{item.tag == 'a'}}">
  173 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  174 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  175 + <template is="wxParse2" data="{{item}}" />
  176 + </block>
  177 + </view>
  178 + </block>
  179 +
  180 + <block wx:elif="{{item.tag == 'br'}}">
  181 + <template is="WxParseBr"></template>
  182 + </block>
  183 + <!--其他块级标签-->
  184 + <block wx:elif="{{item.tagType == 'block'}}">
  185 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  186 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  187 + <template is="wxParse2" data="{{item}}" />
  188 + </block>
  189 + </view>
  190 + </block>
  191 +
  192 + <!--内联标签-->
  193 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  194 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  195 + <template is="wxParse2" data="{{item}}" />
  196 + </block>
  197 + </view>
  198 +
  199 + </block>
  200 +
  201 + <!--判断是否是文本节点-->
  202 + <block wx:elif="{{item.node == 'text'}}">
  203 + <!--如果是,直接进行-->
  204 + <template is="WxEmojiView" data="{{item}}" />
  205 + </block>
  206 +
  207 +</template>
  208 +
  209 +
  210 +<!--循环模版-->
  211 +<template name="wxParse2">
  212 + <!--<template is="wxParse3" data="{{item}}" />-->
  213 + <!--判断是否是标签节点-->
  214 + <block wx:if="{{item.node == 'element'}}">
  215 + <block wx:if="{{item.tag == 'button'}}">
  216 + <button type="default" size="mini">
  217 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  218 + <template is="wxParse3" data="{{item}}" />
  219 + </block>
  220 + </button>
  221 + </block>
  222 + <!--li类型-->
  223 + <block wx:elif="{{item.tag == 'li'}}">
  224 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  225 + <view class="{{item.classStr}} wxParse-li-inner">
  226 + <view class="{{item.classStr}} wxParse-li-text">
  227 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  228 + </view>
  229 + <view class="{{item.classStr}} wxParse-li-text">
  230 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  231 + <template is="wxParse3" data="{{item}}" />
  232 + </block>
  233 + </view>
  234 + </view>
  235 + </view>
  236 + </block>
  237 +
  238 + <!--video类型-->
  239 + <block wx:elif="{{item.tag == 'video'}}">
  240 + <template is="wxParseVideo" data="{{item}}" />
  241 + </block>
  242 +
  243 + <!--img类型-->
  244 + <block wx:elif="{{item.tag == 'img'}}">
  245 + <template is="wxParseImg" data="{{item}}" />
  246 + </block>
  247 +
  248 + <!--a类型-->
  249 + <block wx:elif="{{item.tag == 'a'}}">
  250 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  251 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  252 + <template is="wxParse3" data="{{item}}" />
  253 + </block>
  254 + </view>
  255 + </block>
  256 +
  257 + <block wx:elif="{{item.tag == 'br'}}">
  258 + <template is="WxParseBr"></template>
  259 + </block>
  260 + <!--其他块级标签-->
  261 + <block wx:elif="{{item.tagType == 'block'}}">
  262 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  263 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  264 + <template is="wxParse3" data="{{item}}" />
  265 + </block>
  266 + </view>
  267 + </block>
  268 +
  269 + <!--内联标签-->
  270 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  271 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  272 + <template is="wxParse3" data="{{item}}" />
  273 + </block>
  274 + </view>
  275 +
  276 + </block>
  277 +
  278 + <!--判断是否是文本节点-->
  279 + <block wx:elif="{{item.node == 'text'}}">
  280 + <!--如果是,直接进行-->
  281 + <template is="WxEmojiView" data="{{item}}" />
  282 + </block>
  283 +
  284 +</template>
  285 +
  286 +<!--循环模版-->
  287 +<template name="wxParse3">
  288 + <!--<template is="wxParse4" data="{{item}}" />-->
  289 + <!--判断是否是标签节点-->
  290 + <block wx:if="{{item.node == 'element'}}">
  291 + <block wx:if="{{item.tag == 'button'}}">
  292 + <button type="default" size="mini">
  293 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  294 + <template is="wxParse4" data="{{item}}" />
  295 + </block>
  296 + </button>
  297 + </block>
  298 + <!--li类型-->
  299 + <block wx:elif="{{item.tag == 'li'}}">
  300 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  301 + <view class="{{item.classStr}} wxParse-li-inner">
  302 + <view class="{{item.classStr}} wxParse-li-text">
  303 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  304 + </view>
  305 + <view class="{{item.classStr}} wxParse-li-text">
  306 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  307 + <template is="wxParse4" data="{{item}}" />
  308 + </block>
  309 + </view>
  310 + </view>
  311 + </view>
  312 + </block>
  313 +
  314 + <!--video类型-->
  315 + <block wx:elif="{{item.tag == 'video'}}">
  316 + <template is="wxParseVideo" data="{{item}}" />
  317 + </block>
  318 +
  319 + <!--img类型-->
  320 + <block wx:elif="{{item.tag == 'img'}}">
  321 + <template is="wxParseImg" data="{{item}}" />
  322 + </block>
  323 +
  324 + <!--a类型-->
  325 + <block wx:elif="{{item.tag == 'a'}}">
  326 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  327 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  328 + <template is="wxParse4" data="{{item}}" />
  329 + </block>
  330 + </view>
  331 + </block>
  332 +
  333 + <block wx:elif="{{item.tag == 'br'}}">
  334 + <template is="WxParseBr"></template>
  335 + </block>
  336 + <!--其他块级标签-->
  337 + <block wx:elif="{{item.tagType == 'block'}}">
  338 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  339 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  340 + <template is="wxParse4" data="{{item}}" />
  341 + </block>
  342 + </view>
  343 + </block>
  344 +
  345 + <!--内联标签-->
  346 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  347 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  348 + <template is="wxParse4" data="{{item}}" />
  349 + </block>
  350 + </view>
  351 +
  352 + </block>
  353 +
  354 + <!--判断是否是文本节点-->
  355 + <block wx:elif="{{item.node == 'text'}}">
  356 + <!--如果是,直接进行-->
  357 + <template is="WxEmojiView" data="{{item}}" />
  358 + </block>
  359 +
  360 +</template>
  361 +
  362 +<!--循环模版-->
  363 +<template name="wxParse4">
  364 + <!--<template is="wxParse5" data="{{item}}" />-->
  365 + <!--判断是否是标签节点-->
  366 + <block wx:if="{{item.node == 'element'}}">
  367 + <block wx:if="{{item.tag == 'button'}}">
  368 + <button type="default" size="mini">
  369 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  370 + <template is="wxParse5" data="{{item}}" />
  371 + </block>
  372 + </button>
  373 + </block>
  374 + <!--li类型-->
  375 + <block wx:elif="{{item.tag == 'li'}}">
  376 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  377 + <view class="{{item.classStr}} wxParse-li-inner">
  378 + <view class="{{item.classStr}} wxParse-li-text">
  379 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  380 + </view>
  381 + <view class="{{item.classStr}} wxParse-li-text">
  382 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  383 + <template is="wxParse5" data="{{item}}" />
  384 + </block>
  385 + </view>
  386 + </view>
  387 + </view>
  388 + </block>
  389 +
  390 + <!--video类型-->
  391 + <block wx:elif="{{item.tag == 'video'}}">
  392 + <template is="wxParseVideo" data="{{item}}" />
  393 + </block>
  394 +
  395 + <!--img类型-->
  396 + <block wx:elif="{{item.tag == 'img'}}">
  397 + <template is="wxParseImg" data="{{item}}" />
  398 + </block>
  399 +
  400 + <!--a类型-->
  401 + <block wx:elif="{{item.tag == 'a'}}">
  402 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  403 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  404 + <template is="wxParse5" data="{{item}}" />
  405 + </block>
  406 + </view>
  407 + </block>
  408 +
  409 + <block wx:elif="{{item.tag == 'br'}}">
  410 + <template is="WxParseBr"></template>
  411 + </block>
  412 + <!--其他块级标签-->
  413 + <block wx:elif="{{item.tagType == 'block'}}">
  414 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  415 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  416 + <template is="wxParse5" data="{{item}}" />
  417 + </block>
  418 + </view>
  419 + </block>
  420 +
  421 + <!--内联标签-->
  422 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  423 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  424 + <template is="wxParse5" data="{{item}}" />
  425 + </block>
  426 + </view>
  427 +
  428 + </block>
  429 +
  430 + <!--判断是否是文本节点-->
  431 + <block wx:elif="{{item.node == 'text'}}">
  432 + <!--如果是,直接进行-->
  433 + <template is="WxEmojiView" data="{{item}}" />
  434 + </block>
  435 +
  436 +</template>
  437 +
  438 +<!--循环模版-->
  439 +<template name="wxParse5">
  440 + <!--<template is="wxParse6" data="{{item}}" />-->
  441 + <!--判断是否是标签节点-->
  442 + <block wx:if="{{item.node == 'element'}}">
  443 + <block wx:if="{{item.tag == 'button'}}">
  444 + <button type="default" size="mini">
  445 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  446 + <template is="wxParse6" data="{{item}}" />
  447 + </block>
  448 + </button>
  449 + </block>
  450 + <!--li类型-->
  451 + <block wx:elif="{{item.tag == 'li'}}">
  452 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  453 + <view class="{{item.classStr}} wxParse-li-inner">
  454 + <view class="{{item.classStr}} wxParse-li-text">
  455 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  456 + </view>
  457 + <view class="{{item.classStr}} wxParse-li-text">
  458 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  459 + <template is="wxParse6" data="{{item}}" />
  460 + </block>
  461 + </view>
  462 + </view>
  463 + </view>
  464 + </block>
  465 +
  466 + <!--video类型-->
  467 + <block wx:elif="{{item.tag == 'video'}}">
  468 + <template is="wxParseVideo" data="{{item}}" />
  469 + </block>
  470 +
  471 + <!--img类型-->
  472 + <block wx:elif="{{item.tag == 'img'}}">
  473 + <template is="wxParseImg" data="{{item}}" />
  474 + </block>
  475 +
  476 + <!--a类型-->
  477 + <block wx:elif="{{item.tag == 'a'}}">
  478 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  479 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  480 + <template is="wxParse6" data="{{item}}" />
  481 + </block>
  482 + </view>
  483 + </block>
  484 +
  485 + <block wx:elif="{{item.tag == 'br'}}">
  486 + <template is="WxParseBr"></template>
  487 + </block>
  488 + <!--其他块级标签-->
  489 + <block wx:elif="{{item.tagType == 'block'}}">
  490 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  491 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  492 + <template is="wxParse6" data="{{item}}" />
  493 + </block>
  494 + </view>
  495 + </block>
  496 +
  497 + <!--内联标签-->
  498 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  499 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  500 + <template is="wxParse6" data="{{item}}" />
  501 + </block>
  502 + </view>
  503 +
  504 + </block>
  505 +
  506 + <!--判断是否是文本节点-->
  507 + <block wx:elif="{{item.node == 'text'}}">
  508 + <!--如果是,直接进行-->
  509 + <template is="WxEmojiView" data="{{item}}" />
  510 + </block>
  511 +
  512 +</template>
  513 +
  514 +<!--循环模版-->
  515 +<template name="wxParse6">
  516 + <!--<template is="wxParse7" data="{{item}}" />-->
  517 + <!--判断是否是标签节点-->
  518 + <block wx:if="{{item.node == 'element'}}">
  519 + <block wx:if="{{item.tag == 'button'}}">
  520 + <button type="default" size="mini">
  521 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  522 + <template is="wxParse7" data="{{item}}" />
  523 + </block>
  524 + </button>
  525 + </block>
  526 + <!--li类型-->
  527 + <block wx:elif="{{item.tag == 'li'}}">
  528 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  529 + <view class="{{item.classStr}} wxParse-li-inner">
  530 + <view class="{{item.classStr}} wxParse-li-text">
  531 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  532 + </view>
  533 + <view class="{{item.classStr}} wxParse-li-text">
  534 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  535 + <template is="wxParse7" data="{{item}}" />
  536 + </block>
  537 + </view>
  538 + </view>
  539 + </view>
  540 + </block>
  541 +
  542 + <!--video类型-->
  543 + <block wx:elif="{{item.tag == 'video'}}">
  544 + <template is="wxParseVideo" data="{{item}}" />
  545 + </block>
  546 +
  547 + <!--img类型-->
  548 + <block wx:elif="{{item.tag == 'img'}}">
  549 + <template is="wxParseImg" data="{{item}}" />
  550 + </block>
  551 +
  552 + <!--a类型-->
  553 + <block wx:elif="{{item.tag == 'a'}}">
  554 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  555 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  556 + <template is="wxParse7" data="{{item}}" />
  557 + </block>
  558 + </view>
  559 + </block>
  560 +
  561 + <block wx:elif="{{item.tag == 'br'}}">
  562 + <template is="WxParseBr"></template>
  563 + </block>
  564 + <!--其他块级标签-->
  565 + <block wx:elif="{{item.tagType == 'block'}}">
  566 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  567 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  568 + <template is="wxParse7" data="{{item}}" />
  569 + </block>
  570 + </view>
  571 + </block>
  572 +
  573 + <!--内联标签-->
  574 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  575 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  576 + <template is="wxParse7" data="{{item}}" />
  577 + </block>
  578 + </view>
  579 +
  580 + </block>
  581 +
  582 + <!--判断是否是文本节点-->
  583 + <block wx:elif="{{item.node == 'text'}}">
  584 + <!--如果是,直接进行-->
  585 + <template is="WxEmojiView" data="{{item}}" />
  586 + </block>
  587 +
  588 +</template>
  589 +<!--循环模版-->
  590 +<template name="wxParse7">
  591 + <!--<template is="wxParse8" data="{{item}}" />-->
  592 + <!--判断是否是标签节点-->
  593 + <block wx:if="{{item.node == 'element'}}">
  594 + <block wx:if="{{item.tag == 'button'}}">
  595 + <button type="default" size="mini">
  596 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  597 + <template is="wxParse8" data="{{item}}" />
  598 + </block>
  599 + </button>
  600 + </block>
  601 + <!--li类型-->
  602 + <block wx:elif="{{item.tag == 'li'}}">
  603 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  604 + <view class="{{item.classStr}} wxParse-li-inner">
  605 + <view class="{{item.classStr}} wxParse-li-text">
  606 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  607 + </view>
  608 + <view class="{{item.classStr}} wxParse-li-text">
  609 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  610 + <template is="wxParse8" data="{{item}}" />
  611 + </block>
  612 + </view>
  613 + </view>
  614 + </view>
  615 + </block>
  616 +
  617 + <!--video类型-->
  618 + <block wx:elif="{{item.tag == 'video'}}">
  619 + <template is="wxParseVideo" data="{{item}}" />
  620 + </block>
  621 +
  622 + <!--img类型-->
  623 + <block wx:elif="{{item.tag == 'img'}}">
  624 + <template is="wxParseImg" data="{{item}}" />
  625 + </block>
  626 +
  627 + <!--a类型-->
  628 + <block wx:elif="{{item.tag == 'a'}}">
  629 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  630 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  631 + <template is="wxParse8" data="{{item}}" />
  632 + </block>
  633 + </view>
  634 + </block>
  635 +
  636 + <block wx:elif="{{item.tag == 'br'}}">
  637 + <template is="WxParseBr"></template>
  638 + </block>
  639 + <!--其他块级标签-->
  640 + <block wx:elif="{{item.tagType == 'block'}}">
  641 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  642 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  643 + <template is="wxParse8" data="{{item}}" />
  644 + </block>
  645 + </view>
  646 + </block>
  647 +
  648 + <!--内联标签-->
  649 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  650 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  651 + <template is="wxParse8" data="{{item}}" />
  652 + </block>
  653 + </view>
  654 +
  655 + </block>
  656 +
  657 + <!--判断是否是文本节点-->
  658 + <block wx:elif="{{item.node == 'text'}}">
  659 + <!--如果是,直接进行-->
  660 + <template is="WxEmojiView" data="{{item}}" />
  661 + </block>
  662 +
  663 +</template>
  664 +
  665 +<!--循环模版-->
  666 +<template name="wxParse8">
  667 + <!--<template is="wxParse9" data="{{item}}" />-->
  668 + <!--判断是否是标签节点-->
  669 + <block wx:if="{{item.node == 'element'}}">
  670 + <block wx:if="{{item.tag == 'button'}}">
  671 + <button type="default" size="mini">
  672 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  673 + <template is="wxParse9" data="{{item}}" />
  674 + </block>
  675 + </button>
  676 + </block>
  677 + <!--li类型-->
  678 + <block wx:elif="{{item.tag == 'li'}}">
  679 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  680 + <view class="{{item.classStr}} wxParse-li-inner">
  681 + <view class="{{item.classStr}} wxParse-li-text">
  682 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  683 + </view>
  684 + <view class="{{item.classStr}} wxParse-li-text">
  685 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  686 + <template is="wxParse9" data="{{item}}" />
  687 + </block>
  688 + </view>
  689 + </view>
  690 + </view>
  691 + </block>
  692 +
  693 + <!--video类型-->
  694 + <block wx:elif="{{item.tag == 'video'}}">
  695 + <template is="wxParseVideo" data="{{item}}" />
  696 + </block>
  697 +
  698 + <!--img类型-->
  699 + <block wx:elif="{{item.tag == 'img'}}">
  700 + <template is="wxParseImg" data="{{item}}" />
  701 + </block>
  702 +
  703 + <!--a类型-->
  704 + <block wx:elif="{{item.tag == 'a'}}">
  705 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  706 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  707 + <template is="wxParse9" data="{{item}}" />
  708 + </block>
  709 + </view>
  710 + </block>
  711 +
  712 + <block wx:elif="{{item.tag == 'br'}}">
  713 + <template is="WxParseBr"></template>
  714 + </block>
  715 + <!--其他块级标签-->
  716 + <block wx:elif="{{item.tagType == 'block'}}">
  717 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  718 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  719 + <template is="wxParse9" data="{{item}}" />
  720 + </block>
  721 + </view>
  722 + </block>
  723 +
  724 + <!--内联标签-->
  725 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  726 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  727 + <template is="wxParse9" data="{{item}}" />
  728 + </block>
  729 + </view>
  730 +
  731 + </block>
  732 +
  733 + <!--判断是否是文本节点-->
  734 + <block wx:elif="{{item.node == 'text'}}">
  735 + <!--如果是,直接进行-->
  736 + <template is="WxEmojiView" data="{{item}}" />
  737 + </block>
  738 +
  739 +</template>
  740 +
  741 +<!--循环模版-->
  742 +<template name="wxParse9">
  743 + <!--<template is="wxParse10" data="{{item}}" />-->
  744 + <!--判断是否是标签节点-->
  745 + <block wx:if="{{item.node == 'element'}}">
  746 + <block wx:if="{{item.tag == 'button'}}">
  747 + <button type="default" size="mini">
  748 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  749 + <template is="wxParse10" data="{{item}}" />
  750 + </block>
  751 + </button>
  752 + </block>
  753 + <!--li类型-->
  754 + <block wx:elif="{{item.tag == 'li'}}">
  755 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  756 + <view class="{{item.classStr}} wxParse-li-inner">
  757 + <view class="{{item.classStr}} wxParse-li-text">
  758 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  759 + </view>
  760 + <view class="{{item.classStr}} wxParse-li-text">
  761 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  762 + <template is="wxParse10" data="{{item}}" />
  763 + </block>
  764 + </view>
  765 + </view>
  766 + </view>
  767 + </block>
  768 +
  769 + <!--video类型-->
  770 + <block wx:elif="{{item.tag == 'video'}}">
  771 + <template is="wxParseVideo" data="{{item}}" />
  772 + </block>
  773 +
  774 + <!--img类型-->
  775 + <block wx:elif="{{item.tag == 'img'}}">
  776 + <template is="wxParseImg" data="{{item}}" />
  777 + </block>
  778 +
  779 + <!--a类型-->
  780 + <block wx:elif="{{item.tag == 'a'}}">
  781 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  782 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  783 + <template is="wxParse10" data="{{item}}" />
  784 + </block>
  785 + </view>
  786 + </block>
  787 +
  788 + <block wx:elif="{{item.tag == 'br'}}">
  789 + <template is="WxParseBr"></template>
  790 + </block>
  791 + <!--其他块级标签-->
  792 + <block wx:elif="{{item.tagType == 'block'}}">
  793 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  794 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  795 + <template is="wxParse10" data="{{item}}" />
  796 + </block>
  797 + </view>
  798 + </block>
  799 +
  800 + <!--内联标签-->
  801 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  802 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  803 + <template is="wxParse10" data="{{item}}" />
  804 + </block>
  805 + </view>
  806 +
  807 + </block>
  808 +
  809 + <!--判断是否是文本节点-->
  810 + <block wx:elif="{{item.node == 'text'}}">
  811 + <!--如果是,直接进行-->
  812 + <template is="WxEmojiView" data="{{item}}" />
  813 + </block>
  814 +
  815 +</template>
  816 +
  817 +<!--循环模版-->
  818 +<template name="wxParse10">
  819 + <!--<template is="wxParse11" data="{{item}}" />-->
  820 + <!--判断是否是标签节点-->
  821 + <block wx:if="{{item.node == 'element'}}">
  822 + <block wx:if="{{item.tag == 'button'}}">
  823 + <button type="default" size="mini">
  824 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  825 + <template is="wxParse11" data="{{item}}" />
  826 + </block>
  827 + </button>
  828 + </block>
  829 + <!--li类型-->
  830 + <block wx:elif="{{item.tag == 'li'}}">
  831 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  832 + <view class="{{item.classStr}} wxParse-li-inner">
  833 + <view class="{{item.classStr}} wxParse-li-text">
  834 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  835 + </view>
  836 + <view class="{{item.classStr}} wxParse-li-text">
  837 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  838 + <template is="wxParse11" data="{{item}}" />
  839 + </block>
  840 + </view>
  841 + </view>
  842 + </view>
  843 + </block>
  844 +
  845 + <!--video类型-->
  846 + <block wx:elif="{{item.tag == 'video'}}">
  847 + <template is="wxParseVideo" data="{{item}}" />
  848 + </block>
  849 +
  850 + <!--img类型-->
  851 + <block wx:elif="{{item.tag == 'img'}}">
  852 + <template is="wxParseImg" data="{{item}}" />
  853 + </block>
  854 +
  855 + <!--a类型-->
  856 + <block wx:elif="{{item.tag == 'a'}}">
  857 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  858 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  859 + <template is="wxParse11" data="{{item}}" />
  860 + </block>
  861 + </view>
  862 + </block>
  863 +
  864 + <block wx:elif="{{item.tag == 'br'}}">
  865 + <template is="WxParseBr"></template>
  866 + </block>
  867 + <!--其他块级标签-->
  868 + <block wx:elif="{{item.tagType == 'block'}}">
  869 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  870 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  871 + <template is="wxParse11" data="{{item}}" />
  872 + </block>
  873 + </view>
  874 + </block>
  875 +
  876 + <!--内联标签-->
  877 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  878 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  879 + <template is="wxParse11" data="{{item}}" />
  880 + </block>
  881 + </view>
  882 +
  883 + </block>
  884 +
  885 + <!--判断是否是文本节点-->
  886 + <block wx:elif="{{item.node == 'text'}}">
  887 + <!--如果是,直接进行-->
  888 + <template is="WxEmojiView" data="{{item}}" />
  889 + </block>
  890 +
  891 +</template>
  892 +
  893 +<!--循环模版-->
  894 +<template name="wxParse11">
  895 + <!--<template is="wxParse12" data="{{item}}" />-->
  896 + <!--判断是否是标签节点-->
  897 + <block wx:if="{{item.node == 'element'}}">
  898 + <block wx:if="{{item.tag == 'button'}}">
  899 + <button type="default" size="mini">
  900 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  901 + <template is="wxParse12" data="{{item}}" />
  902 + </block>
  903 + </button>
  904 + </block>
  905 + <!--li类型-->
  906 + <block wx:elif="{{item.tag == 'li'}}">
  907 + <view class="{{item.classStr}} wxParse-li" style="{{item.styleStr}}">
  908 + <view class="{{item.classStr}} wxParse-li-inner">
  909 + <view class="{{item.classStr}} wxParse-li-text">
  910 + <view class="{{item.classStr}} wxParse-li-circle"></view>
  911 + </view>
  912 + <view class="{{item.classStr}} wxParse-li-text">
  913 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  914 + <template is="wxParse12" data="{{item}}" />
  915 + </block>
  916 + </view>
  917 + </view>
  918 + </view>
  919 + </block>
  920 +
  921 + <!--video类型-->
  922 + <block wx:elif="{{item.tag == 'video'}}">
  923 + <template is="wxParseVideo" data="{{item}}" />
  924 + </block>
  925 +
  926 + <!--img类型-->
  927 + <block wx:elif="{{item.tag == 'img'}}">
  928 + <template is="wxParseImg" data="{{item}}" />
  929 + </block>
  930 +
  931 + <!--a类型-->
  932 + <block wx:elif="{{item.tag == 'a'}}">
  933 + <view bindtap="wxParseTagATap" class="wxParse-inline {{item.classStr}} wxParse-{{item.tag}}" data-src="{{item.attr.href}}" style="{{item.styleStr}}">
  934 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  935 + <template is="wxParse12" data="{{item}}" />
  936 + </block>
  937 + </view>
  938 + </block>
  939 +
  940 + <block wx:elif="{{item.tag == 'br'}}">
  941 + <template is="WxParseBr"></template>
  942 + </block>
  943 + <!--其他块级标签-->
  944 + <block wx:elif="{{item.tagType == 'block'}}">
  945 + <view class="{{item.classStr}} wxParse-{{item.tag}}" style="{{item.styleStr}}">
  946 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  947 + <template is="wxParse12" data="{{item}}" />
  948 + </block>
  949 + </view>
  950 + </block>
  951 +
  952 + <!--内联标签-->
  953 + <view wx:else class="{{item.classStr}} wxParse-{{item.tag}} wxParse-{{item.tagType}}" style="{{item.styleStr}}">
  954 + <block wx:for="{{item.nodes}}" wx:for-item="item" wx:key="">
  955 + <template is="wxParse12" data="{{item}}" />
  956 + </block>
  957 + </view>
  958 +
  959 + </block>
  960 +
  961 + <!--判断是否是文本节点-->
  962 + <block wx:elif="{{item.node == 'text'}}">
  963 + <!--如果是,直接进行-->
  964 + <template is="WxEmojiView" data="{{item}}" />
  965 + </block>
  966 +
  967 +</template>
  1 +
  2 +/**
  3 + * author: Di (微信小程序开发工程师)
  4 + * organization: WeAppDev(微信小程序开发论坛)(http://weappdev.com)
  5 + * 垂直微信小程序开发交流社区
  6 + *
  7 + * github地址: https://github.com/icindy/wxParse
  8 + *
  9 + * for: 微信小程序富文本解析
  10 + * detail : http://weappdev.com/t/wxparse-alpha0-1-html-markdown/184
  11 + */
  12 +
  13 +.wxParse{
  14 + margin: 0 5px;
  15 + font-family: Helvetica,sans-serif;
  16 + font-size: 28rpx;
  17 + color: #666;
  18 + line-height: 1.8;
  19 +}
  20 +view{
  21 + word-break:break-all;
  22 + /*overflow:auto;*/
  23 +}
  24 +.wxParse-inline{
  25 + display: inline;
  26 + line-height: 50rpx;
  27 + margin: 0;
  28 + padding: 0;
  29 +}
  30 +/*//标题 */
  31 +.wxParse-div{margin: 0;padding: 0;}
  32 +.wxParse-h1{ font-size:2em; margin: .67em 0 }
  33 +.wxParse-h2{ font-size:1.5em; margin: .75em 0 }
  34 +.wxParse-h3{ font-size:1.17em; margin: .83em 0 }
  35 +.wxParse-h4{ margin: 1.12em 0}
  36 +.wxParse-h5 { font-size:.83em; margin: 1.5em 0 }
  37 +.wxParse-h6{ font-size:.75em; margin: 1.67em 0 }
  38 +
  39 +.wxParse-h1 {
  40 + font-size: 18px;
  41 + font-weight: 400;
  42 + margin-bottom: .9em;
  43 +}
  44 +.wxParse-h2 {
  45 + font-size: 16px;
  46 + font-weight: 400;
  47 + margin-bottom: .34em;
  48 +}
  49 +.wxParse-h3 {
  50 + font-weight: 400;
  51 + font-size: 15px;
  52 + margin-bottom: .34em;
  53 +}
  54 +.wxParse-h4 {
  55 + font-weight: 400;
  56 + font-size: 14px;
  57 + margin-bottom: .24em;
  58 +}
  59 +.wxParse-h5 {
  60 + font-weight: 400;
  61 + font-size: 13px;
  62 + margin-bottom: .14em;
  63 +}
  64 +.wxParse-h6 {
  65 + font-weight: 400;
  66 + font-size: 12px;
  67 + margin-bottom: .04em;
  68 +}
  69 +
  70 +.wxParse-h1, .wxParse-h2, .wxParse-h3, .wxParse-h4, .wxParse-h5, .wxParse-h6, .wxParse-b, .wxParse-strong { font-weight: bolder }
  71 +
  72 +.wxParse-i,.wxParse-cite,.wxParse-em,.wxParse-var,.wxParse-address{font-style:italic}
  73 +.wxParse-pre,.wxParse-tt,.wxParse-code,.wxParse-kbd,.wxParse-samp{font-family:monospace}
  74 +.wxParse-pre{white-space:pre}
  75 +.wxParse-big{font-size:1.17em}
  76 +.wxParse-small,.wxParse-sub,.wxParse-sup{font-size:.83em}
  77 +.wxParse-sub{vertical-align:sub}
  78 +.wxParse-sup{vertical-align:super}
  79 +.wxParse-s,.wxParse-strike,.wxParse-del{text-decoration:line-through}
  80 +/*wxparse-自定义个性化的css样式*/
  81 +/*增加video的css样式*/
  82 +.wxParse-strong,.wxParse-s{display: inline}
  83 +.wxParse-a{
  84 + color: deepskyblue;
  85 + word-break:break-all;
  86 + overflow:auto;
  87 +}
  88 +
  89 +.wxParse-video{
  90 + text-align: center;
  91 + margin: 10px 0;
  92 +}
  93 +
  94 +.wxParse-video-video{
  95 + width:100%;
  96 +}
  97 +
  98 +.wxParse-img{
  99 + /*background-color: #efefef;*/
  100 + overflow: hidden;
  101 +}
  102 +
  103 +.wxParse-blockquote {
  104 + margin: 0;
  105 + padding:10px 0 10px 5px;
  106 + font-family:Courier, Calibri,"宋体";
  107 + background:#f5f5f5;
  108 + border-left: 3px solid #dbdbdb;
  109 +}
  110 +
  111 +.wxParse-code,.wxParse-wxxxcode-style{
  112 + display: inline;
  113 + background:#f5f5f5;
  114 +}
  115 +.wxParse-ul{
  116 + margin: 20rpx 10rpx;
  117 +}
  118 +
  119 +.wxParse-li,.wxParse-li-inner{
  120 + display: flex;
  121 + align-items: baseline;
  122 + margin: 10rpx 0;
  123 +}
  124 +.wxParse-li-text{
  125 +
  126 + align-items: center;
  127 + line-height: 20px;
  128 +}
  129 +
  130 +.wxParse-li-circle{
  131 + display: inline-flex;
  132 + width: 5px;
  133 + height: 5px;
  134 + background-color: #333;
  135 + margin-right: 5px;
  136 +}
  137 +
  138 +.wxParse-li-square{
  139 + display: inline-flex;
  140 + width: 10rpx;
  141 + height: 10rpx;
  142 + background-color: #333;
  143 + margin-right: 5px;
  144 +}
  145 +.wxParse-li-ring{
  146 + display: inline-flex;
  147 + width: 10rpx;
  148 + height: 10rpx;
  149 + border: 2rpx solid #333;
  150 + border-radius: 50%;
  151 + background-color: #fff;
  152 + margin-right: 5px;
  153 +}
  154 +
  155 +/*.wxParse-table{
  156 + width: 100%;
  157 + height: 400px;
  158 +}
  159 +.wxParse-thead,.wxParse-tfoot,.wxParse-tr{
  160 + display: flex;
  161 + flex-direction: row;
  162 +}
  163 +.wxParse-th,.wxParse-td{
  164 + display: flex;
  165 + width: 580px;
  166 + overflow: auto;
  167 +}*/
  168 +
  169 +.wxParse-u {
  170 + text-decoration: underline;
  171 +}
  172 +.wxParse-hide{
  173 + display: none;
  174 +}
  175 +.WxEmojiView{
  176 + align-items: center;
  177 +}
  178 +.wxEmoji{
  179 + width: 16px;
  180 + height:16px;
  181 +}
  182 +.wxParse-tr{
  183 + display: flex;
  184 + border-right:1px solid #e0e0e0;
  185 + border-bottom:1px solid #e0e0e0;
  186 + border-top:1px solid #e0e0e0;
  187 +}
  188 +.wxParse-th,
  189 +.wxParse-td{
  190 + flex:1;
  191 + padding:5px;
  192 + font-size:28rpx;
  193 + border-left:1px solid #e0e0e0;
  194 + word-break: break-all;
  195 +}
  196 +.wxParse-td:last{
  197 + border-top:1px solid #e0e0e0;
  198 +}
  199 +.wxParse-th{
  200 + background:#f0f0f0;
  201 + border-top:1px solid #e0e0e0;
  202 +}
  203 +.wxParse-del{
  204 + display: inline;
  205 +}
  206 +.wxParse-figure {
  207 + overflow: hidden;
  208 +}