-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpost.html
395 lines (340 loc) · 14.5 KB
/
post.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
<header>
<hgroup>
<h1>Faça você mesmo um slideshow em jQuery</h1>
</hgroup>
<nav>
<p>Demos:</p>
<ul>
<li><a href="../../projetos/slideshow/zero.html">zero</a></li>
<li><a href="../../projetos/slideshow/um.html">um</a></li>
<li><a href="../../projetos/slideshow/dois.html">dois</a></li>
<li><a href="../../projetos/slideshow/tres.html">três</a></li>
<li><a href="../../projetos/slideshow/quatro.html">quatro</a></li>
</ul>
</nav>
</header>
<article>
<time datetime="2010-12-15" pubdate>15 de dezembro de 2010</time>
<p>Neste artigo vamos descobrir que não é necessário ir atrás de plugins toda vez que precisamos fazer algo em jQuery. A biblioteca tem muitas ferramentas e quando a tarefa é simples é mais rápido fazer você mesmo (DIY) do que procurar plugins por aí.</p>
<p>Por exemplo, outro dia eu precisava que um conteúdo fosse trocado por outro quando o user clicasse em "próximo", alguns chamam de slideshow, outros de apresentação, tem muitos plugins por aí, que fazem todo tipo de coisa com todo tipo de animação, mil tipos de piruetas psicodélicas...</p>
<p>Até encontrar um que fizesse o necessário e sem bugs e fácil de usar demoraria um pouco, e para piorar o sistema em que eu estava mexendo não permitia incluir outros arquivos javascripts e tinha pouco tempo para entregar. Muitas vezes a gente perde um tempão procurando plugins e no final nenhum dos 15 encontrados servem, esse parecia ser o caso, os plugins eram bem diferentes um do outro. Então resolvi fazer eu mesmo.</p>
<section>
<header>
<h1>
<a href="../../projetos/slideshow/zero.html">zero</a>: html e css
</h1>
</header>
<p>Este será o html usado neste exemplo, todos os slides estão em uma lista e aparecem um abaixo do outro:</p>
<code>
<!DOCTYPE html>
<html lang="pt">
<head>
<meta charset="utf-8" />
<title>tutorial DIY slideshow</title>
<style type="text/css">
#slides{
list-style-type: none;
}
.slide{
width: 200px;
height: 200px;
border: 1px solid #000;
}
</style>
</head>
<body>
<div id="slideshow" >
<p>
<button id="slideanterior" class="slideshow-button" >anterior </button>
<button id="slideproximo" class="slideshow-button" > próximo</button>
</p>
<ul id="slides">
<li class="slide">
1
</li>
<li class="slide">
2
</li>
<li class="slide">
3
</li>
<li class="slide">
4
</li>
<li class="slide">
5
</li>
</ul>
</div><!-- fim slideshow -->
</body>
</html>
</code>
<p>Demo <a href="../../projetos/slideshow/zero.html">zero</a></p>
</section>
<section>
<header>
<h1>
<a href="../../projetos/slideshow/um.html">Um</a>: javascript inicial
</h1>
</header>
<p>A ideia é pegar o html, esconder os slides e mostrar o primeiro.</p>
<p>Então adicionar um listener em cada botão, ao clicar o slide ativo é escondido substituído pelo próximo.</p>
<p>Mais uma linhazinha para quando chegar ao último voltar ao primeiro.</p>
<p>Em 17 linhas (descontando os comentários) já está montado um slideshow funcional:</p>
<code>
<script src="http://ajax.googleapis.com/ajax/libs/jquery/1.4/jquery.min.js" type="text/javascript"></script>
<script type="text/javascript">
jQuery(document).ready(function($) {
//identifica o slideshow
$slideshow = $("#slideshow");
//inicialmente esconde os slides
$slideshow.find("li.slide").hide();
//encontra o prmeiro slide e ativa-o
$slideativo = $slideshow.find("li.slide").first().addClass('slideatual').show();
//ao clicar mostra o proximo slide
$('#slideproximo').click(function(){
//esconde o slide atual
$slideativo.hide();
//procura o proximo
$slideativo = $slideshow.find("li.slideatual").next();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").first();//volta ao primeiro
//remove o marcador do slide anterior
$slideshow.find("li.slideatual").removeClass("slideatual");
//coloca o marcador e mostra
$slideativo.addClass("slideatual").show();
});
//ao clicar mostra o slide anterior
$('#slideanterior').click(function(){
$slideativo.hide();
$slideativo = $slideshow.find("li.slideatual").prev();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").last();//volta ao ultimo
$slideshow.find("li.slideatual").removeClass("slideatual");
$slideativo.addClass("slideatual").show();
});
});
</script>
</code>
<p>Demo <a href="../../projetos/slideshow/um.html">um</a></p>
</section>
<section>
<header>
<h1>
<a href="../../projetos/slideshow/dois.html">Dois</a>: animação
</h1>
</header>
<p>Tá, funcionou, mas ficou sem graça, os slides trocam automaticamente sem efeito nenhum.</p>
<p>Vamos trocar o show() e hide() por um animate().</p>
<p></p>
<code>
//ao clicar mostra o proximo slide
$('#slideproximo').click(function(){
//esconde o slide atual
$slideativo.animate({
"width": "toggle", "opacity": "toggle"
}, "slow", function() {//com callback
//
//procura o proximo
$slideativo = $slideshow.find("li.slideatual").next();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").first();//volta ao primeiro
//remove o marcador do slide anterior
$slideshow.find("li.slideatual").removeClass("slideatual");
//coloca o marcador e mostra
$slideativo.addClass("slideatual").animate({
"width": "toggle", "opacity": "toggle"
}, "slow");
});
});
//ao clicar mostra o slide anterior
$('#slideanterior').click(function(){
$slideativo.animate({
"width": "toggle", "opacity": "toggle"
}, "slow");//sem callback
$slideativo = $slideshow.find("li.slideatual").prev();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").last();//volta ao ultimo
$slideshow.find("li.slideatual").removeClass("slideatual");
$slideativo.addClass("slideatual").animate({
"width": "toggle", "opacity": "toggle"
}, "slow");
});
</code>
<p>Demo <a href="../../projetos/slideshow/dois.html">dois</a></p>
<p>Repare que a animação de recolher usa um callback para começar a mostrar o outro slide só depois que o primeiro já terminou de recolher.</p>
<p>Mas ainda está meio estranha, tanto a animação "anterior" quanto "próxima" estão iguais, seria melhor que fossem diferentes.</p>
</section>
<section>
<header>
<h1>
<a href="../../projetos/slideshow/tres.html">Três</a>: um pra cada lado
</h1>
</header>
<p>Para animar dqa esquerda para a direita primeiro o elemento precisa estar à esquerda <em>$slideativo.show().css("left", $slideativo.outerWidth()*-1).css('opacity', '0');</em>. Vários truques se escondem nessa linha:</p>
<p><em>outerWidth()</em> serve para determinar o tamanho do slide</p>
<p><em>css("left", $slideativo.outerWidth()*-1)</em> *-1 para colocar o slide à esquerda</p>
<p><em>show()</em>, mas só funciona se o elemento estiver com "show"</p>
<p><em>css('opacity', '0')</em> está com "show" mas ainda não deve aparecer.</p>
<code>
$slideativo = $slideshow.find("li.slide").first().addClass('slideatual').css("left","0").show();
//ao clicar mostra o proximo slide
$('#slideproximo').click(function(){
//esconde o slide atual para a direita
$slideativo.animate({
"left": "+="+$slideativo.outerWidth(), "opacity": "0"
}, "slow", function() {//callback
//procura o proximo
$slideativo = $slideshow.find("li.slideatual").next();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").first();//volta ao primeiro
//remove o marcador do slide anterior
$slideshow.find("li.slideatual").removeClass("slideatual");
//posiciona na esquerda
$slideativo.show().css("left", $slideativo.outerWidth()*-1).css('opacity', '0');
//coloca o marcador e mostra
$slideativo.addClass("slideatual").animate({
"left": "0", "opacity": "1"
}, "slow");
});
});
//ao clicar mostra o slide anterior
$('#slideanterior').click(function(){
//esconde o slide atual para a esquerda
$slideativo.animate({
"left": "-="+$slideativo.outerWidth(), "opacity": "0"
}, "slow");
//procura o proximo
$slideativo = $slideshow.find("li.slideatual").prev();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").last();//volta ao ultimo
//remove o marcador do slide anterior
$slideshow.find("li.slideatual").removeClass("slideatual");
//posiciona na direita
$slideativo.show().css("left", $slideativo.outerWidth()).css('opacity', '0');
//coloca o marcador e mostra
$slideativo.addClass("slideatual").animate({
"left": "0", "opacity": "1"
}, "slow");
});
</code>
<p>Repare que em uma está usando callback e na outra não, dessa maneira uma espera desaparecer para mostrar enquanto na outra os slides correm grudados, fica à escolha do freguês, veja a <a href="../../projetos/slideshow/tres.html">demo três</a>.</p>
<p>Foi necessário também adicionar um <code>.css("left","0")</code> para posicionar corretamente os slides.</p>
<p>O css também teve que mudar um pouco:</p>
<code>
#slides{
list-style-type: none;
width: 200px;
overflow: hidden;
}
.slide{
width: 200px;
height: 200px;
border: 1px solid #000;
left: -200px;
position: absolute;
}
</code>
<p>Pronto, agora a animação é diferente, ao clicar em "anterior" os slides correm para a esquerda e em "próximo" os slides correm para a direita.</p>
<p>Demo <a href="../../projetos/slideshow/tres.html">Três</a></p>
</section>
<section>
<header>
<h1>
<a href="../../projetos/slideshow/quatro.html">Quatro</a>: Plugin
</h1>
</header>
<p>Beleza, agora se quiser mesmo fazer um plugin fica fácil, esse DIY não é sobre fazer plugin então vou economizar e usar o <a href="http://starter.pixelgraphics.us/">starter</a>, um gerador de código que gera um template para plugins jQuery.</p>
<p>O código então fica assim (<a href="../../projetos/slideshow/jquery.slideshow.js">jquery.slideshow.js</a>) (um pouco maior que as 17 linhas iniciais):</p>
<code>
(function ($) {
//http://starter.pixelgraphics.us/
$.slideshow = function (el, options) {
// To avoid scope issues, use 'base' instead of 'this'
// to reference this class from internal events and functions.
var base = this;
// Access to jQuery and DOM versions of element
base.$el = $(el);
base.el = el;
// Add a reverse reference to the DOM object
base.$el.data("slideshow", base);
base.init = function(){
//junta as opcoes default com as passadas na chamada do plugin
base.options = $.extend({},$.slideshow.defaultOptions, options);
//um nome mais pratico para base.$el
$slideshow = base.$el;
//inicialmente esconde os slides
$slideshow.find("li.slide").hide();
//encontra o prmeiro slide e ativa-o
$slideativo = $slideshow.find("li.slide").first().addClass('slideatual').css("left","0").show();
base.proximo();
base.anterior();
};
base.proximo = function(paramaters){
//ao clicar mostra o proximo slide
$('#slideproximo').click(function(){
//esconde o slide atual para a direita
$slideativo.animate({
"left": "+="+$slideativo.outerWidth(),
"opacity": "0"
}, "slow");
//procura o proximo
$slideativo = $slideshow.find("li.slideatual").next();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").first();//volta ao primeiro
//remove o marcador do slide anterior
$slideshow.find("li.slideatual").removeClass("slideatual");
//posiciona na esquerda
$slideativo.show().css("left", $slideativo.outerWidth()*-1).css('opacity', '0');
//coloca o marcador e mostra
$slideativo.addClass("slideatual").animate({
"left": "0",
"opacity": "1"
}, "slow");
});
};
base.anterior = function(paramaters){
//ao clicar mostra o slide anterior
$('#slideanterior').click(function(){
//esconde o slide atual para a esquerda
$slideativo.animate({
"left": "-="+$slideativo.outerWidth(),
"opacity": "0"
}, "slow");
//procura o proximo
$slideativo = $slideshow.find("li.slideatual").prev();
if(!$slideativo.size()) $slideativo = $slideshow.find("li.slide").last();//volta ao ultimo
//remove o marcador do slide anterior
$slideshow.find("li.slideatual").removeClass("slideatual");
//posiciona na direita
$slideativo.show().css("left", $slideativo.outerWidth()).css('opacity', '0');
//coloca o marcador e mostra
$slideativo.addClass("slideatual").animate({
"left": "0",
"opacity": "1"
}, "slow");
});
};
// Run initializer
base.init();
};
$.slideshow.defaultOptions = {
//colocar aqui opcoes default
};
$.fn.slideshow = function(options){
return this.each(function () {
(new $.slideshow(this, options));
});
};
})(jQuery)
</code>
<p>E para chamar o plugin basta incluir o script acima e <em>"$('#slideshow').slideshow();"</em></p>
<code>
<script src="jquery.slideshow.js" type="text/javascript"></script>
<script type="text/javascript">
jQuery(document).ready(function($) {
//chama o plugin
$('#slideshow').slideshow();
});
</script>
</code>
<p>Demo <a href="../../projetos/slideshow/quatro.html">quatro</a></p>
</section>
</article>
<p>No total demorou 1 hora para chegar na etapa 3, acho que demoraria mais ou menos a mesma coisa para encontrar um plugin que fizesse algo parecido, mas também poderia gastar esse tempo todo e não encontrar nenhum adequado.</p>
<p>Você já passou por isso também? Deixe um comentário.</p>
<p><em>PS:</em> este post foi feito em html5, confira o código na <a href="../../projetos/slideshow/index.html">demo principal</a>
<p>Também no github: <a href="https://github.com/codexico/diy-slideshow-jQuery">https://github.com/codexico/diy-slideshow-jQuery</a></p>