-
Notifications
You must be signed in to change notification settings - Fork 52
百度阿拉丁礼物页面
LYF edited this page Apr 27, 2016
·
1 revision
/*
* Author:李彦峰
* Date:2015-12-20
* 调调
*/
"use strict";
;(function(document,undefined){
/*
原生ajax函数。用法与jquery ajax一致。
author:pod4g
*/
// 图像占位
var placeholderImg = "";
// 一次加载12条数据
var LOAD_COUNT = 12;
// var intersectArray = [];
var imgUrlPrev = '';
/*存储原生工具方法 start*/
var arrayPrototype = Array.prototype;
var forEach = arrayPrototype.forEach;
var ElementPrototype = HTMLElement.prototype;
/*存储原生工具方法 end*/
/* 扩展原生Node方法 start */
ElementPrototype.hasClass = function(cls){
return this.className.match(new RegExp('(\\s|^)' + cls + '(\\s|$)'));
}
ElementPrototype.addClass = function(cls){
if (!this.hasClass(cls)){
this.className += " " + cls;
}
}
ElementPrototype.removeClass = function(cls){
if (this.hasClass(cls)) {
var reg = new RegExp('(\\s|^)' + cls + '(\\s|$)');
this.className = this.className.replace(reg, ' ');
}
}
ElementPrototype.toggleClass = function(cls){
if(this.hasClass(cls)){
this.removeClass(cls);
}else{
this.addClass(cls);
}
}
ElementPrototype.css = function(key,value){
if(arguments.length == 2){
this.style[key] = value;
}else if(typeof key === 'object'){
var cssText = '';
for(var attr in key){
cssText += attr + ":" + key[attr]+";";
}
this.style.cssText = cssText;
}
}
/*
html gsetter 方法
*/
ElementPrototype.html = function(val){
if(val){
this.innerHTML = val;
}else{
return this.innerHTML;
}
}
function getById(id){
return document.getElementById(id);
}
function getByClass(className,context){
return (context || document).getElementsByClassName(className);
}
function getByTag(tagName,context){
return (context || document).getElementsByTagName(tagName);
}
/* 扩展原生Node方法 end */
/*
生成卡片HTML结构
*/
function getCardHTMLString(shortId,nail,isPrev){
var className = '',imgSrc,article = data.m[shortId];
if(nail){
className = 'class="'+nail+'"';
}
// http://a.diaox2.com/cms/sites/default/files/
imgSrc = placeholderImg;
if(isPrev){
imgSrc = 'http://a.diaox2.com/cms/sites/default/files/'+ article.p;
}
return '<li '+className+' data-id="'+shortId+'"><a href="http://c.diaox2.com/aladin/'+(4294967296 * shortId + (+shortId))+'.html" media="handheld"><figure><img src="'+imgSrc+'"><figcaption><h3>'+article.t+'</h3></figcaption></figure></a></li>';
}
function htmlWork(arr){
var li = '',nail = 'nail';
arr.forEach(function(article,index){
var isPrev = false;
if(index < LOAD_COUNT){
isPrev = true;
}
if(index !== 0 && index % LOAD_COUNT === 0){
li += getCardHTMLString(article,nail,isPrev);
}else{
li += getCardHTMLString(article,'',isPrev);
}
})
getById('article-list').html(li);
}
/*
数组乱序
*/
function shuffle(o){
for(var j, x, i = o.length; i; j = Math.floor(Math.random() * i), x = o[--i], o[i] = o[j], o[j] = x);
return o;
}
function a(list, m) {
//list 求交集来的list,或者是单个tag的list
var sep, spos, output = [];
list.forEach(function(item,i){
if(sep == undefined) {
sep = m[item].w;
spos = i;
return;
} else {
if(m[item].w == sep) {
return; //同上一个值
} else {
output = output.concat(shuffle(list.slice(spos, i)));
sep = m[item].w;
spos = i;
}
}
})
return output;
}
/*
求交集。
若其中一个参数为空数组,则返回另一个非空数组
*/
/*
每次求交之后更新 intersectArray 的值,供懒加载使用
*/
function intersect(arr1,arr2){
var ret = [],m = data.m;
if(arr1 == undefined && arr2 == undefined){
return ret;
}
// 如果arr1为空数组或null,返回arr2
if(arr1 == undefined || arr1.length === 0){
// 无论如何都保证intersectArray最新
// intersectArray = arr2;
return a(arr2,m);
}
// 如果arr2为空数组或null,返回arr1
if(arr2 == undefined || arr2.length == 0){
// intersectArray = arr1;
return a(arr1,m);
}
// intersectArray =
ret = arr1.filter(function(item){
return arr2.indexOf(item) !== -1;
})
// console.log('交集-----------------------------------------------');
// console.log('第一个数组长度:%d',arr1.length);
// console.log('第二个数组长度:%d',arr2.length);
// console.log('交集长度:%d',ret.length);
// console.log('\n')
// 求完交集之后在这儿需要shuffle
return a(ret,data.m);
}
/*
解析URL,返回参数
*/
function parseURL(){
/*
giveingObject:送礼对象
scene:适用节日
*/
var hash = location.hash,
tagList = null,
ret = {},
map,
giveingObjectTagList,
sceneTagList;
if(hash){
hash = decodeURI(hash).replace('#','');
tagList = hash.split("_");
// 如果格式不符合要求,直接返回
if(!tagList.length){
return ret;
}
map = arrayPrototype.map;
giveingObjectTagList = map.call(getByTag('li',getById('left-menu')),function(li){
return li.html();
})
sceneTagList = map.call(getByTag('li',getById('right-menu')),function(li){
return li.html();
})
var giveingObjectVal;
var sceneVal;
tagList.forEach(function(tag){
giveingObjectTagList.forEach(function(giveingObjectTag){
if(giveingObjectTag.indexOf(tag) !== -1){
giveingObjectVal = giveingObjectTag;
}
})
sceneTagList.forEach(function(sceneTag){
if(sceneTag.indexOf(tag) !== -1){
sceneVal = sceneTag;
}
})
})
ret.giveingObject = giveingObjectVal;
ret.scene = sceneVal;
}
// ret = {
// giveingObject:'老婆女友',
// scene:'生日'
// }
return ret;
}
function change(key){
var map = {
"老婆女友":"女友",
"老公男友":"男友",
"春节回家":"春节",
"过年回家":"春节"
}
return map[key] == undefined ? key : map[key];
}
/*
根据参数,来触发相应的tab和menuList的active状态
*/
function triggerSelect(argObj){
var giveingObject = argObj.giveingObject,
scene = argObj.scene,menu,select,
giveObjectIdList = [],
sceneIdList = [],
inter;
// 如果没有参数,直接返回
if(!giveingObject && !scene){
return;
}
/* 1 找出该触发select的元素 */
menu = getByTag('menu')[0],
select = function(arg){
var
toCompare = arg.giveingObject,
name = arg.giveingObject,
elementList;
if(arg.scene){
elementList = getByTag('li',getById('right-menu'));
toCompare = arg.scene;
name = arg.scene;
}else{
elementList = getByTag('li',getById('left-menu'));
}
forEach.call(elementList,function(li){
if(li.html() === toCompare || li.html().indexOf(toCompare) !== -1){
li.click();
return;
}
});
// 默认是折叠状态
// getById('left-menu-tab').addClass('rotate');
// getById('right-menu-tab').addClass('rotate');
// getById('article').addClass('up');
// 返回当前被选中tag的对应的id的集合
return data[name];
}
if(giveingObject){
giveObjectIdList = select({giveingObject:change(giveingObject)});
}
if(scene){
sceneIdList = select({scene:change(scene)});
}
inter = intersect(giveObjectIdList,sceneIdList);
if(!inter || !inter.length){
noresult();
}else{
hasresult();
// 更新DOM
htmlWork(inter);
}
}
/*
判断2个数组是否相等
[] == []
[1,2] == [2,1]
*/
function arrayEqual(arr,arr2){
if(arr == undefined || arr2 == undefined ){
return false;
}
if(arr.length == 0 && arr2.length == 0){
return true;
}
if(arr === arr2){
return true;
}else{
if(arr.length === arr2.length){
var ret = [];
var ret = arr.filter(function(item,index){
// return item !== arr2[index];
return arr2.indexOf(item) === -1;
})
if(ret.length === 0){
return true;
}else{
return false;
}
}else{
return false;
}
}
}
function defaultUpdateDOM(){
var all = [],shortId;
for(shortId in data.m){
all.push(shortId);
}
changeHash();
// 不管怎样都要保证intersectArray是最新的
// intersectArray = all;
htmlWork(all);
}
function noresult(){
var articleList = getById('article-list');
articleList.parentNode.dataset.flag = "hidden";
articleList.html('<p class="no-result">没有结果 换个筛选条件试试~</p>')
// noresultEle = articleList.firstElementChild;
/*
(viwePortHeight - headerHeight - articleMarginTop - noresultEleHeight)/2
*/
// noresultEle.style.cssText = "margin:"+(document.documentElement.clientHeight - getById('left-menu-tab').offsetHeight - parseInt(getComputedStyle(articleList.parentNode,null).marginTop) - noresultEle.offsetHeight)/2+"px 0";
}
function hasresult(){
getById('article').dataset.flag = "";
}
function changeHash(tagList){
var value = '';
if(!tagList || !tagList.length){
location.hash = "";
return;
}
if(tagList.length === 1){
value = encodeURI(tagList[0]);
}else{
value = encodeURI(tagList[0]) + "_" +encodeURI(tagList[1]);
}
location.hash = value;
}
var oldTagList;
function intersectAndUpdateDOM(){
var menu = getByTag('menu')[0],
selectMenu = getByClass('select-menu',menu),
newTagList = [],
l = selectMenu.length,inter;
if(l == 0){
defaultUpdateDOM(newTagList);
return;
}
if(l == 1){
newTagList.push(selectMenu[0].html());
}else{
newTagList.push(selectMenu[0].html());
newTagList.push(selectMenu[1].html());
}
if(arrayEqual(newTagList,oldTagList)){
return;
}
changeHash(newTagList);
inter = getMetaListByTagArray(newTagList);
if(!inter || !inter.length){
noresult();
}else{
hasresult();
htmlWork(inter);
}
oldTagList = newTagList;
}
function getMetaListByTagArray(newTagList){
return newTagList.length === 1 ? intersect(data[change(newTagList[0])]) : intersect(data[change(newTagList[0])],data[change(newTagList[1])]);
}
/*
*/
document.addEventListener('DOMContentLoaded',function(){
// 选出tab和menuList的容器 menu
var menu = getByTag('menu')[0],
// 选出所有tab
tabList = getByClass('tab',menu),
selectTabClassName = "select-tab",
selectMenuClassName = "select-menu",
url = "http://c.diaox2.com/aladin/",
// 记录原始的title,供后续使用
leftMenuTitle = getById("left-menu-tab").firstElementChild.firstChild.nodeValue,
rightMenuTitle = getById("right-menu-tab").firstElementChild.firstChild.nodeValue,
articleList = getById('article-list'),
viewportHeight = document.documentElement.clientHeight,
argObj,flag = true,attr;
/*
循环给每个tab添加点击事件
*/
forEach.call(tabList,function(tab,index){
tab.onclick = function(){
var menuListContainer = getById(this.id.replace('-tab',''));
// 点击的是已经被选中的tab
if(this.hasClass(selectTabClassName)){
this.firstElementChild.firstElementChild.toggleClass('rotate');
getById('article').toggleClass('up');
}else{// 点击的是为被选中的tab
// 选出被选中的tab
var selectTab = getByClass(selectTabClassName,menu)[0],
// 获取被选中的tab对应的 menuList Id
selectMunuListContainer = getById(selectTab.id.replace('-tab',''));
// 取消被选中的tab的选中状态
selectTab.removeClass(selectTabClassName);
selectTab.firstElementChild.firstElementChild.addClass('rotate');
// 给当前被点击的tab添加选中状态
this.addClass(selectTabClassName);
// 显示当前选中的tab对应的 menuList
menuListContainer.css('display','block');
// 保证被选中的tab对应的menuList是展开状态
this.firstElementChild.firstElementChild.removeClass('rotate');
getById('article').removeClass('up');
// 隐藏上一次被选中的tab对应的 menuList
selectMunuListContainer.css('display','none');
}
}
})
/*
menuList中的li的点击事件
*/
forEach.call(getByClass('menu',menu),function(menuListContainer){
menuListContainer.onclick = function(e){
var target = e.target,oldSelectMenu,tab;
if(target.tagName.toUpperCase() === "LI"){
// 查询当前container中已经被选中的menu
oldSelectMenu = getByClass(selectMenuClassName,this)[0];
// 去掉已经被选中menu的选中状态
if(oldSelectMenu){
console.log();
oldSelectMenu.removeClass(selectMenuClassName);
}
// 给被点击的menu添加选中状态
target.addClass(selectMenuClassName);
// 更新tab title
tab = getById(this.id+"-tab").firstElementChild;
tab.firstChild.nodeValue = target.html();
// 选择完tag之后,折叠面板
tab.firstElementChild.addClass('rotate');
getById('article').addClass('up');
intersectAndUpdateDOM();
}
}
})
/*
清除所有条件的点击事件
*/
forEach.call(getByClass('clear',menu),function(clear){
clear.onclick = function(){
// 找出当前clear对应的menuList
var menuList = clear.parentNode,
menuListContainer = menuList.parentNode,
// 在当前menuList中找到被选中的menu
selectMenu = getByClass(selectMenuClassName,menuList)[0];
if(!selectMenu){
// 如果已经清除了筛选条件,直接返回
return;
}
// 清除被选中menu的选中状态
selectMenu.removeClass(selectMenuClassName);
var tab = getById(menuList.parentNode.id+"-tab").firstElementChild;
if(menuListContainer.hasClass("left-menu")){
tab.firstChild.nodeValue = leftMenuTitle;
}else{
tab.firstChild.nodeValue = rightMenuTitle;
}
tab.firstElementChild.addClass('rotate');
getById('article').addClass('up');
intersectAndUpdateDOM();
}
})
/*
根据参数
1、触发select
2、根据select,求交集,发送ajax
3、无参数或参数不符合要求,发送ajax取默认json
*/
// 获取参数
argObj = parseURL();
if(argObj){
flag = true;
for(attr in argObj){
if(argObj[attr] != undefined){
flag = false;
break;
}
}
// 如果没有一个参数,就执行默认的
if(flag){
defaultUpdateDOM();
}
}
// 触发select
triggerSelect(argObj);
/*
惰性加载
*/
// 3007 2377
setInterval(function(){
var
/*
把articleArray 放入定时器内部,
保证每次都是拿到最新的li集合,
如果放到定时器外部,如果切换了tag,但是articleArray其实并没有更新,
所以切换一下tag,在下拉,就不会懒加载了。。事实上永远加载不了。。
因为start永远为-1
*/
articleArray = arrayPrototype.slice.call(articleList.children),
nail = getByClass('nail',articleList)[0],
start = articleArray.indexOf(nail),
end = start + LOAD_COUNT;
if(!nail){
return;
}
if((document.body.scrollTop || document.documentElement.scrollTop) + viewportHeight >= nail.offsetTop){
articleArray.slice(start,end).forEach(function(item,index){
getByTag('img',item)[0].src =
'http://a.diaox2.com/cms/sites/default/files/'+data.m[item.dataset.id].p;
});
nail.removeClass('nail');
}
},60)
},false)
})(document)