selectpage.js 79 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263
  1. /**
  2. * @summary SelectPage
  3. * @desc Simple and powerful selection plugin
  4. * @file selectpage.js
  5. * @version 2.18
  6. * @author TerryZeng
  7. * @contact https://terryz.github.io/
  8. * @license MIT License
  9. *
  10. */
  11. ;
  12. (function ($) {
  13. "use strict";
  14. /**
  15. * Default options
  16. */
  17. var defaults = {
  18. /**
  19. * Data source
  20. * @type {string|Object}
  21. *
  22. * string:server side request url address
  23. * Object:JSON array,format:[{a:1,b:2,c:3},{...}]
  24. */
  25. data: undefined,
  26. /**
  27. * Language ('cn', 'en', 'ja', 'es', 'pt-br')
  28. * @type string
  29. * @default 'cn'
  30. */
  31. lang: 'cn',
  32. /**
  33. * Multiple select mode(tags)
  34. * @type boolean
  35. * @default false
  36. */
  37. multiple: false,
  38. /**
  39. * pagination or not
  40. * @type boolean
  41. * @default true
  42. */
  43. pagination: true,
  44. /**
  45. * Show up menu button
  46. * @type boolean
  47. * @default true
  48. */
  49. dropButton: true,
  50. /**
  51. * Result list visible size in pagination bar close
  52. * @type number
  53. * @default 10
  54. */
  55. listSize: 10,
  56. /**
  57. * Show control bar in multiple select mode
  58. * @type boolean
  59. * @default true
  60. */
  61. multipleControlbar: true,
  62. /**
  63. * Max selected item limited in multiple select mode
  64. * @type number
  65. * @default 0(unlimited)
  66. */
  67. maxSelectLimit: 0,
  68. /**
  69. * Select result item to close list, work on multiple select mode
  70. * @type boolean
  71. * @default false
  72. */
  73. selectToCloseList: false,
  74. /**
  75. * Init selected item key, the result will match to option.keyField option
  76. * @type string
  77. */
  78. initRecord: undefined,
  79. /**
  80. * The table parameter in server side mode
  81. * @type string
  82. */
  83. dbTable: 'tbl',
  84. /**
  85. * The value field, the value will fill to hidden element
  86. * @type string
  87. * @default 'id'
  88. */
  89. keyField: 'id',
  90. /**
  91. * The show text field, the text will show to input element or tags(multiple mode)
  92. * @type string
  93. * @default 'name'
  94. */
  95. showField: 'name',
  96. /**
  97. * Actually used to search field
  98. * @type string
  99. */
  100. searchField: undefined,
  101. /**
  102. * Search type ('AND' or 'OR')
  103. * @type string
  104. * @default 'AND'
  105. */
  106. andOr: 'AND',
  107. /**
  108. * Result sort type
  109. * @type array - if not set, will default used showField field
  110. * @example
  111. * orderBy : ['id desc']
  112. */
  113. orderBy: undefined,
  114. /**
  115. * Page size
  116. * @type number
  117. * @default 10
  118. */
  119. pageSize: 10,
  120. /**
  121. * Server side request parameters
  122. * @type function
  123. * @return object
  124. * @example params : function(){return {'name':'aa','sex':1};}
  125. */
  126. params: undefined,
  127. /**
  128. * Custom result list item show text
  129. * @type function
  130. * @param data {object} row data
  131. * @return string
  132. */
  133. formatItem: undefined,
  134. /**
  135. * Have some highlight item and lost focus, auto select the highlight item
  136. * @type boolean
  137. * @default false
  138. */
  139. autoFillResult: false,
  140. /**
  141. * Auto select first item in show up result list or search result
  142. * depend on `autoFillResult` option set to true
  143. * @type boolean
  144. * @default false
  145. */
  146. autoSelectFirst: false,
  147. /**
  148. * Whether clear input element text when enter some keywords to search and no result return
  149. * @type boolean
  150. * @default true
  151. */
  152. noResultClean: true,
  153. /**
  154. * Select only mode
  155. * @type boolean
  156. */
  157. selectOnly: false,
  158. /**
  159. * Input to search delay time, work on ajax data source
  160. * @type number
  161. * @default 0.5
  162. */
  163. inputDelay: 0.5,
  164. /**
  165. * -----------------------------------------Callback--------------------------------------------
  166. */
  167. /**
  168. * Result list item selected callback
  169. * @type function
  170. * @param object - selected item json data
  171. * @param self - plugin object
  172. */
  173. eSelect: undefined,
  174. /**
  175. * Before result list show up callback, you can do anything prepared
  176. * @param self - plugin object
  177. */
  178. eOpen: undefined,
  179. /**
  180. * Server side return data convert callback
  181. * @type function
  182. * @param data {object} server side return data
  183. * @param self {object} plugin object
  184. * @return {object} return data format:
  185. * @example
  186. * {
  187. * list : [{name:'aa',sex:1},{name:'bb',sex:1}...],
  188. * totalRow : 100
  189. * }
  190. */
  191. eAjaxSuccess: undefined,
  192. /**
  193. * Close selected item tag callback (multiple mode)
  194. * @type function
  195. * @param removeCount {number} remove item count
  196. * @param self {object} plugin object
  197. */
  198. eTagRemove: undefined,
  199. /**
  200. * Clear selected item callback(single select mode)
  201. * @type function
  202. * @param self {object} plugin object
  203. */
  204. eClear: undefined
  205. };
  206. /**
  207. * SelectPage class definition
  208. * @constructor
  209. * @param {Object} input - input element
  210. * @param {Object} option
  211. */
  212. var SelectPage = function (input, option) {
  213. //特殊字段处理
  214. $.each({data: 'source', keyField: 'primaryKey', showField: 'field', pageSize: 'perPage'}, function (i, j) {
  215. if (typeof option[j] !== 'undefined') {
  216. option[i] = option[j];
  217. delete option[j];
  218. }
  219. });
  220. this.setOption(option);
  221. this.setLanguage();
  222. this.setCssClass();
  223. this.setProp();
  224. this.setElem(input);
  225. this.setButtonAttrDefault();
  226. this.setInitRecord();
  227. this.eDropdownButton();
  228. this.eInput();
  229. this.eWhole();
  230. };
  231. /**
  232. * Plugin version number
  233. */
  234. SelectPage.version = '2.18';
  235. /**
  236. * Plugin object cache key
  237. */
  238. SelectPage.dataKey = 'selectPageObject';
  239. /**
  240. * Options set
  241. * @param {Object} option
  242. */
  243. SelectPage.prototype.setOption = function (option) {
  244. //use showField to default
  245. option.searchField = option.searchField || option.showField;
  246. option.andOr = option.andOr.toUpperCase();
  247. if (option.andOr !== 'AND' && option.andOr !== 'OR')
  248. option.andOr = 'AND';
  249. //support multiple field set
  250. var arr = ['searchField'];
  251. for (var i = 0; i < arr.length; i++) {
  252. option[arr[i]] = this.strToArray(option[arr[i]]);
  253. }
  254. //set default order field
  255. option.orderBy = option.orderBy || option.showField;
  256. //set multiple order field
  257. //example: [ ['id', 'ASC'], ['name', 'DESC'] ]
  258. option.orderBy = this.setOrderbyOption(option.orderBy, option.showField);
  259. //close auto fill result and auto select first in multiple mode and select item not close list
  260. if (option.multiple && !option.selectToCloseList) {
  261. option.autoFillResult = false;
  262. option.autoSelectFirst = false;
  263. }
  264. //show all item when pagination bar close, limited 200
  265. if (!option.pagination)
  266. option.pageSize = 200;
  267. if ($.type(option.listSize) !== 'number' || option.listSize < 0)
  268. option.listSize = 10;
  269. this.option = option;
  270. };
  271. /**
  272. * String convert to array
  273. * @param str {string}
  274. * @return {Array}
  275. */
  276. SelectPage.prototype.strToArray = function (str) {
  277. if (!str)
  278. return '';
  279. return str.replace(/[\s ]+/g, '').split(',');
  280. };
  281. /**
  282. * Set order field
  283. * @param {Array} arg_order
  284. * @param {string} arg_field
  285. * @return {Array}
  286. */
  287. SelectPage.prototype.setOrderbyOption = function (arg_order, arg_field) {
  288. var arr = [], orders = [];
  289. if (typeof arg_order == 'object') {
  290. for (var i = 0; i < arg_order.length; i++) {
  291. orders = $.trim(arg_order[i]).split(' ');
  292. arr[i] = (orders.length == 2) ? orders : [orders[0], 'ASC'];
  293. }
  294. } else {
  295. orders = $.trim(arg_order).split(' ');
  296. arr[0] = (orders.length == 2) ? orders : (orders[0].match(/^(ASC|DESC)$/i)) ? [arg_field, orders[0]] : [orders[0], 'ASC'];
  297. }
  298. return arr;
  299. };
  300. /**
  301. * i18n
  302. */
  303. SelectPage.prototype.setLanguage = function () {
  304. var message, p = this.option;
  305. switch (p.lang) {
  306. // English
  307. case 'en':
  308. message = {
  309. add_btn: 'Add button',
  310. add_title: 'add a box',
  311. del_btn: 'Del button',
  312. del_title: 'delete a box',
  313. next: 'Next',
  314. next_title: 'Next' + p.pageSize + ' (Right key)',
  315. prev: 'Prev',
  316. prev_title: 'Prev' + p.pageSize + ' (Left key)',
  317. first_title: 'First (Shift + Left key)',
  318. last_title: 'Last (Shift + Right key)',
  319. get_all_btn: 'Get All (Down key)',
  320. get_all_alt: '(button)',
  321. close_btn: 'Close (Tab key)',
  322. close_alt: '(button)',
  323. loading: 'loading...',
  324. loading_alt: '(loading)',
  325. page_info: 'page_num of page_count',
  326. select_ng: 'Attention : Please choose from among the list.',
  327. select_ok: 'OK : Correctly selected.',
  328. not_found: 'not found',
  329. ajax_error: 'An error occurred while connecting to server.',
  330. clear: 'Clear content',
  331. select_all: 'Select current page',
  332. unselect_all: 'Clear current page',
  333. clear_all: 'Clear all selected',
  334. max_selected: 'You can only select up to max_selected_limit items'
  335. };
  336. break;
  337. // 中文
  338. case 'cn':
  339. default:
  340. message = {
  341. add_btn: '添加按钮',
  342. add_title: '添加区域',
  343. del_btn: '删除按钮',
  344. del_title: '删除区域',
  345. next: '下一页',
  346. next_title: '下' + p.pageSize + ' (→)',
  347. prev: '上一页',
  348. prev_title: '上' + p.pageSize + ' (←)',
  349. first_title: '首页 (Shift + ←)',
  350. last_title: '尾页 (Shift + →)',
  351. get_all_btn: '获得全部 (↓)',
  352. get_all_alt: '(按钮)',
  353. close_btn: '关闭 (Tab键)',
  354. close_alt: '(按钮)',
  355. loading: '读取中...',
  356. loading_alt: '(读取中)',
  357. page_info: '第 page_num 页(共page_count页)',
  358. select_ng: '请注意:请从列表中选择.',
  359. select_ok: 'OK : 已经选择.',
  360. not_found: '无查询结果',
  361. ajax_error: '连接到服务器时发生错误!',
  362. clear: '清除内容',
  363. select_all: '选择当前页项目',
  364. unselect_all: '取消选择当前页项目',
  365. clear_all: '清除全部已选择项目',
  366. max_selected: '最多只能选择 max_selected_limit 个项目'
  367. };
  368. break;
  369. }
  370. this.message = message;
  371. };
  372. /**
  373. * Css classname defined
  374. */
  375. SelectPage.prototype.setCssClass = function () {
  376. var css_class = {
  377. container: 'sp_container',
  378. container_open: 'sp_container_open',
  379. re_area: 'sp_result_area',
  380. result_open: 'sp_result_area_open',
  381. control_box: 'sp_control_box',
  382. //multiple select mode
  383. element_box: 'sp_element_box',
  384. navi: 'sp_navi',
  385. //result list
  386. results: 'sp_results',
  387. re_off: 'sp_results_off',
  388. select: 'sp_over',
  389. select_ok: 'sp_select_ok',
  390. select_ng: 'sp_select_ng',
  391. selected: 'sp_selected',
  392. input_off: 'sp_input_off',
  393. message_box: 'sp_message_box',
  394. disabled: 'sp_disabled',
  395. button: 'sp_button',
  396. btn_on: 'sp_btn_on',
  397. btn_out: 'sp_btn_out',
  398. input: 'sp_input',
  399. clear_btn: 'sp_clear_btn',
  400. align_right: 'sp_align_right'
  401. };
  402. this.css_class = css_class;
  403. };
  404. /**
  405. * Plugin inner properties
  406. */
  407. SelectPage.prototype.setProp = function () {
  408. this.prop = {
  409. //input disabled status
  410. disabled: false,
  411. current_page: 1,
  412. //total page
  413. max_page: 1,
  414. //ajax data loading status
  415. is_loading: false,
  416. xhr: false,
  417. key_paging: false,
  418. key_select: false,
  419. //last selected item value
  420. prev_value: '',
  421. //last selected item text
  422. selected_text: '',
  423. last_input_time: undefined,
  424. init_set: false
  425. };
  426. this.template = {
  427. tag: {
  428. content: '<li class="selected_tag" itemvalue="#item_value#">#item_text#<span class="tag_close"><i class="spfont sp-close"></i></span></li>',
  429. textKey: '#item_text#',
  430. valueKey: '#item_value#'
  431. },
  432. page: {
  433. current: 'page_num',
  434. total: 'page_count'
  435. },
  436. msg: {
  437. maxSelectLimit: 'max_selected_limit'
  438. }
  439. };
  440. };
  441. /**
  442. * Get the actual width/height of invisible DOM elements with jQuery.
  443. * Source code come from dreamerslab/jquery.actual
  444. * @param element
  445. * @param method
  446. * @returns {*}
  447. */
  448. SelectPage.prototype.elementRealSize = function (element, method) {
  449. var defaults = {
  450. absolute: false,
  451. clone: false,
  452. includeMargin: false,
  453. display: 'block'
  454. };
  455. var configs = defaults, $target = element.eq(0), fix, restore, tmp = [], style = '', $hidden;
  456. fix = function () {
  457. // get all hidden parents
  458. $hidden = $target.parents().addBack().filter(':hidden');
  459. style += 'visibility: hidden !important; display: ' + configs.display + ' !important; ';
  460. if (configs.absolute === true)
  461. style += 'position: absolute !important;';
  462. // save the origin style props
  463. // set the hidden el css to be got the actual value later
  464. $hidden.each(function () {
  465. // Save original style. If no style was set, attr() returns undefined
  466. var $this = $(this), thisStyle = $this.attr('style');
  467. tmp.push(thisStyle);
  468. // Retain as much of the original style as possible, if there is one
  469. $this.attr('style', thisStyle ? thisStyle + ';' + style : style);
  470. });
  471. };
  472. restore = function () {
  473. // restore origin style values
  474. $hidden.each(function (i) {
  475. var $this = $(this), _tmp = tmp[ i ];
  476. if (_tmp === undefined)
  477. $this.removeAttr('style');
  478. else
  479. $this.attr('style', _tmp);
  480. });
  481. };
  482. fix();
  483. // get the actual value with user specific methed
  484. // it can be 'width', 'height', 'outerWidth', 'innerWidth'... etc
  485. // configs.includeMargin only works for 'outerWidth' and 'outerHeight'
  486. var actual = /(outer)/.test(method) ?
  487. $target[ method ](configs.includeMargin) :
  488. $target[ method ]();
  489. restore();
  490. // IMPORTANT, this plugin only return the value of the first element
  491. return actual;
  492. };
  493. /**
  494. * Dom building
  495. * @param {Object} combo_input - original input element
  496. */
  497. SelectPage.prototype.setElem = function (combo_input) {
  498. // 1. build Dom object
  499. var elem = {}, p = this.option, css = this.css_class, msg = this.message, input = $(combo_input);
  500. var orgWidth = input.outerWidth();
  501. // fix input width in hidden situation
  502. if (orgWidth <= 0)
  503. orgWidth = this.elementRealSize(input, 'outerWidth');
  504. if (orgWidth < 150)
  505. orgWidth = 150;
  506. elem.combo_input = input.attr({'autocomplete': 'off'}).addClass(css.input).wrap('<div>');
  507. if (p.selectOnly)
  508. elem.combo_input.prop('readonly', true);
  509. elem.container = elem.combo_input.parent().addClass(css.container);
  510. if (elem.combo_input.prop('disabled')) {
  511. if (p.multiple)
  512. elem.container.addClass(css.disabled);
  513. else
  514. elem.combo_input.addClass(css.input_off);
  515. }
  516. // set outer box width
  517. elem.container.width(orgWidth);
  518. elem.button = $('<div>').addClass(css.button);
  519. //drop down button
  520. elem.dropdown = $('<span class="sp_caret"></span>');
  521. //clear button 'X' in single mode
  522. elem.clear_btn = $('<div>').html($('<i>').addClass('spfont sp-close')).addClass(css.clear_btn).attr('title', msg.clear);
  523. if (!p.dropButton)
  524. elem.clear_btn.addClass(css.align_right);
  525. //main box in multiple mode
  526. elem.element_box = $('<ul>').addClass(css.element_box);
  527. if (p.multiple && p.multipleControlbar)
  528. elem.control = $('<div>').addClass(css.control_box);
  529. //result list box
  530. elem.result_area = $('<div>').addClass(css.re_area);
  531. //pagination bar
  532. if (p.pagination)
  533. elem.navi = $('<div>').addClass('sp_pagination').append('<ul>');
  534. elem.results = $('<ul>').addClass(css.results);
  535. var namePrefix = '_text',
  536. input_id = elem.combo_input.attr('id') || elem.combo_input.attr('name'),
  537. input_name = elem.combo_input.attr('name') || 'selectPage',
  538. hidden_name = input_name,
  539. hidden_id = input_id;
  540. //switch the id and name attributes of input/hidden element
  541. elem.hidden = $('<input type="hidden" class="sp_hidden" />').attr({
  542. name: hidden_name,
  543. id: hidden_id
  544. }).val('');
  545. elem.combo_input.attr({
  546. name: input_name + namePrefix,
  547. id: input_id + namePrefix
  548. });
  549. // 2. DOM element put
  550. elem.container.append(elem.hidden);
  551. if (p.dropButton) {
  552. elem.container.append(elem.button)
  553. elem.button.append(elem.dropdown);
  554. }
  555. $(document.body).append(elem.result_area);
  556. elem.result_area.append(elem.results);
  557. if (p.pagination)
  558. elem.result_area.append(elem.navi);
  559. //Multiple select mode
  560. if (p.multiple) {
  561. if (p.multipleControlbar) {
  562. elem.control.append('<button type="button" class="btn btn-default sp_clear_all" ><i class="spfont sp-clear"></i></button>');
  563. elem.control.append('<button type="button" class="btn btn-default sp_unselect_all" ><i class="spfont sp-unselect-all"></i></button>');
  564. elem.control.append('<button type="button" class="btn btn-default sp_select_all" ><i class="spfont sp-select-all"></i></button>');
  565. elem.control_text = $('<p>');
  566. elem.control.append(elem.control_text);
  567. elem.result_area.prepend(elem.control);
  568. }
  569. elem.container.addClass('sp_container_combo');
  570. elem.combo_input.addClass('sp_combo_input').before(elem.element_box);
  571. var li = $('<li>').addClass('input_box');
  572. li.append(elem.combo_input);
  573. elem.element_box.append(li);
  574. if (elem.combo_input.attr('placeholder'))
  575. elem.combo_input.attr('placeholder_bak', elem.combo_input.attr('placeholder'));
  576. }
  577. this.elem = elem;
  578. };
  579. /**
  580. * Drop down button set to default
  581. */
  582. SelectPage.prototype.setButtonAttrDefault = function () {
  583. /*
  584. if (this.option.selectOnly) {
  585. if ($(this.elem.combo_input).val() !== '') {
  586. if ($(this.elem.hidden).val() !== '') {
  587. //选择条件
  588. $(this.elem.combo_input).attr('title', this.message.select_ok).removeClass(this.css_class.select_ng).addClass(this.css_class.select_ok);
  589. } else {
  590. //输入方式
  591. $(this.elem.combo_input).attr('title', this.message.select_ng).removeClass(this.css_class.select_ok).addClass(this.css_class.select_ng);
  592. }
  593. } else {
  594. $(this.elem.hidden).val('');
  595. $(this.elem.combo_input).removeAttr('title').removeClass(this.css_class.select_ng);
  596. }
  597. }
  598. */
  599. //this.elem.button.attr('title', this.message.get_all_btn);
  600. if (this.option.dropButton)
  601. this.elem.button.attr('title', this.message.close_btn);
  602. };
  603. /**
  604. * Set item need selected after init
  605. * set selected item ways:
  606. * <input value="key">
  607. * <input data-init="key">
  608. */
  609. SelectPage.prototype.setInitRecord = function (refresh) {
  610. var self = this, p = self.option, el = self.elem, key = '';
  611. if ($.type(el.combo_input.data('init')) != 'undefined')
  612. p.initRecord = String(el.combo_input.data('init'));
  613. //data-init and value attribute can be init plugin selected item
  614. //but, if set data-init and value attribute in the same time, plugin will choose data-init attribute first
  615. if (!refresh && !p.initRecord && el.combo_input.val())
  616. p.initRecord = el.combo_input.val();
  617. el.combo_input.val('');
  618. if (!refresh)
  619. el.hidden.val(p.initRecord);
  620. key = refresh && el.hidden.val() ? el.hidden.val() : p.initRecord;
  621. if (key) {
  622. if (typeof p.data === 'object') {
  623. var data = new Array();
  624. var keyarr = key.split(',');
  625. $.each(keyarr, function (index, row) {
  626. for (var i = 0; i < p.data.length; i++) {
  627. if (p.data[i][p.keyField] == row) {
  628. data.push(p.data[i]);
  629. break;
  630. }
  631. }
  632. });
  633. if (!p.multiple && data.length > 1)
  634. data = [data[0]];
  635. self.afterInit(self, data);
  636. } else {//ajax data source mode to init selected item
  637. $.ajax({
  638. dataType: 'json',
  639. type: 'POST',
  640. url: p.data,
  641. data: {
  642. searchTable: p.dbTable,
  643. searchKey: p.keyField,
  644. searchValue: key,
  645. orderBy: p.orderBy,
  646. showField: p.showField,
  647. keyField: p.keyField,
  648. keyValue: key,
  649. searchField: p.searchField
  650. },
  651. success: function (json) {
  652. var d = null;
  653. if (p.eAjaxSuccess && $.isFunction(p.eAjaxSuccess))
  654. d = p.eAjaxSuccess(json);
  655. self.afterInit(self, d.list);
  656. },
  657. error: function (jqXHR, textStatus, errorThrown) {
  658. self.ajaxErrorNotify(self, errorThrown);
  659. }
  660. });
  661. }
  662. }
  663. };
  664. /**
  665. * Selected item set to plugin
  666. * @param {Object} self
  667. * @param {Object} data - selected item data
  668. */
  669. SelectPage.prototype.afterInit = function (self, data) {
  670. if (!data || ($.isArray(data) && data.length === 0))
  671. return;
  672. if (!$.isArray(data))
  673. data = [data];
  674. var p = self.option, css = self.css_class;
  675. var getText = function (row) {
  676. var text = row[p.showField];
  677. if (p.formatItem && $.isFunction(p.formatItem)) {
  678. try {
  679. text = p.formatItem(row);
  680. } catch (e) {
  681. }
  682. }
  683. return text;
  684. };
  685. if (p.multiple) {
  686. self.prop.init_set = true;
  687. self.clearAll(self);
  688. $.each(data, function (i, row) {
  689. var item = {text: getText(row), value: row[p.keyField]};
  690. if (!self.isAlreadySelected(self, item))
  691. self.addNewTag(self, item);
  692. });
  693. self.tagValuesSet(self);
  694. self.inputResize(self);
  695. console.log(self.elem.hidden.blur());
  696. self.prop.init_set = false;
  697. } else {
  698. var row = data[0];
  699. self.elem.combo_input.val(getText(row));
  700. self.elem.hidden.val(row[p.keyField]);
  701. self.prop.prev_value = getText(row);
  702. self.prop.selected_text = getText(row);
  703. if (p.selectOnly) {
  704. self.elem.combo_input.attr('title', self.message.select_ok).removeClass(css.select_ng).addClass(css.select_ok);
  705. }
  706. self.putClearButton();
  707. }
  708. };
  709. /**
  710. * Drop down button event bind
  711. */
  712. SelectPage.prototype.eDropdownButton = function () {
  713. var self = this;
  714. if (self.option.dropButton) {
  715. self.elem.button.mouseup(function (ev) {
  716. ev.stopPropagation();
  717. if (self.elem.result_area.is(':hidden') && !self.elem.combo_input.prop('disabled')) {
  718. self.elem.combo_input.focus();
  719. } else
  720. self.hideResults(self);
  721. });
  722. }
  723. };
  724. /**
  725. * Events bind
  726. */
  727. SelectPage.prototype.eInput = function () {
  728. var self = this, p = self.option, el = self.elem, msg = self.message;
  729. var showList = function () {
  730. self.prop.page_move = false;
  731. self.suggest(self);
  732. self.setCssFocusedInput(self);
  733. };
  734. el.combo_input.keyup(function (e) {
  735. self.processKey(self, e);
  736. }).keydown(function (e) {
  737. self.processControl(self, e);
  738. }).focus(function (e) {
  739. //When focus on input, show the result list
  740. if (el.result_area.is(':hidden')) {
  741. e.stopPropagation();
  742. self.prop.first_show = true;
  743. showList();
  744. }
  745. });
  746. el.container.on('click.SelectPage', 'div.' + self.css_class.clear_btn, function (e) {
  747. e.stopPropagation();
  748. if (!self.disabled(self)) {
  749. self.clearAll(self);
  750. self.elem.hidden.change();
  751. if (p.eClear && $.isFunction(p.eClear))
  752. p.eClear(self);
  753. }
  754. });
  755. el.result_area.on('mousedown.SelectPage', function (e) {
  756. e.stopPropagation();
  757. });
  758. if (p.multiple) {
  759. if (p.multipleControlbar) {
  760. //Select all item of current page
  761. el.control.find('.sp_select_all').on('click.SelectPage', function (e) {
  762. self.selectAllLine(self);
  763. }).hover(function () {
  764. el.control_text.html(msg.select_all);
  765. }, function () {
  766. el.control_text.html('');
  767. });
  768. //Cancel select all item of current page
  769. el.control.find('.sp_unselect_all').on('click.SelectPage', function (e) {
  770. self.unSelectAllLine(self);
  771. }).hover(function () {
  772. el.control_text.html(msg.unselect_all);
  773. }, function () {
  774. el.control_text.html('');
  775. });
  776. //Clear all selected item
  777. el.control.find('.sp_clear_all').on('click.SelectPage', function (e) {
  778. self.clearAll(self);
  779. }).hover(function () {
  780. el.control_text.html(msg.clear_all);
  781. }, function () {
  782. el.control_text.html('');
  783. });
  784. }
  785. el.element_box.on('click.SelectPage', function (e) {
  786. var srcEl = e.target || e.srcElement;
  787. if ($(srcEl).is('ul'))
  788. el.combo_input.focus();
  789. });
  790. //Tag close
  791. el.element_box.on('click.SelectPage', 'span.tag_close', function () {
  792. var li = $(this).closest('li');
  793. self.removeTag(self, li);
  794. showList();
  795. if (p.eTagRemove && $.isFunction(p.eTagRemove))
  796. p.eTagRemove(1, self);
  797. });
  798. self.inputResize(self);
  799. }
  800. };
  801. /**
  802. * Out of plugin area click event handler
  803. */
  804. SelectPage.prototype.eWhole = function () {
  805. var self = this, css = self.css_class;
  806. var cleanContent = function (obj) {
  807. obj.elem.combo_input.val('');
  808. if (!obj.option.multiple)
  809. obj.elem.hidden.val('');
  810. obj.prop.selected_text = '';
  811. };
  812. //Out of plugin area
  813. $(document.body).off('mousedown.selectPage').on('mousedown.selectPage', function (e) {
  814. var ele = e.target || e.srcElement;
  815. var sp = $(ele).closest('div.' + css.container);
  816. //Open status result list
  817. $('div.' + css.container + '.' + css.container_open).each(function () {
  818. if (this == sp[0])
  819. return;
  820. var $this = $(this), d = $this.find('input.' + css.input).data(SelectPage.dataKey);
  821. if (!d.elem.combo_input.val() && d.elem.hidden.val() && !d.option.multiple) {
  822. d.prop.current_page = 1;//reset page to 1
  823. cleanContent(d);
  824. d.hideResults(d);
  825. return true;
  826. }
  827. if (d.elem.results.find('li').not('.' + css.message_box).size()) {
  828. if (d.option.autoFillResult) {
  829. //have selected item, then hide result list
  830. if (d.elem.hidden.val())
  831. d.hideResults(d);
  832. else if (d.elem.results.find('li.sp_over').size()) {
  833. //no one selected and have highlight item, select the highlight item
  834. d.selectCurrentLine(d, true);
  835. } else if (d.option.autoSelectFirst) {
  836. //no one selected, no one highlight, select the first item
  837. d.nextLine(d);
  838. d.selectCurrentLine(d, true);
  839. } else
  840. d.hideResults(d);
  841. } else
  842. d.hideResults(d);
  843. } else {
  844. //when no one item match, clear search keywords
  845. if (d.option.noResultClean)
  846. cleanContent(d);
  847. else {
  848. if (!d.option.multiple)
  849. d.elem.hidden.val('');
  850. }
  851. d.hideResults(d);
  852. }
  853. });
  854. });
  855. };
  856. /**
  857. * Result list event bind
  858. */
  859. SelectPage.prototype.eResultList = function () {
  860. var self = this, css = this.css_class;
  861. self.elem.results.children('li').hover(function () {
  862. if (self.prop.key_select) {
  863. self.prop.key_select = false;
  864. return;
  865. }
  866. if (!$(this).hasClass(css.selected) && !$(this).hasClass(css.message_box)) {
  867. $(this).addClass(css.select);
  868. self.setCssFocusedResults(self);
  869. }
  870. }, function () {
  871. $(this).removeClass(css.select);
  872. }).click(function (e) {
  873. if (self.prop.key_select) {
  874. self.prop.key_select = false;
  875. return;
  876. }
  877. e.preventDefault();
  878. e.stopPropagation();
  879. if (!$(this).hasClass(css.selected))
  880. self.selectCurrentLine(self, false);
  881. });
  882. };
  883. /**
  884. * Reposition result list when list beyond the visible area
  885. */
  886. SelectPage.prototype.eScroll = function () {
  887. var self = this, css = this.css_class;
  888. $(window).on('scroll.SelectPage', function (e) {
  889. $('div.' + css.container + '.' + css.container_open).each(function () {
  890. var $this = $(this), d = $this.find('input.' + css.input).data(SelectPage.dataKey),
  891. offset = d.elem.result_area.offset(),
  892. screenScrollTop = $(window).scrollTop(),
  893. docHeight = $(document).height(),
  894. viewHeight = $(window).height(),
  895. listHeight = d.elem.result_area.outerHeight(),
  896. listBottom = offset.top + listHeight,
  897. hasOverflow = docHeight > viewHeight,
  898. down = d.elem.result_area.hasClass('shadowDown');
  899. if (hasOverflow) {
  900. if (down) {//open down
  901. if (listBottom > (viewHeight + screenScrollTop))
  902. d.calcResultsSize(d);
  903. } else {//open up
  904. if (offset.top < screenScrollTop)
  905. d.calcResultsSize(d);
  906. }
  907. }
  908. });
  909. });
  910. };
  911. /**
  912. * Page bar button event bind
  913. */
  914. SelectPage.prototype.ePaging = function () {
  915. var self = this;
  916. if (!self.option.pagination)
  917. return;
  918. self.elem.navi.find('li.csFirstPage').off('click').on('click', function (ev) {
  919. //$(self.elem.combo_input).focus();
  920. ev.preventDefault();
  921. self.firstPage(self);
  922. });
  923. self.elem.navi.find('li.csPreviousPage').off('click').on('click', function (ev) {
  924. //$(self.elem.combo_input).focus();
  925. ev.preventDefault();
  926. self.prevPage(self);
  927. });
  928. self.elem.navi.find('li.csNextPage').off('click').on('click', function (ev) {
  929. //$(self.elem.combo_input).focus();
  930. ev.preventDefault();
  931. self.nextPage(self);
  932. });
  933. self.elem.navi.find('li.csLastPage').off('click').on('click', function (ev) {
  934. //$(self.elem.combo_input).focus();
  935. ev.preventDefault();
  936. self.lastPage(self);
  937. });
  938. };
  939. /**
  940. * Ajax request fail
  941. * @param {Object} self
  942. * @param {string} errorThrown
  943. */
  944. SelectPage.prototype.ajaxErrorNotify = function (self, errorThrown) {
  945. self.showMessage(self.message.ajax_error);
  946. };
  947. /**
  948. * Message box
  949. * @param {Object} self
  950. * @param msg {string} the text need to show
  951. */
  952. SelectPage.prototype.showMessage = function (self, msg) {
  953. if (!msg)
  954. return;
  955. var msgLi = '<li class="' + self.css_class.message_box + '"><i class="spfont sp-warning"></i> ' + msg + '</li>';
  956. self.elem.results.empty().append(msgLi).show();
  957. self.calcResultsSize(self);
  958. self.setOpenStatus(self, true);
  959. self.elem.control.hide();
  960. if (self.option.pagination)
  961. self.elem.navi.hide();
  962. };
  963. /**
  964. * @desc Scroll
  965. * @param {Object} self
  966. * @param {boolean} enforce
  967. */
  968. SelectPage.prototype.scrollWindow = function (self, enforce) {
  969. var current_result = self.getCurrentLine(self),
  970. target_top = (current_result && !enforce) ? current_result.offset().top : self.elem.container.offset().top,
  971. target_size;
  972. self.prop.size_li = self.elem.results.children('li:first').outerHeight();
  973. target_size = self.prop.size_li;
  974. var gap, client_height = $(window).height(),
  975. scroll_top = $(window).scrollTop(),
  976. scroll_bottom = scroll_top + client_height - target_size;
  977. if (current_result.length) {
  978. if (target_top < scroll_top || target_size > client_height) {
  979. //scroll to top
  980. gap = target_top - scroll_top;
  981. } else if (target_top > scroll_bottom) {
  982. //scroll down
  983. gap = target_top - scroll_bottom;
  984. } else
  985. return; //do not scroll
  986. } else if (target_top < scroll_top)
  987. gap = target_top - scroll_top;
  988. window.scrollBy(0, gap);
  989. };
  990. /**
  991. * change css class by status
  992. * @param self
  993. * @param status {boolean} true: open, false: close
  994. */
  995. SelectPage.prototype.setOpenStatus = function (self, status) {
  996. var el = self.elem, css = self.css_class;
  997. if (status) {
  998. el.container.addClass(css.container_open);
  999. el.result_area.addClass(css.result_open);
  1000. } else {
  1001. el.container.removeClass(css.container_open);
  1002. el.result_area.removeClass(css.result_open);
  1003. }
  1004. };
  1005. /**
  1006. * input element in focus css class set
  1007. * @param {Object} self
  1008. */
  1009. SelectPage.prototype.setCssFocusedInput = function (self) {
  1010. //$(self.elem.results).addClass(self.css_class.re_off);
  1011. //$(self.elem.combo_input).removeClass(self.css_class.input_off);
  1012. };
  1013. /**
  1014. * set result list get focus and input element lost focus
  1015. * @param {Object} self
  1016. */
  1017. SelectPage.prototype.setCssFocusedResults = function (self) {
  1018. //$(self.elem.results).removeClass(self.css_class.re_off);
  1019. //$(self.elem.combo_input).addClass(self.css_class.input_off);
  1020. };
  1021. /**
  1022. * Quick search input keywords listener
  1023. * @param {Object} self
  1024. */
  1025. SelectPage.prototype.checkValue = function (self) {
  1026. var now_value = self.elem.combo_input.val();
  1027. if (now_value != self.prop.prev_value) {
  1028. self.prop.prev_value = now_value;
  1029. self.prop.first_show = false;
  1030. if (self.option.selectOnly)
  1031. self.setButtonAttrDefault();
  1032. if (!self.option.multiple && !now_value) {
  1033. self.elem.combo_input.val('');
  1034. self.elem.hidden.val('');
  1035. self.elem.clear_btn.remove();
  1036. }
  1037. self.suggest(self);
  1038. }
  1039. };
  1040. /**
  1041. * Input handle(regular input)
  1042. * @param {Object} self
  1043. * @param {Object} e - event object
  1044. */
  1045. SelectPage.prototype.processKey = function (self, e) {
  1046. if ($.inArray(e.keyCode, [37, 38, 39, 40, 27, 9, 13]) === -1) {
  1047. if (e.keyCode != 16)
  1048. self.setCssFocusedInput(self); // except Shift(16)
  1049. self.inputResize(self);
  1050. if ($.type(self.option.data) === 'string') {
  1051. self.prop.last_input_time = e.timeStamp;
  1052. setTimeout(function () {
  1053. if ((e.timeStamp - self.prop.last_input_time) === 0)
  1054. self.checkValue(self);
  1055. }, self.option.inputDelay * 1000);
  1056. } else {
  1057. self.checkValue(self);
  1058. }
  1059. }
  1060. }
  1061. /**
  1062. * Input handle(control key)
  1063. * @param {Object} self
  1064. * @param {Object} e - event object
  1065. */
  1066. SelectPage.prototype.processControl = function (self, e) {
  1067. if (($.inArray(e.keyCode, [37, 38, 39, 40, 27, 9]) > -1 && self.elem.result_area.is(':visible')) ||
  1068. ($.inArray(e.keyCode, [13, 9]) > -1 && self.getCurrentLine(self))) {
  1069. e.preventDefault();
  1070. e.stopPropagation();
  1071. e.cancelBubble = true;
  1072. e.returnValue = false;
  1073. switch (e.keyCode) {
  1074. case 37:// left
  1075. if (e.shiftKey)
  1076. self.firstPage(self);
  1077. else
  1078. self.prevPage(self);
  1079. break;
  1080. case 38:// up
  1081. self.prop.key_select = true;
  1082. self.prevLine(self);
  1083. break;
  1084. case 39:// right
  1085. if (e.shiftKey)
  1086. self.lastPage(self);
  1087. else
  1088. self.nextPage(self);
  1089. break;
  1090. case 40:// down
  1091. if (self.elem.results.children('li').length) {
  1092. self.prop.key_select = true;
  1093. self.nextLine(self);
  1094. } else
  1095. self.suggest(self);
  1096. break;
  1097. case 9:// tab
  1098. self.prop.key_paging = true;
  1099. self.selectCurrentLine(self, true);
  1100. //self.hideResults(self);
  1101. break;
  1102. case 13:// return
  1103. self.selectCurrentLine(self, true);
  1104. break;
  1105. case 27:// escape
  1106. self.prop.key_paging = true;
  1107. self.hideResults(self);
  1108. break;
  1109. }
  1110. }
  1111. };
  1112. /**
  1113. * Abort Ajax request
  1114. * @param {Object} self
  1115. */
  1116. SelectPage.prototype.abortAjax = function (self) {
  1117. if (self.prop.xhr) {
  1118. self.prop.xhr.abort();
  1119. self.prop.xhr = false;
  1120. }
  1121. };
  1122. /**
  1123. * Suggest result of search keywords
  1124. * @param {Object} self
  1125. */
  1126. SelectPage.prototype.suggest = function (self) {
  1127. var q_word, val = $.trim(self.elem.combo_input.val());
  1128. if (self.option.multiple)
  1129. q_word = val;
  1130. else {
  1131. if (val && val === self.prop.selected_text)
  1132. q_word = '';
  1133. else
  1134. q_word = val;
  1135. }
  1136. q_word = q_word.split(/[\s ]+/);
  1137. //Before show up result list callback
  1138. if (self.option.eOpen && $.isFunction(self.option.eOpen))
  1139. self.option.eOpen.call(self);
  1140. self.abortAjax(self);
  1141. //self.setLoading(self);
  1142. var which_page_num = self.prop.current_page || 1;
  1143. if (typeof self.option.data == 'object')
  1144. self.searchForJson(self, q_word, which_page_num);
  1145. else
  1146. self.searchForDb(self, q_word, which_page_num);
  1147. };
  1148. /**
  1149. * Loading
  1150. * @param {Object} self
  1151. */
  1152. SelectPage.prototype.setLoading = function (self) {
  1153. if (self.elem.results.html() === '') {
  1154. //self.calcResultsSize(self);
  1155. self.setOpenStatus(self, true);
  1156. }
  1157. };
  1158. /**
  1159. * Search for ajax
  1160. * @param {Object} self
  1161. * @param {Array} q_word - query keyword
  1162. * @param {number} which_page_num - target page number
  1163. */
  1164. SelectPage.prototype.searchForDb = function (self, q_word, which_page_num) {
  1165. var p = self.option;
  1166. if (!p.eAjaxSuccess || !$.isFunction(p.eAjaxSuccess))
  1167. self.hideResults(self);
  1168. var _paramsFunc = p.params, _params = {}, searchKey = p.searchField;
  1169. //when have new query keyword, then reset page number to 1.
  1170. if (q_word.length && q_word[0] && q_word[0] !== self.prop.prev_value)
  1171. which_page_num = 1;
  1172. var _orgParams = {
  1173. q_word: q_word,
  1174. pageNumber: which_page_num,
  1175. pageSize: p.pageSize,
  1176. andOr: p.andOr,
  1177. orderBy: p.orderBy,
  1178. searchTable: p.dbTable,
  1179. showField: self.option.showField,
  1180. keyField: self.option.keyField,
  1181. searchField: self.option.searchField
  1182. };
  1183. _orgParams[searchKey] = q_word[0];
  1184. if (_paramsFunc) {
  1185. var result = $.isFunction(_paramsFunc) ? _paramsFunc() : _paramsFunc;
  1186. if (result && $.isPlainObject(result)) {
  1187. _params = $.extend({}, _orgParams, result);
  1188. } else {
  1189. _params = _orgParams;
  1190. }
  1191. } else
  1192. _params = _orgParams;
  1193. self.prop.xhr = $.ajax({
  1194. dataType: 'json',
  1195. url: p.data,
  1196. type: 'POST',
  1197. data: _params,
  1198. success: function (returnData) {
  1199. if (!returnData || !$.isPlainObject(returnData)) {
  1200. self.hideResults(self);
  1201. self.ajaxErrorNotify(self, errorThrown);
  1202. return;
  1203. }
  1204. var data = {}, json = {};
  1205. try {
  1206. data = p.eAjaxSuccess(returnData);
  1207. json.originalResult = data.list;
  1208. json.cnt_whole = data.totalRow;
  1209. } catch (e) {
  1210. self.showMessage(self, self.message.ajax_error);
  1211. return;
  1212. }
  1213. if(self.elem.navi) {
  1214. $(self.elem.navi).toggleClass("hide", json.cnt_whole <= json.originalResult.length);
  1215. }
  1216. json.candidate = [];
  1217. json.keyField = [];
  1218. if (typeof json.originalResult != 'object') {
  1219. self.prop.xhr = null;
  1220. self.notFoundSearch(self);
  1221. return;
  1222. }
  1223. json.cnt_page = json.originalResult.length;
  1224. for (var i = 0; i < json.cnt_page; i++) {
  1225. for (var key in json.originalResult[i]) {
  1226. if (key == p.keyField) {
  1227. json.keyField.push(json.originalResult[i][key]);
  1228. }
  1229. if (key == p.showField) {
  1230. json.candidate.push(json.originalResult[i][key]);
  1231. }
  1232. }
  1233. }
  1234. self.prepareResults(self, json, q_word, which_page_num);
  1235. },
  1236. error: function (jqXHR, textStatus, errorThrown) {
  1237. if (textStatus != 'abort') {
  1238. self.hideResults(self);
  1239. self.ajaxErrorNotify(self, errorThrown);
  1240. }
  1241. },
  1242. complete: function () {
  1243. self.prop.xhr = null;
  1244. }
  1245. });
  1246. };
  1247. /**
  1248. * Search for json data source
  1249. * @param {Object} self
  1250. * @param {Array} q_word
  1251. * @param {number} which_page_num
  1252. */
  1253. SelectPage.prototype.searchForJson = function (self, q_word, which_page_num) {
  1254. var p = self.option, matched = [], esc_q = [], sorted = [], json = {}, i = 0, arr_reg = [];
  1255. //query keyword filter
  1256. do {
  1257. //'/\W/g'正则代表全部不是字母,数字,下划线,汉字的字符
  1258. //将非法字符进行转义
  1259. esc_q[i] = q_word[i].replace(/\W/g, '\\$&').toString();
  1260. arr_reg[i] = new RegExp(esc_q[i], 'gi');
  1261. i++;
  1262. } while (i < q_word.length);
  1263. // SELECT * FROM data WHERE field LIKE q_word;
  1264. for (i = 0; i < p.data.length; i++) {
  1265. var flag = false, row = p.data[i], itemText;
  1266. for (var j = 0; j < arr_reg.length; j++) {
  1267. itemText = row[p.searchField];
  1268. if (p.formatItem && $.isFunction(p.formatItem))
  1269. itemText = p.formatItem(row);
  1270. if (itemText.match(arr_reg[j])) {
  1271. flag = true;
  1272. if (p.andOr == 'OR')
  1273. break;
  1274. } else {
  1275. flag = false;
  1276. if (p.andOr == 'AND')
  1277. break;
  1278. }
  1279. }
  1280. if (flag)
  1281. matched.push(row);
  1282. }
  1283. // (CASE WHEN ...) then く order some column
  1284. var reg1 = new RegExp('^' + esc_q[0] + '$', 'gi'),
  1285. reg2 = new RegExp('^' + esc_q[0], 'gi'),
  1286. matched1 = [], matched2 = [], matched3 = [];
  1287. for (i = 0; i < matched.length; i++) {
  1288. var orderField = p.orderBy[0][0];
  1289. var orderValue = String(matched[i][orderField]);
  1290. if (orderValue.match(reg1)) {
  1291. matched1.push(matched[i]);
  1292. } else if (orderValue.match(reg2)) {
  1293. matched2.push(matched[i]);
  1294. } else {
  1295. matched3.push(matched[i]);
  1296. }
  1297. }
  1298. if (p.orderBy[0][1].match(/^asc$/i)) {
  1299. matched1 = self.sortAsc(self, matched1);
  1300. matched2 = self.sortAsc(self, matched2);
  1301. matched3 = self.sortAsc(self, matched3);
  1302. } else {
  1303. matched1 = self.sortDesc(self, matched1);
  1304. matched2 = self.sortDesc(self, matched2);
  1305. matched3 = self.sortDesc(self, matched3);
  1306. }
  1307. sorted = sorted.concat(matched1).concat(matched2).concat(matched3);
  1308. /*
  1309. if (sorted.length === undefined || sorted.length === 0 ) {
  1310. self.notFoundSearch(self);
  1311. return;
  1312. }
  1313. */
  1314. json.cnt_whole = sorted.length;
  1315. //page_move used to distinguish between init plugin or page moving
  1316. if (!self.prop.page_move) {
  1317. //only single mode can be used page number relocation
  1318. if (!p.multiple) {
  1319. //get selected item belong page number
  1320. var currentValue = self.elem.hidden.val();
  1321. if ($.type(currentValue) !== 'undefined' && $.trim(currentValue) !== '') {
  1322. var index = 0;
  1323. $.each(sorted, function (i, row) {
  1324. if (row[p.keyField] == currentValue) {
  1325. index = i + 1;
  1326. return false;
  1327. }
  1328. });
  1329. which_page_num = Math.ceil(index / p.pageSize);
  1330. if (which_page_num < 1)
  1331. which_page_num = 1;
  1332. self.prop.current_page = which_page_num;
  1333. }
  1334. }
  1335. } else {
  1336. //set page number to 1 when result number less then page size
  1337. if (sorted.length <= ((which_page_num - 1) * p.pageSize)) {
  1338. which_page_num = 1;
  1339. self.prop.current_page = 1;
  1340. }
  1341. }
  1342. //LIMIT xx OFFSET xx
  1343. var start = (which_page_num - 1) * p.pageSize, end = start + p.pageSize;
  1344. //save original data
  1345. json.originalResult = [];
  1346. //after data filter handle
  1347. for (i = start; i < end; i++) {
  1348. if (sorted[i] === undefined)
  1349. break;
  1350. json.originalResult.push(sorted[i]);
  1351. for (var key in sorted[i]) {
  1352. if (key == p.keyField) {
  1353. if (json.keyField === undefined)
  1354. json.keyField = [];
  1355. json.keyField.push(sorted[i][key]);
  1356. }
  1357. if (key == p.showField) {
  1358. if (json.candidate === undefined)
  1359. json.candidate = [];
  1360. json.candidate.push(sorted[i][key]);
  1361. }
  1362. }
  1363. }
  1364. if (json.candidate === undefined)
  1365. json.candidate = [];
  1366. json.cnt_page = json.candidate.length;
  1367. self.prepareResults(self, json, q_word, which_page_num);
  1368. };
  1369. /**
  1370. * Set order asc
  1371. * @param {Object} self
  1372. * @param {Array} arr - result array
  1373. */
  1374. SelectPage.prototype.sortAsc = function (self, arr) {
  1375. arr.sort(function (a, b) {
  1376. var valA = a[self.option.orderBy[0][0]], valB = b[self.option.orderBy[0][0]];
  1377. return $.type(valA) === 'number' ? valA - valB : String(valA).localeCompare(String(valB));
  1378. });
  1379. return arr;
  1380. };
  1381. /**
  1382. * Set order desc
  1383. * @param {Object} self
  1384. * @param {Array} arr - result array
  1385. */
  1386. SelectPage.prototype.sortDesc = function (self, arr) {
  1387. arr.sort(function (a, b) {
  1388. var valA = a[self.option.orderBy[0][0]], valB = b[self.option.orderBy[0][0]];
  1389. return $.type(valA) === 'number' ? valB - valA : String(valB).localeCompare(String(valA));
  1390. });
  1391. return arr;
  1392. };
  1393. /**
  1394. * Not result found handle
  1395. * @param {Object} self
  1396. */
  1397. SelectPage.prototype.notFoundSearch = function (self) {
  1398. self.elem.results.empty();
  1399. self.calcResultsSize(self);
  1400. self.setOpenStatus(self, true);
  1401. self.setCssFocusedInput(self);
  1402. };
  1403. /**
  1404. * Prepare data to show
  1405. * @param {Object} self
  1406. * @param {Object} json - data result
  1407. * @param {Array} q_word - query keyword
  1408. * @param {number} which_page_num - target page number
  1409. */
  1410. SelectPage.prototype.prepareResults = function (self, json, q_word, which_page_num) {
  1411. if (self.option.pagination)
  1412. self.setNavi(self, json.cnt_whole, json.cnt_page, which_page_num);
  1413. if (!json.keyField)
  1414. json.keyField = false;
  1415. if (self.option.selectOnly && json.candidate.length === 1 && json.candidate[0] == q_word[0]) {
  1416. self.elem.hidden.val(json.keyField[0]);
  1417. this.setButtonAttrDefault();
  1418. }
  1419. var is_query = false;
  1420. if (q_word && q_word.length && q_word[0])
  1421. is_query = true;
  1422. self.displayResults(self, json, is_query);
  1423. };
  1424. /**
  1425. * Build page bar
  1426. * @param {Object} self
  1427. * @param {number} cnt_whole - total record count
  1428. * @param {number} cnt_page
  1429. * @param {number} page_num - current page number
  1430. */
  1431. SelectPage.prototype.setNavi = function (self, cnt_whole, cnt_page, page_num) {
  1432. var msg = self.message;
  1433. /**
  1434. * build pagination bar
  1435. */
  1436. var buildPageNav = function (self, pagebar, page_num, last_page) {
  1437. var updatePageInfo = function () {
  1438. var pageInfo = msg.page_info;
  1439. return pageInfo.replace(self.template.page.current, page_num).replace(self.template.page.total, last_page);
  1440. };
  1441. if (pagebar.find('li').size() === 0) {
  1442. pagebar.hide().empty();
  1443. var iconFist = 'spfont sp-first',
  1444. iconPrev = 'spfont sp-previous',
  1445. iconNext = 'spfont sp-next',
  1446. iconLast = 'spfont sp-last';
  1447. pagebar.append('<li class="csFirstPage" title="' + msg.first_title + '" ><a href="javascript:void(0);"> <i class="' + iconFist + '"></i> </a></li>');
  1448. pagebar.append('<li class="csPreviousPage" title="' + msg.prev_title + '" ><a href="javascript:void(0);"><i class="' + iconPrev + '"></i></a></li>');
  1449. //pagination information
  1450. pagebar.append('<li class="pageInfoBox"><a href="javascript:void(0);"> ' + updatePageInfo() + ' </a></li>');
  1451. pagebar.append('<li class="csNextPage" title="' + msg.next_title + '" ><a href="javascript:void(0);"><i class="' + iconNext + '"></i></a></li>');
  1452. pagebar.append('<li class="csLastPage" title="' + msg.last_title + '" ><a href="javascript:void(0);"> <i class="' + iconLast + '"></i> </a></li>');
  1453. pagebar.show();
  1454. } else {
  1455. pagebar.find('li.pageInfoBox a').html(updatePageInfo());
  1456. }
  1457. };
  1458. var pagebar = self.elem.navi.find('ul'),
  1459. last_page = Math.ceil(cnt_whole / self.option.pageSize); //calculate total page
  1460. if (last_page === 0)
  1461. page_num = 0;
  1462. else {
  1463. if (last_page < page_num)
  1464. page_num = last_page;
  1465. else if (page_num === 0)
  1466. page_num = 1;
  1467. }
  1468. self.prop.current_page = page_num;//update current page number
  1469. self.prop.max_page = last_page;//update page count
  1470. buildPageNav(self, pagebar, page_num, last_page);
  1471. //update paging status
  1472. var dClass = 'disabled',
  1473. first = pagebar.find('li.csFirstPage'),
  1474. previous = pagebar.find('li.csPreviousPage'),
  1475. next = pagebar.find('li.csNextPage'),
  1476. last = pagebar.find('li.csLastPage');
  1477. //first and previous
  1478. if (page_num === 1 || page_num === 0) {
  1479. if (!first.hasClass(dClass))
  1480. first.addClass(dClass);
  1481. if (!previous.hasClass(dClass))
  1482. previous.addClass(dClass);
  1483. } else {
  1484. if (first.hasClass(dClass))
  1485. first.removeClass(dClass);
  1486. if (previous.hasClass(dClass))
  1487. previous.removeClass(dClass);
  1488. }
  1489. //next and last
  1490. if (page_num === last_page || last_page === 0) {
  1491. if (!next.hasClass(dClass))
  1492. next.addClass(dClass);
  1493. if (!last.hasClass(dClass))
  1494. last.addClass(dClass);
  1495. } else {
  1496. if (next.hasClass(dClass))
  1497. next.removeClass(dClass);
  1498. if (last.hasClass(dClass))
  1499. last.removeClass(dClass);
  1500. }
  1501. if (last_page > 1)
  1502. self.ePaging(); //pagination event bind
  1503. };
  1504. /**
  1505. * Render result list
  1506. * @param {Object} self
  1507. * @param {Object} json - result data
  1508. * @param {boolean} is_query - used to different from search to open and just click to open
  1509. */
  1510. SelectPage.prototype.displayResults = function (self, json, is_query) {
  1511. var p = self.option, el = self.elem;
  1512. el.results.hide().empty();
  1513. if (p.multiple && $.type(p.maxSelectLimit) === 'number' && p.maxSelectLimit > 0) {
  1514. var selectedSize = el.element_box.find('li.selected_tag').size();
  1515. if (selectedSize > 0 && selectedSize >= p.maxSelectLimit) {
  1516. var msg = self.message.max_selected;
  1517. self.showMessage(self, msg.replace(self.template.msg.maxSelectLimit, p.maxSelectLimit));
  1518. return;
  1519. }
  1520. }
  1521. if (json.candidate.length) {
  1522. var arr_candidate = json.candidate,
  1523. arr_primary_key = json.keyField,
  1524. keystr = el.hidden.val(),
  1525. keyArr = keystr ? keystr.split(',') : new Array(),
  1526. itemText = '';
  1527. for (var i = 0; i < arr_candidate.length; i++) {
  1528. if (p.formatItem && $.isFunction(p.formatItem)) {
  1529. try {
  1530. itemText = p.formatItem(json.originalResult[i]);
  1531. } catch (e) {
  1532. console.error('formatItem内容格式化函数内容设置不正确!');
  1533. itemText = arr_candidate[i];
  1534. }
  1535. } else
  1536. itemText = arr_candidate[i];
  1537. var list = $('<li>').html(itemText).attr({
  1538. pkey: arr_primary_key[i]
  1539. });
  1540. if (!p.formatItem)
  1541. list.attr('title', itemText);
  1542. //Set selected item highlight
  1543. if ($.inArray(arr_primary_key[i].toString(), keyArr) !== -1) {
  1544. list.addClass(self.css_class.selected);
  1545. }
  1546. //cache item data
  1547. list.data('dataObj', json.originalResult[i]);
  1548. el.results.append(list);
  1549. }
  1550. } else {
  1551. var li = '<li class="' + self.css_class.message_box + '"><i class="spfont sp-warning"></i> ' +
  1552. self.message.not_found + '</li>';
  1553. el.results.append(li);
  1554. }
  1555. el.results.show();
  1556. if (p.multiple && p.multipleControlbar)
  1557. el.control.show();
  1558. if (p.pagination)
  1559. el.navi.show();
  1560. self.calcResultsSize(self);
  1561. self.setOpenStatus(self, true);
  1562. //Result item event bind
  1563. self.eResultList();
  1564. //scrolling listen
  1565. self.eScroll();
  1566. //auto highlight first item in search, have result and set autoSelectFirst to true situation
  1567. if (is_query && json.candidate.length && p.autoSelectFirst)
  1568. self.nextLine(self);
  1569. };
  1570. /**
  1571. * Calculate result list size and position
  1572. * @param {Object} self
  1573. */
  1574. SelectPage.prototype.calcResultsSize = function (self) {
  1575. var p = self.option, el = self.elem;
  1576. var rePosition = function () {
  1577. if (el.container.css('position') === 'static') {
  1578. // position: static
  1579. var offset = el.combo_input.offset();
  1580. el.result_area.css({
  1581. top: offset.top + el.combo_input.outerHeight() + 'px',
  1582. left: offset.left + 'px'
  1583. });
  1584. } else {
  1585. if (!p.pagination) {
  1586. var itemHeight = el.results.find('li:first').outerHeight(true),
  1587. listHeight = itemHeight * p.listSize;
  1588. el.results.css({
  1589. 'max-height': listHeight,
  1590. 'overflow-y': 'auto'
  1591. });
  1592. }
  1593. //handle result list show up side(left, right, up or down)
  1594. var docWidth = $(document).width(),
  1595. docHeight = $(document).height(), //the document full height
  1596. viewHeight = $(window).height(), //browser visible area height
  1597. offset = el.container.offset(),
  1598. screenScrollTop = $(window).scrollTop(),
  1599. listWidth = el.result_area.outerWidth(),
  1600. //result list height
  1601. listHeight = el.result_area.outerHeight(),
  1602. //default left used input element left
  1603. defaultLeft = offset.left, //p.multiple ? -1 : 0;
  1604. //input element height
  1605. inputHeight = el.container.outerHeight(),
  1606. left = (offset.left + listWidth) > docWidth ?
  1607. defaultLeft - (listWidth - el.container.outerWidth()) :
  1608. defaultLeft,
  1609. //the actual top coordinate of input element(outer div)
  1610. screenTop = offset.top, //$(el.container).scrollTop();//offset.top - screenScrollTop;
  1611. top = 0, dist = 5, //set distance between input element and result list
  1612. //the actual top coordinate of result list
  1613. listBottom = screenTop + inputHeight + listHeight + dist,
  1614. hasOverflow = docHeight > viewHeight;
  1615. if ((screenTop - screenScrollTop - dist > listHeight) &&
  1616. (hasOverflow && listBottom > (viewHeight + screenScrollTop)) ||
  1617. (!hasOverflow && listBottom > viewHeight)) {
  1618. //open up
  1619. top = offset.top - listHeight - dist;
  1620. el.result_area.removeClass('shadowUp shadowDown').addClass('shadowUp');
  1621. } else {
  1622. //open down
  1623. top = offset.top + (p.multiple ? el.container.outerHeight() : inputHeight);
  1624. el.result_area.removeClass('shadowUp shadowDown').addClass('shadowDown');
  1625. top += dist;
  1626. }
  1627. return {
  1628. top: top + 'px', left: left + 'px'
  1629. };
  1630. }
  1631. };
  1632. if (el.result_area.is(':visible')) {
  1633. el.result_area.css(rePosition());
  1634. } else {
  1635. var pss = rePosition();
  1636. el.result_area.css(pss).show(1, function () {
  1637. var repss = rePosition();
  1638. if (pss.top !== repss.top || pss.left !== repss.left)
  1639. el.result_area.css(repss);
  1640. });
  1641. }
  1642. };
  1643. /**
  1644. * hide result list
  1645. * @param {Object} self
  1646. */
  1647. SelectPage.prototype.hideResults = function (self) {
  1648. if (self.prop.key_paging) {
  1649. self.scrollWindow(self, true);
  1650. self.prop.key_paging = false;
  1651. }
  1652. self.setCssFocusedInput(self);
  1653. if (self.option.autoFillResult) {
  1654. //self.selectCurrentLine(self, true);
  1655. }
  1656. self.elem.results.empty();
  1657. self.elem.result_area.hide();
  1658. self.setOpenStatus(self, false);
  1659. //unbind window scroll listen
  1660. $(window).off('scroll.SelectPage');
  1661. self.abortAjax(self);
  1662. self.setButtonAttrDefault();
  1663. };
  1664. /**
  1665. * set plugin to disabled / enabled
  1666. * @param self
  1667. * @param disabled
  1668. */
  1669. SelectPage.prototype.disabled = function (self, disabled) {
  1670. var p = self.option, el = self.elem;
  1671. if ($.type(disabled) === 'undefined')
  1672. return el.combo_input.prop('disabled');
  1673. if ($.type(disabled) === 'boolean') {
  1674. el.combo_input.prop('disabled', disabled);
  1675. if (disabled)
  1676. el.container.addClass(self.css_class.disabled);
  1677. else
  1678. el.container.removeClass(self.css_class.disabled);
  1679. }
  1680. };
  1681. /**
  1682. * Go fist page
  1683. * @param {Object} self
  1684. */
  1685. SelectPage.prototype.firstPage = function (self) {
  1686. if (self.prop.current_page > 1) {
  1687. self.prop.current_page = 1;
  1688. self.prop.page_move = true;
  1689. self.suggest(self);
  1690. }
  1691. };
  1692. /**
  1693. * Go previous page
  1694. * @param {Object} self
  1695. */
  1696. SelectPage.prototype.prevPage = function (self) {
  1697. if (self.prop.current_page > 1) {
  1698. self.prop.current_page--;
  1699. self.prop.page_move = true;
  1700. self.suggest(self);
  1701. }
  1702. };
  1703. /**
  1704. * Go next page
  1705. * @param {Object} self
  1706. */
  1707. SelectPage.prototype.nextPage = function (self) {
  1708. if (self.prop.current_page < self.prop.max_page) {
  1709. self.prop.current_page++;
  1710. self.prop.page_move = true;
  1711. self.suggest(self);
  1712. }
  1713. };
  1714. /**
  1715. * Go last page
  1716. * @param {Object} self
  1717. */
  1718. SelectPage.prototype.lastPage = function (self) {
  1719. if (self.prop.current_page < self.prop.max_page) {
  1720. self.prop.current_page = self.prop.max_page;
  1721. self.prop.page_move = true;
  1722. self.suggest(self);
  1723. }
  1724. };
  1725. /**
  1726. * do something after select/unSelect action
  1727. * @param {Object} self
  1728. */
  1729. SelectPage.prototype.afterAction = function (self) {
  1730. self.inputResize(self);
  1731. self.elem.combo_input.change();
  1732. self.setCssFocusedInput(self);
  1733. if (self.prop.init_set)
  1734. return;
  1735. if (self.option.multiple) {
  1736. if (self.option.selectToCloseList) {
  1737. self.hideResults(self);
  1738. self.elem.combo_input.blur();
  1739. } else {
  1740. self.suggest(self);
  1741. self.elem.combo_input.focus();
  1742. }
  1743. } else {
  1744. self.hideResults(self);
  1745. self.elem.combo_input.blur();
  1746. }
  1747. };
  1748. /**
  1749. * Select current list item
  1750. * @param {Object} self
  1751. * @param {boolean} is_enter_key
  1752. */
  1753. SelectPage.prototype.selectCurrentLine = function (self, is_enter_key) {
  1754. self.scrollWindow(self, true);
  1755. var p = self.option, current = self.getCurrentLine(self);
  1756. if (current) {
  1757. if (!p.multiple) {
  1758. self.elem.combo_input.val(current.text());
  1759. self.elem.hidden.val(current.attr('pkey'));
  1760. } else {
  1761. //build tags in multiple selection mode
  1762. self.elem.combo_input.val('');
  1763. var item = {text: current.text(), value: current.attr('pkey')};
  1764. if (!self.isAlreadySelected(self, item)) {
  1765. self.addNewTag(self, item);
  1766. self.tagValuesSet(self);
  1767. }
  1768. }
  1769. if (p.selectOnly)
  1770. self.setButtonAttrDefault();
  1771. //Select item callback
  1772. if (p.eSelect && $.isFunction(p.eSelect))
  1773. p.eSelect(current.data('dataObj'), self);
  1774. self.prop.prev_value = self.elem.combo_input.val();
  1775. self.prop.selected_text = self.elem.combo_input.val();
  1776. self.putClearButton();
  1777. }
  1778. self.afterAction(self);
  1779. };
  1780. /**
  1781. * Show clear button when item selected in single selection mode
  1782. */
  1783. SelectPage.prototype.putClearButton = function () {
  1784. if (!this.option.multiple && !this.elem.combo_input.prop('disabled'))
  1785. this.elem.container.append(this.elem.clear_btn);
  1786. };
  1787. /**
  1788. * Select all list item
  1789. * @param {Object} self
  1790. */
  1791. SelectPage.prototype.selectAllLine = function (self) {
  1792. var p = self.option, jsonarr = new Array();
  1793. self.elem.results.find('li').each(function (i, row) {
  1794. var $row = $(row);
  1795. var item = {text: $row.text(), value: $row.attr('pkey')};
  1796. if (!self.isAlreadySelected(self, item)) {
  1797. self.addNewTag(self, item);
  1798. self.tagValuesSet(self);
  1799. }
  1800. jsonarr.push($row.data('dataObj'));
  1801. //limited max selected items
  1802. if ($.type(p.maxSelectLimit) === 'number' &&
  1803. p.maxSelectLimit > 0 &&
  1804. p.maxSelectLimit === self.elem.element_box.find('li.selected_tag').size()) {
  1805. return false;
  1806. }
  1807. });
  1808. if (p.eSelect && $.isFunction(p.eSelect))
  1809. p.eSelect(jsonarr, self);
  1810. self.afterAction(self);
  1811. };
  1812. /**
  1813. * Cancel select all item in current page
  1814. * @param {Object} self
  1815. */
  1816. SelectPage.prototype.unSelectAllLine = function (self) {
  1817. var p = self.option, size = self.elem.results.find('li').size();
  1818. self.elem.results.find('li').each(function (i, row) {
  1819. var key = $(row).attr('pkey');
  1820. var tag = self.elem.element_box.find('li.selected_tag[itemvalue="' + key + '"]');
  1821. self.removeTag(self, tag);
  1822. });
  1823. self.afterAction(self);
  1824. if (p.eTagRemove && $.isFunction(p.eTagRemove))
  1825. p.eTagRemove(size, self);
  1826. };
  1827. /**
  1828. * Clear all selected items
  1829. * @param {Object} self
  1830. */
  1831. SelectPage.prototype.clearAll = function (self) {
  1832. var p = self.option, size = 0;
  1833. if (p.multiple) {
  1834. size = self.elem.element_box.find('li.selected_tag').size();
  1835. self.elem.element_box.find('li.selected_tag').remove();
  1836. }
  1837. self.reset(self);
  1838. self.afterAction(self);
  1839. if (!p.multiple)
  1840. self.elem.clear_btn.remove();
  1841. if (p.multiple) {
  1842. if (p.eTagRemove && $.isFunction(p.eTagRemove))
  1843. p.eTagRemove(size, self);
  1844. }
  1845. };
  1846. /**
  1847. * reset
  1848. */
  1849. SelectPage.prototype.reset = function (self) {
  1850. self.elem.combo_input.val('');
  1851. self.elem.hidden.val('');
  1852. self.prop.prev_value = '';
  1853. self.prop.selected_text = '';
  1854. self.prop.current_page = 1;
  1855. };
  1856. /**
  1857. * Get current highlight item
  1858. * @param {Object} self
  1859. */
  1860. SelectPage.prototype.getCurrentLine = function (self) {
  1861. if (self.elem.result_area.is(':hidden'))
  1862. return false;
  1863. var obj = self.elem.results.find('li.' + self.css_class.select);
  1864. if (obj.size())
  1865. return obj;
  1866. else
  1867. return false;
  1868. };
  1869. /**
  1870. * Check the result item is already selected or not
  1871. * @param {Object} self
  1872. * @param {Object} item - item info
  1873. */
  1874. SelectPage.prototype.isAlreadySelected = function (self, item) {
  1875. var isExist = false;
  1876. if (item.value) {
  1877. var keys = self.elem.hidden.val();
  1878. if (keys) {
  1879. var karr = keys.split(',');
  1880. if (karr && karr.length && $.inArray(item.value, karr) != -1)
  1881. isExist = true;
  1882. }
  1883. }
  1884. return isExist;
  1885. };
  1886. /**
  1887. * Add a new tag in multiple selection mode
  1888. * @param {Object} self
  1889. * @param {Object} item
  1890. */
  1891. SelectPage.prototype.addNewTag = function (self, item) {
  1892. if (!self.option.multiple || !item)
  1893. return;
  1894. var tmp = self.template.tag.content, tag;
  1895. tmp = tmp.replace(self.template.tag.textKey, item.text);
  1896. tmp = tmp.replace(self.template.tag.valueKey, item.value);
  1897. tag = $(tmp);
  1898. if (self.elem.combo_input.prop('disabled'))
  1899. tag.find('span.tag_close').hide();
  1900. self.elem.combo_input.closest('li').before(tag);
  1901. };
  1902. /**
  1903. * Remove a tag in multiple selection mode
  1904. * @param {Object} self
  1905. * @param {Object} item
  1906. */
  1907. SelectPage.prototype.removeTag = function (self, item) {
  1908. var key = $(item).attr('itemvalue');
  1909. var keys = self.elem.hidden.val();
  1910. if ($.type(key) != 'undefined' && keys) {
  1911. var keyarr = keys.split(','),
  1912. index = $.inArray(key.toString(), keyarr);
  1913. if (index != -1) {
  1914. keyarr.splice(index, 1);
  1915. self.elem.hidden.val(keyarr.toString());
  1916. }
  1917. }
  1918. $(item).remove();
  1919. self.afterAction(self);
  1920. };
  1921. /**
  1922. * Selected item value(keyField) put in to hidden element
  1923. * @param {Object} self
  1924. */
  1925. SelectPage.prototype.tagValuesSet = function (self) {
  1926. if (!self.option.multiple)
  1927. return;
  1928. var tags = self.elem.element_box.find('li.selected_tag');
  1929. if (tags && tags.size()) {
  1930. var result = new Array();
  1931. $.each(tags, function (i, li) {
  1932. var v = $(li).attr('itemvalue');
  1933. if ($.type(v) !== 'undefined')
  1934. result.push(v);
  1935. });
  1936. if (result.length) {
  1937. self.elem.hidden.val(result.join(','));
  1938. }
  1939. }
  1940. };
  1941. /**
  1942. * auto resize input element width in multiple select mode
  1943. * @param {Object} self
  1944. */
  1945. SelectPage.prototype.inputResize = function (self) {
  1946. if (!self.option.multiple)
  1947. return;
  1948. var width = '',
  1949. inputLi = self.elem.combo_input.closest('li');
  1950. var setDefaultSize = function (self, inputLi) {
  1951. inputLi.removeClass('full_width');
  1952. var minimumWidth = self.elem.combo_input.val().length + 1,
  1953. width = (minimumWidth * 0.75) + 'em';
  1954. self.elem.combo_input.css('width', width).removeAttr('placeholder');
  1955. };
  1956. if (self.elem.element_box.find('li.selected_tag').size() === 0) {
  1957. if (self.elem.combo_input.attr('placeholder_bak')) {
  1958. if (!inputLi.hasClass('full_width'))
  1959. inputLi.addClass('full_width');
  1960. self.elem.combo_input.attr('placeholder', self.elem.combo_input.attr('placeholder_bak')).removeAttr('style');
  1961. } else
  1962. setDefaultSize(self, inputLi);
  1963. } else
  1964. setDefaultSize(self, inputLi);
  1965. };
  1966. /**
  1967. * Move to next line
  1968. * @param {Object} self
  1969. */
  1970. SelectPage.prototype.nextLine = function (self) {
  1971. var obj = self.getCurrentLine(self), idx;
  1972. if (!obj)
  1973. idx = -1;
  1974. else {
  1975. idx = self.elem.results.children('li').index(obj);
  1976. obj.removeClass(self.css_class.select);
  1977. }
  1978. idx++;
  1979. if (idx < self.elem.results.children('li').length) {
  1980. var next = self.elem.results.children('li').eq(idx);
  1981. next.addClass(self.css_class.select);
  1982. self.setCssFocusedResults(self);
  1983. } else
  1984. self.setCssFocusedInput(self);
  1985. self.scrollWindow(self, false);
  1986. };
  1987. /**
  1988. * Move to previous line
  1989. * @param {Object} self
  1990. */
  1991. SelectPage.prototype.prevLine = function (self) {
  1992. var obj = self.getCurrentLine(self), idx;
  1993. if (!obj)
  1994. idx = self.elem.results.children('li').length;
  1995. else {
  1996. idx = self.elem.results.children('li').index(obj);
  1997. obj.removeClass(self.css_class.select);
  1998. }
  1999. idx--;
  2000. if (idx > -1) {
  2001. var prev = self.elem.results.children('li').eq(idx);
  2002. prev.addClass(self.css_class.select);
  2003. self.setCssFocusedResults(self);
  2004. } else
  2005. self.setCssFocusedInput(self);
  2006. self.scrollWindow(self, false);
  2007. };
  2008. /**
  2009. * SelectPage plugin definition
  2010. * @global
  2011. * @param option {Object} init plugin option
  2012. */
  2013. function Plugin(option) {
  2014. return this.each(function () {
  2015. var $this = $(this),
  2016. data = $this.data(SelectPage.dataKey),
  2017. params = $.extend({}, defaults, $this.data(), data && data.option, typeof option === 'object' && option);
  2018. if (!data)
  2019. $this.data(SelectPage.dataKey, (data = new SelectPage(this, params)));
  2020. });
  2021. }
  2022. /**
  2023. * Get plugin object
  2024. * @param {object} obj
  2025. * @returns
  2026. */
  2027. function getPlugin(obj) {
  2028. return $(obj).closest('div.sp_container').find('input.sp_input');
  2029. }
  2030. /**
  2031. * Clear all selected item
  2032. */
  2033. function ClearSelected() {
  2034. return this.each(function () {
  2035. var $this = getPlugin(this),
  2036. data = $this.data(SelectPage.dataKey);
  2037. if (data) {
  2038. data.prop.init_set = true;
  2039. data.clearAll(data);
  2040. data.prop.init_set = false;
  2041. }
  2042. });
  2043. }
  2044. /**
  2045. * Refresh result list
  2046. * use case:
  2047. * 1.use $(obj).val('xxx') to modify selectpage selected item key
  2048. * 2.refresh selected item show content/tag text
  2049. */
  2050. function SelectedRefresh() {
  2051. return this.each(function () {
  2052. var $this = getPlugin(this),
  2053. data = $this.data(SelectPage.dataKey);
  2054. if (data && data.elem.hidden.val())
  2055. data.setInitRecord(true);
  2056. });
  2057. }
  2058. /**
  2059. * Modify plugin datasource, only work on json datasource mode
  2060. * @param {array} data - new datasource
  2061. * @example
  2062. * [{name:'aa',sex:1},{name:'bb',sex:0},{...}]
  2063. */
  2064. function ModifyDataSource(data) {
  2065. return this.each(function () {
  2066. if (data && $.isArray(data)) {
  2067. var $this = getPlugin(this),
  2068. plugin = $this.data(SelectPage.dataKey);
  2069. if (plugin) {
  2070. plugin.clearAll(plugin);
  2071. plugin.option.data = data;
  2072. }
  2073. }
  2074. });
  2075. }
  2076. /**
  2077. * Get plugin disabled status or Modify plugin disabled status
  2078. * @param disabled {boolean} set disabled status
  2079. */
  2080. function PluginDisabled(disabled) {
  2081. var status = false;
  2082. this.each(function () {
  2083. var $this = getPlugin(this),
  2084. plugin = $this.data(SelectPage.dataKey);
  2085. if (plugin) {
  2086. if ($.type(disabled) !== 'undefined')
  2087. plugin.disabled(plugin, disabled);
  2088. else
  2089. status = plugin.disabled(plugin);
  2090. }
  2091. });
  2092. return status;
  2093. }
  2094. /**
  2095. * Get selected item text
  2096. * @returns {string}
  2097. */
  2098. function GetInputText() {
  2099. var str = '';
  2100. this.each(function () {
  2101. var $this = getPlugin(this), data = $this.data(SelectPage.dataKey);
  2102. if (data) {
  2103. if (data.option.multiple) {
  2104. var tags = [];
  2105. data.elem.element_box.find('li.selected_tag').each(function (i, tag) {
  2106. tags.push($(tag).text());
  2107. });
  2108. str += tags.toString();
  2109. } else {
  2110. str += data.elem.combo_input.val();
  2111. }
  2112. }
  2113. });
  2114. return str;
  2115. }
  2116. var old = $.fn.selectPage;
  2117. $.fn.selectPage = Plugin;
  2118. $.fn.selectPage.Constructor = SelectPage;
  2119. $.fn.selectPageClear = ClearSelected;
  2120. $.fn.selectPageRefresh = SelectedRefresh;
  2121. $.fn.selectPageData = ModifyDataSource;
  2122. $.fn.selectPageDisabled = PluginDisabled;
  2123. $.fn.selectPageText = GetInputText;
  2124. // SelectPage no conflict
  2125. // =================
  2126. $.fn.selectPage.noConflict = function () {
  2127. $.fn.selectPage = old;
  2128. return this;
  2129. };
  2130. })(window.jQuery);