燕之庐网站建设 - 优质网站设计公司

Posts Tagged ‘ajax’

php

2009/12/01

基于PHP的AJAX,实现文件的异步上传

Tags: , ,

异步的文件上传是在现代的AJAX实现的Web应用里面经常要遇到,必须解决的问题。但是标准的AJAX类(XmlHttpRequest)无 法实现传输文件的功能。因此,这里讨论的内容就是如何在AJAX的技术的基础之上构建异步的文件上传功能。在这个功能当中需要使用到内置的框及 (IFRAME)来传输文件。这个功能实现的效果是页面在上传文件的时候,用户还可以使用该页面并且填写文件描述。

这个例子是我们引用AJAX的经典案例进行分析的。

系统环境

· 较新版本的浏览器。例如Opera,Firefox或者 Internet Explorer。

· PHP 4.3.0 或更高版本

· PHP 5 版本

· PHP 中的 ’short_open_tag’ 选项开启(否则会发生解析错误)。

功能分析

通过内置的IFRAME(框架)进行文件上传。具备包括三个部分组成。

· 在页面中间有一个简单的<form…表单,表单只包含了<input type=”file” … >控件。这个表单的目标链接就是一个隐藏得IFRAME(通过 CSS的风格” display: none;”实现)并且表单里面唯一一个控件的OnChange事件用来触发JavaScript函数。这个函数的作用是检查用户提交的扩展名,然后提交表单。

· 在服务器端用PHP编写了一个处理过程(用FILEFRAME坐注释了)。这个处理过程用来把从客户端上传的文件进行检查后保存在服务器,并且通过 Javascript代码的形式返回给用户。返回给用户的Javascript脚本通过”parent.window.document”更改了用户现在 正在查看的页面,设置了文件的名称并启用了让用户提交表单的按钮。启用按钮的操作是通过getElementById函数实现的。

· 在主页面还有一个表单,它包含了用户提交的描述和隐藏的文件名。用户可以在文件上传的同时填写文件的描述。当文件上传结束以后,用户点击按钮,就可以看上传以后返回给用户的文件信息了。(通过返回来的文件名和用户输入的描述构成文件信息)。

可能你会说这么操作不符合常理:文件在用户确认之前就已经被提交了。如果用户没有提交的话,情况会如何呢。你可以自己在扩展处理被用户放弃的文件。

这个例子把文件存储在一个文件系统的目录下。你需要在脚本开始运行的时候配置下这个目录,具体的包含这个目录信息的变量是$upload_dir 和$web_upload_dir。这里有一个对目录是否可写的权限检查。

这里我们用到了以下几个PHP函数:

· move_uploaded_file – 转移一经上传到服务器的文件

· fopen – 打开文件

· fwrite – 把内容写入文件

· fclose – 关闭文件

· str_replace – 替换字符串

· filesize – 返回文件大小

· filemtime – 返回处理时间

你可以通过手册查到这些函数如果使用。请注意要把HTM(<, >, &)标记替换为(<, > 和 &).
源代码

<?php
$upload_dir = “/var/www/anyexample/aeu”; // 文件存储的路径
$web_upload_dir = “/aeu”; // 文件在Web目录下的路径
$tf = $upload_dir.’/’.md5(rand()).”.test”;
$f = @fopen($tf, “w”);
if ($f == false)
die(”Fatal error! {$upload_dir} is not writable. Set ‘chmod 777 {$upload_dir}’
or something like this”);
fclose($f);
unlink($tf);

//处理上传的文件
if (isset($_POST['fileframe']))
{
$result = ‘ERROR’;
$result_msg = ‘No FILE field found’;

if (isset($_FILES['file'])) // 从浏览器接受文件
{
if ($_FILES['file']['error'] == UPLOAD_ERR_OK) // 没有错误
{
$filename = $_FILES['file']['name']; // 文件名
move_uploaded_file($_FILES['file']['tmp_name'], $upload_dir.’/’.$filename);
// 处理的主过程-转移文件到 $upload_dir
$result = ‘OK’;
}
elseif ($_FILES['file']['error'] == UPLOAD_ERR_INI_SIZE)
$result_msg = ‘The uploaded file exceeds the upload_max_filesize directive in php.ini’;
else
$result_msg = ‘Unknown error’;
}

echo ‘<html><head><title>-</title></head><body>’;
echo ‘<script language=”JavaScript” type=”text/javascript”>’.”\n”;
echo ‘var parDoc = window.parent.document;’;

if ($result == ‘OK’)
{
echo ‘parDoc.getElementById(”upload_status”).value = “file successfully uploaded”;’;
echo ‘parDoc.getElementById(”filename”).value = “‘.$filename.’”;’;
echo ‘parDoc.getElementById(”filenamei”).value = “‘.$filename.’”;’;
echo ‘parDoc.getElementById(”upload_button”).disabled = false;’;
}
else
{
echo ‘parDoc.getElementById(”upload_status”).value = “ERROR: ‘.$result_msg.’”;’;
}

echo “\n”.’</script></body></html>’;
exit();
}

function safehtml($s)
{
$s=str_replace(”&”, “&”, $s);
$s=str_replace(”<”, “<”, $s);
$s=str_replace(”>”, “>”, $s);
$s=str_replace(”‘”, “'”, $s);
$s=str_replace(”\”", “"”, $s);
return $s;
}

if (isset($_POST['description']))
{
$filename = $_POST['filename'];
$size = filesize($upload_dir.’/’.$filename);
$date = date(’r', filemtime($upload_dir.’/’.$filename));
$description = safehtml($_POST['description']);

$html =<<<END
<html><head><title>{$filename} [uploaded by IFRAME Async file uploader]</title></head>
<body>
<h1>{$filename}</h1>
<p>This is a file information page for your uploaded file. Bookmark it, or send to anyone…</p>
<p>Date: {$date}</p>
<p>Size: {$size} bytes</p>
<p>Description:
<pre>{$description}</pre>
</p>
<p><a href=”{$web_upload_dir}/{$filename}” style=”font-size: large;”>download file</a><br>
<a href=”{$PHP_SELF}” style=”font-size: small;”>back to file uploading</a><br>
<a href=”{$web_upload_dir}/upload-log.html” style=”font-size: small;”>upload-log</a></p>
<br><br>Example by <a href=”http://www.anyexample.com/”>AnyExample</a>
</body></html>
END;

$f = fopen($upload_dir.’/’.$filename.’-desc.html’, “w”);
fwrite($f, $html);
fclose($f);
$msg = “File {$filename} uploaded,
<a href=’{$web_upload_dir}/{$filename}-desc.html’>see file information page</a>”;

$f = fopen($upload_dir.”/upload-log.html”, “a”);
fwrite($f, “<p>$msg</p>\n”);
fclose($f);

setcookie(’msg’, $msg);
header(”Location: http://”.$_SERVER['HTTP_HOST'].$PHP_SELF);
exit();
}

if (isset($_COOKIE['msg']) && $_COOKIE['msg'] != ”)
{
if (get_magic_quotes_gpc())
$msg = stripslashes($_COOKIE['msg']);
else
$msg = $_COOKIE['msg'];
setcookie(’msg’, ”);
}
?>
<!– Beginning of main page –>
<html><head>
<title>IFRAME Async file uploader example</title>
</head>
<body>
<?php
if (isset($msg))
echo ‘<p style=”font-weight: bold;”>’.$msg.’</p>’;
?>
<h1>Upload file:</h1>
<p>File will begin to upload just after selection. </p>
<p>You may write file description, while you file is being uploaded.</p>

<form action=”<?=$PHP_SELF?>” target=”upload_iframe” method=”post” enctype=”multipart/form-data”>
<input type=”hidden” name=”fileframe” value=”true”>
<!– Target of the form is set to hidden iframe –>
<!– From will send its post data to fileframe section of this PHP script (see above) –>

<label for=”file”>text file uploader:</label><br>
<!– JavaScript is called by OnChange attribute –>
<input type=”file” name=”file” id=”file” onChange=”jsUpload(this)”>
</form>
<script type=”text/javascript”>
/* This function is called when user selects file in file dialog */
function jsUpload(upload_field)
{
// this is just an example of checking file extensions
// if you do not need extension checking, remove
// everything down to line
// upload_field.form.submit();

var re_text = /\.txt|\.xml|\.zip/i;
var filename = upload_field.value;

/* Checking file type */
if (filename.search(re_text) == -1)
{
alert(”File does not have text(txt, xml, zip) extension”);
upload_field.form.reset();
return false;
}

upload_field.form.submit();
document.getElementById(’upload_status’).value = “uploading file…”;
upload_field.disabled = true;
return true;
}
</script>
<iframe name=”upload_iframe” style=”width: 400px; height: 100px; display: none;”>
</iframe>
<!– For debugging purposes, it’s often useful to remove
“display: none” from style=”" attribute –>

<br>
Upload status:<br>
<input type=”text” name=”upload_status” id=”upload_status”
value=”not uploaded” size=”64″ disabled>
<br><br>

File name:<br>
<input type=”text” name=”filenamei” id=”filenamei” value=”none” disabled>

<form action=”<?=$PHP_SELF?>” method=”POST”>
<!– one field is “disabled” for displaying-only. Other, hidden one is for sending data –>
<input type=”hidden” name=”filename” id=”filename”>
<br><br>

<label for=”photo”>File description:</label><br>
<textarea rows=”5″ cols=”50″ name=”description”></textarea>

<br><br>
<input type=”submit” id=”upload_button” value=”save file” disabled>
</form>
<br><br>
<a href=”<?=$web_upload_dir?>/upload-log.html”>upload-log</a>
<br><br><br>

Example by <a href=”http://www.anyexample.com/”>AnyExample</a>
</body>
</html>

以上的讲解就是提供一种思路供大家参考。大家也可以根据自己的需求进行相应的优化。

php

2009/11/21

AJAX应用中浏览器的BACK后退按钮问题

Tags: ,

AJAX越来越多的应用到网页交互上,但是用AJAX做交互也有一些问题,比如传统网页的浏览是一个一个页面切换,所以可以用浏览器的后退按钮和前进按钮切换到指定的浏览过的页面,也就是浏览器的history.
但是用AJAX做交互的时候,是通过js+xmlhttp来获取其他页面信息,对于浏览用户来说,是一个隐性的UE,感觉不到浏览器地址栏的URL的变化,对浏览器来说,地址栏的URL没发变化,产生不了HISTORY。
不是说在用AJAX做交互的时候,BACK按钮的问题就解决不了,对于AJAX应用很成熟的网站比如google等,这些都已经解决。有热心的老外已经提供了解决方法,粗粗看了看,原理是用了iframe,有兴趣的自己去研究吧
老外管这个叫:Really Simple History framework,还有专门的AJAX history libraries

原文介绍:点击这里浏览
先看一个演示吧:
http://www.donotremove.co.uk/extra/ajax-nav/index.html

这个演示的下载地址,是PHP
http://www.contentwithstyle.co.uk/resources/ajax-nav/ajax-nav.zip

另一篇文章介绍:点击这里浏览

这篇文章例子的下载:点击这里下载

Browser Bookmarking Back button
IE6/PC Yes Yes
IE5.5/PC Yes Yes
IE5/PC Yes Yes
IE5/Mac No No
Firefox/PC Yes Yes
Firefox/Mac Yes Yes
Safari1.2/Mac Yes No

php

2009/11/19

利用PHP和AJAX创建RSS聚合器

Tags: , ,

想象使用一个简单HTML文件来把一个请求发送到一个服务器端脚本,收到一个基于该请求的定制XML文件,然后把它显示给用户而几乎不需要刷新浏览器!本文作者将同你一起探讨怎样在普通Web应用程序中联合   javascript:;” onClick=”javascript:tagshow(event, ‘PHP’);” target=”_self”>PHP 和 AJAX 技术来创建实时的数据传输而不需要进行浏览器刷新。

尽管本文所使用的是PHP语言,但是请记住任何服务器端语言都会正常工作。为了理解本文,我假定你基本理解JavaScript和PHP或一类似服务器端语言。

本文示例使用AJAX来把一请求从一个RSS馈送发送到一定制的PHP对象。该PHP对象复制一份在本地服务器上的该馈送并返回这一路径。该请求对象收到这一路径,分析它,并且把数据以HTML形式显示给用户。这听起来涉及很多步骤,其实它仅由4个小文件组成。之所以使用了4个小文件,是为了平衡它们各自特定的力量而使整个系统的处理极富效率性。

我想,有些读者可能会问,为什么你要创建在本地服务器上的馈送的一个副本而不是简单分析最原始的馈送。原因是,这样以来可以允许绕过XML HTTP Request对象所强加的跨域限制。后面,我还会解释怎样创建这个定制的PHP对象;但是首先,让我们从表单创建开始。

创建发出请求的表单

你要做的第一事情是,在你的HTML的head标签之间包括你可能想使用的JavaScript和任何CSS文件。我包括了一个式样表来实现该聚合器的最后布局并用一个JavaScript文件来发出请求和进行馈送分析:

<link href=”css/layout.css” rel=”stylesheet” type=”text/css” />
<script src=”js/request.js”></script>

下一步,创建一个表单,它针对你所选择的一个RSS馈送发出请求。我创建的表单只包括一个输入字段和一个提交该请求的按钮。该请求的查询是一个字符串,它由馈送输入值和一个将在服务器端被校验的口令字组成;作为一个示例,我使用了下面形式:

“password=mypassword

该代码在每次页面加载之时发出一次请求;因此,如果页面被刷新,现有的在该输入域中的馈送串将在页面加载时被请求。下面是一个表单数据的示例,连同一些div标签用来显示已分析的馈送的特定结点:

<body onload=”javascript:makeRequest(’request.php?request=’ + document.feedForm.feed.value + ‘”password=mypassword’);”>
<form name=”feedForm” method=”post” action=”javascript:makeRequest(’request.php?request=’ + document.feedForm.feed.value + ‘”password=mypassword’);”>
Enter a feed: <input type=”text” name=”feed” id=”feed” size=”20″>
<input type=”submit” name=”submit” value=”Add Feed”>
</form>
<div id=”logo”></div>
<hr/>
<div id=”copy”></div>
<div id=”details”></div>
</body>

我所创建的这三个div标签是logo,copy和details,其中每一个都在布局样式表中有一个与之相关联的样式。当我们分析馈送时将会用到它们,但是我们首先需要能够存取我们所请求的馈送。这可以使用我前面所提到的PHP对象来完成。
创建定制的PHP对象

我用PHP创建了一个小型RSS类,它在本地服务器上创建一个请求馈送的副本,这样它可以为我们稍后要创建的XML HTTP Request对象所存取。典型地,你不能跨域请求一个文件,这意味着你要请求的文件需要位于本地服务器上。这个类是一种解决跨域问题的办法,因为它创建该馈送的一个副本,这个副本在本地服务器上被请求并且把本地路径返回到该馈送,然后它由该Request对象来存取。

这个类中唯一的方法是一个请求方法,它仅有一个指向所请求的RSS 馈送的URL的参数。然后,它通过rss的名字来检查是否一目录位于本地服务器上。如果不存在,就创建一个并把其权限模式设置为0666,这意味着该目录 可读写。当被设置为可读的时,该目录就可以在以后被存取;而当被设置为可写的时,就可以把该馈送的一个副本写向本地服务器上的目录:

//如果不存在目录就创建一个
$dir = “rss”;
if(!is_dir($dir))
{
mkdir($dir, 0666);
}

注意

在一台Windows机器上,对于PHP 4.2.0及以上版本中模式设置是不被要求的。但是,如果它存在的话,它将被忽略;因此,我保留了它,以备该工程被迁移到一台UNIX或Linux服务器上。

在把馈送复制到该服务器前,我们需要一个唯一的文件名。我对这个完整的URL使用了md5加密方法以确保所有馈送的名字是唯一的。通过这个新的文件名,它可以连接一个描述指向该文件的目录的字符串;这将在创建该馈送的副本时使用:

//创建唯一的命名
$file=md5($rss_url);
$path=”$dir/$file.xml”;

通过使用被定义在上面的路径和到原始的被请求的馈送的URL的参考,现在我们能创建该文件的一个副本。最后,把该路径返回到该新文件,作为对该请求的响应:

//复制馈送到本地服务器
copy($rss_url,”$path”);
return $path;
Following is the small, yet powerful RSS class in its entirety:
<?php
class RSS
{
function get($rss_url)
{
if($rss_url != “”)
{
//如果不存在目录就创建一个
$dir = “rss”;
if(!is_dir($dir))
{
mkdir($dir, 0666);
}
// 创建一个唯一的名字
$file = md5($rss_url);
$path = “$dir/$file.xml”;
//复制馈送到本地服务器
copy($rss_url, “$path”);
return $path;
}
}
}
?>

为了存取该PHP类中的方法,需要有一个请求文件来担当到该类的一个接口,这也正是我们正在请求的文件。这个文件首先验证从该请求查询的一口令变量,或 者返回一条指定该请求者不是一名经授权的用户的消息,或者用指向RSS馈送(该馈送在由请求方法处理后被复制到本地服务器)的路径作出响应。为了响应该 RSS馈送,需要包含这个RSS对象并把它实例化,并且需要通过使用被请求的馈送的URL作为一参数来激活请求方法:

<?
if($password == “mypassword”)
{
require_once(’classes/RSS.class.php’);
$rss = new RSS();
echo $rss->get($request);
}
else
{
echo “You are an unauthorized user”;
}
?>

GET/POST与AJAX相结合

为了POST请求,我们首先需要创建该请求对象。如果你没有创建请求对象的经验,那么可以读一下我的文章《How To Use AJAX》或简单地研究一下本文的示例源代码。一旦创建该请求对象,就可以调用sendFeed方法并传递由表单所创建的URL:

function sendFeed(url){
post.onreadystatechange = sendRequest;
post.open(”POST”, url, true);
post.send(url);
}

一旦收到来自于PHP对象的响应并被正确加载,则对与该响应相应的本地文件发出另一个请求。在这种情况中,post.responseText提供给我们该新文件的路径:

function sendRequest(){
if(checkReadyState(post)){
request = createRequestObject();
request.onreadystatechange = onResponse;
request.open(”GET”, post.responseText, true);
request.send(null);
}
}

分析响应

由于RSS馈送之间的区别,分析响应具有一定的挑战性。一些含有包含标题和描述结点的图像,而其它则没有。因此,当我们分析回馈时,我们需要做一点检查来译解它是否包括一图像。如果它包括一图像,我们就可以,与该馈送的标题和链接一起,在image div标签中显示该图像:

var _logo = “”;
var _title = response.getElementsByTagName(’title’)[0].firstChild.data;
var _link = response.getElementsByTagName(’link’)[0].firstChild.data;;
_logo += “<a href=’” + _link + “‘ target=’_blank’>” + _title + “</a><br/>”;
if(checkForTag(response.getElementsByTagName(’image’)[0]))
{
var _url = response.getElementsByTagName(’url’)[0].firstChild.data;
_logo += “<img src=’” + _url + “‘ border=’0′><br/>”
}
document.getElementById(’logo’).innerHTML = _logo;

我们不仅必须检查每个图像以显示它,当遍历馈送中所有的项时我们还需要对之进行检查。因为如果存在一个图像,那么所有另外的标题和链接结点索引都将无法正常工作。因此,当发现图像标签时,我们应该通过在每一次遍历中增加索引值(+1)来调整标题和链接结点的索引:

if(checkForTag(response.getElementsByTagName(’image’)[0]) “” i>0){
var _title=response.getElementsByTagName(’title’)[i+1].firstChild.data;
var _link=response.getElementsByTagName(’link’)[i+1].firstChild.data;
}
else{
var _title =response.getElementsByTagName(’title’)[i].firstChild.data;
var _link = response.getElementsByTagName(’link’)[i].firstChild.data;
}

你可以使用checkForTag方法来检查是否存在特定的标签:

function checkForTag(tag){
if(tag != undefined) {
return true;
}
else{
return false;
}
}

存在许多种进行馈送分析的可能性。例如,你可以把项赋到类别上并使得该类别可折迭,这样用户就可以对其想观看的内容进行选择。作为一个示例,我使用日期 来对项进行分类-这可以通过译解是否针对一个特定项的pubDate不同于前一个项的pubDate并且相应地显示一新的日期来实现:

if(i>1){
var previousPubDate = response.getElementsByTagName(’pubDate’)[i-1].firstChild.data;
}
if(pubDate != previousPubDate || previousPubDate == undefined){
_copy += “<div id=’detail’>” + pubDate + “</div><hr align=’left’ width=’90%’/>”;
}
_copy += “<a href=\”javascript:showDetails(’” + i + “‘);\”>” + _title + “</a><br/><br/>”;
document.getElementById(’copy’).innerHTML += _copy;

注意,上面的最后一部分是showDetails方法,它用于当一用户从一个馈送中选择一特定的项时进行细节显示。这个方法有一个参数(项索引值),这个索引用于发现在该馈送中details结点的索引:

function showDetails(index){
document.getElementById(’details’).innerHTML = response.getElementsByTagName(’description’)[index].firstChild.data;
}

结论

使用AJAX发送查询字符串到一个服务器端脚本并检索一个基于该串的定制响应,这对于任何web开发者都有实现的可能。这样以来,你的下一个web应用程序也将会充满了新的可能性。

php

通过PHP和Sajax使用Ajax之JavaScript

Tags: , ,

多年以来,创建真正具有响应性的 Web 应用程序这一目标一直被 Web 开发的一个简单事实所阻碍:要改变页面某一部分的信息,用户就必须重载整个页面。但是以后不再会这样了。感谢异步 Javascript 脚本和 XML(Ajax),我们现在可以从服务器端请求新内容,只修改页面的一部分。这个教程解释了如何把 Ajax 用于 PHP ,并介绍了简单 Ajax 工具包(Sajax),这是一个用 PHP 编写的工具,可以把服务器端 PHP 与 JavaScript 集成。

开始之前

这份教程针对的是对于开发富 Web 应用程序感兴趣的人,富 Web 应用程序把异步 JavaScript 和 XML(Ajax)与 PHP 结合起来,用户每次点击时,不必刷新整个页面,就可以动态更新内容。这份教程假设读者了解基本的 PHP 概念,包括 if 和 switch 语句以及函数的使用。

关于本教程

在本教程中,将学习 Ajax 以及围绕它的应用的问题。将用 PHP 构建一个 Ajax 应用程序,显示以前写的一个教程中的面板。点击面板链接只会重新载入内容区,并用选定面板的内容替换它,从而节约了带宽和页面装入的时间。然后将把简单 Ajax 工具包(Sajax)集成进 Ajax 应用程序,它可以同步 Ajax 的使用,从而简化开发。

概述

在深入之前,先看看 Ajax、示例 PHP 应用程序和 Sajax。 Ajax

Ajax 允许 Web 开发人员创建交互的 Web 页面,同时避免必须等候页面载入这一瓶颈。通过 Ajax 创建的应用程序,只需点击按钮,就可以用全新的内容替换 Web 页面某一区域的内容。它的精彩之处在于不必等候页面装入,只有这一个区域的内容需要载入。以 Google Maps 为例:可以点击和四处移动地图,却不必等候页面载入。

Ajax 的问题

在使用 Ajax 时有些事需要注意。像其他 Web 页面一样,Ajax 页面是可以加书签的,所以在使用 GET 与 POST 进行请求时就会造成问题。国际化和编码方案数量的增加,使得把这些编码方案标准化变得日益重要。在这份教程中将了解这些重要的问题。

示例 PHP 应用程序

首先要用 Ajax 创建一个应用程序,然后用 Sajax 创建,以展现使用这个工具包的好处。应用程序是以前编写的教程中的一部分,带有面板链接。它被用作示例,以展示使用 Ajax 的优势。因为在各个面板上点击时,它们会异步装入,而不必等候页面剩下的部分再次装入。这个示例应用程序还会展示如何创建自己的 Ajax 应用程序。

Sajax

如果想创建 Ajax 应用程序,又不想受 Ajax 复杂的细节所累。答案就是 Sajax。通过使用 ModernMethod 人员开发的库,Sajax 为 Web 开发人员抽象出了 Ajax 的高层细节。在底层,Sajax 的工作与 Ajax 相同。但是,通过使用 Sajax 库提供的高层函数,可以忽略 Ajax 的技术细节。

什么是 Ajax?

这一节是个入门介绍,用示例解释 Ajax 的概念,包括在点击链接时发生了什么,Ajax 用于 PHP 应用程序时需要的 HTML 和 JavaScript 代码。下一节将更深入一些,实际地使用在这一节学习的 Ajax 概念创建 PHP 应用程序。

幕后内容

Ajax 是异步 JavaScript 和 XML 的组合。之所以说异步,是因为可以点击页面上的链接,然后它只装入与点击对应的内容,同时保持标题或其他任何设定的信息不动。

点击链接时,在背后工作的是 JavaScript 函数。JavaScript 创建与 Web 浏览器通信的对象,并告诉浏览器装入特定页面。然后可以像平常一样浏览同一页面上的其他内容,当浏览器完全装入新页面的时候,浏览器会在 HTML 的 div 标记指定的位置显示内容。

CSS 样式代码用来和 span 标记一起创建链接。

CSS 样式代码

示例应用程序需要 CSS 代码,这样 span 标记看起来就像使用常规的锚标记(<a href=… >)创建的真正链接一样,也会像真正的链接一样被点击。

清单 1. 指定 span 标记的显示信息


<style type=”text/css”>
span:visited{ text-decoration:none; color:#293d6b; }
span:hover{ text-decoration:underline; color:#293d6b; }
span {color:#293d6b; cursor: pointer}
</style>

这些 span 标记用在示例应用程序中,颜色符合所有 IBM developerWorks 教程中链接使用的颜色。样式标记的第一行指定已经访问过的链接的颜色保持不变。鼠标经过时加下划线,光标变成指针,就像普通的锚标记(<a href… >)一样。现在来看看如何创建使用这个 CSS 样式代码的链接。

创建使用 span 标记的链接

在“构建 PHP 应用程序”一节中要创建的链接,将用来通过 JavaScript 与浏览器通信,告诉浏览器要去什么地方,要提取什么内容。它们不是使用锚标记的传统链接,而是使用 span 标记创建的。span 标记的观感由清单 1 的 CSS 代码决定。这里是示例:

<span onclick=”loadHTML(’panels-ajax.php?panel_id=0′,
‘content’)”>Managing content</span>

onclick 处理程序指定这个 span 被点击时要运行哪个脚本。还有其他几个与 onclick 类似的指示符可以使用,包括 onmouseover 和 ondblclick。请注意在 onclick 字段中显示的是 JavaScript 函数 loadHTML ,而不是传统的 http:// 链接或由清单 panels-ajax.php? 创建的相对链接。接下来学习 loadHTML 函数。

XMLHttpRequest 对象

如果正在使用 Mozilla、Opera 或其他这类浏览器中的一个,那么可以使用内置的 XMLHttpRequest 对象动态地取得内容。Microsoft 的 Internet Explorer 浏览器采用另外一个对象,稍后将会学到。它们使用的方式实际上相同,而且对它们提供支持,只是添加几行额外代码的问题。

XMLHttpRequest 对象用来通过 JavaScript 检索页面内容。稍后在示例应用程序中会使用这个代码,同 ActiveXObject 的 loadHTML 函数一起使用。请参阅清单 2 了解用法。

清单 2. 初始化和使用 XMLHttpRequest 对象


<style>
<script type=”text/javascript”>
var request;
var dest;

function loadHTML(URL, destination){
dest = destination;
if(window.XMLHttpRequest){
request = new XMLHttpRequest();
request.onreadystatechange = processStateChange;
request.open(”GET”, URL, true);
request.send(null);
}
}
</script>

在清单 2 中作为参数传递的 destination 变量指出 XMLHttpRequest 对象要去装入内容的地方,由 <div id=”content”></div> 标记指定。然后代码会检查 XMLHttpRequest 对象是否存在,如果存在,就创建一个新的。然后,事件处理程序被设置为 processStateChange 函数,这个函数是对象在每次状态变化时都会调用的函数。请求剩下的部分就是用 open 方法进行设置,设置传输类型为 GET,并设置对象要装入的 URL。最后调用对象的 send 方法,让对象实际发挥作用。
ActiveXObject

在 Internet Explorer 中,用 ActiveXObject 代替 XMLHttpRequest。它的函数与 XMLHttpRequest 的函数相同,甚至函数名都是一样的,如清单 3 所示。

清单 3. 初始化和使用 ActiveXObject


function loadHTML(URL, destination){
dest = destination;
if(window.XMLHttpRequest){

} else if (window.ActiveXObject) {
request = new ActiveXObject(”Microsoft.XMLHTTP”);
if (request) {
request.onreadystatechange = processStateChange;
request.open(”GET”, URL, true);
request.send();
}
}
}
</script>

在这种情况下(使用 Internet Explorer),实例化一个 Microsoft.XMLHTTP 类型 的新 ActiveXObject 对象。然后,设置事件处理程序,调用对象的 open 函数。然后调用对象的 send 函数,这样 ActiveXObject 就工作了。

processStateChange 函数

这里描述的函数叫作事件处理程序或回调函数。回调函数的目的是当对象状态发生变化时,能够处理状态变化。在具体的应用中,这个函数的目的应当是处理状态变化、检验对象是否到达预期状态、读取动态装入的内容。

processStateChange 函数由 XMLHttpRequest 或 ActiveXObject 对象在对象状态发生变化时调用。当对象进入状态 4 时,表明页面的内容已经接收完成(请参阅清单 4)。

清单 4. 处理状态变化


var dest;

function processStateChange(){
if (request.readyState == 4){
contentDiv = document.getElementById(dest);
if (request.status == 200){
response = request.responseText;
contentDiv.innerHTML = response;http://httpd.apache.org/download.cgi
}
}
}

function loadHTML(URL, destination){

当 XML HTTP 对象到达状态 4 时,内容就已经就绪,可以提取并在浏览器的预期位置上显示。位置是 contentDiv,内容从文档中检索。如果请求是正确的,而且也按正确的顺序检索,那么响应的状态应当是 200。HTML 响应保存在 request.responseText 中,把它设置到 contentDiv.innerHTML,就可以在浏览器中显示它。

如果在传输过程中没有发生错误,一切正常,那么新内容就会在浏览器中出现;否则,request.status 就不等于 200。请参阅清单 5 了解错误处理代码。

清单 5. 错误处理


if (request.status == 200){
response = request.responseText;
contentDiv.innerHTML = response;
} else {
contentDiv.innerHTML = “Error: Status “+request.status;
}

清单 5 会把有关传输错误的信息发送给浏览器。在示例应用程序中将把这个函数用作回调函数。接下来,学习 GET 和 POST 的问题和它们的差异。

GET 与 POST

GET 和 POST 是进行 HTTP 请求和在请求中传递变量的两种方法。开发人员不应当随意选择使用哪种方法,因为两种方法都有使用意义。GET 请求把变量嵌在 URL 中,这意味着它们是可以加书签的。如果变量可能会修改数据库、购买商品或者做其他类似操作,这种方法会有副作用。假设偶然间加了书签的页面有一个购买商品的 URL ,里面包含地址、信用卡号以及 $100 的产品,全都嵌在 URL 中,那么重新访问这个 URL 就意味着购买这件商品。

所以,只有在变量没有副作用,可以经常重新载入也不会发生什么的时候,才可以进行 GET 请求。适合 GET 请求的一个变量可能是分类 ID。可以反复重新载入,分类会反复显示,但是没有任何累积的效果。

在另一方面,当变量对源(例如数据库)有作用时或者为了个人信息安全,应当采用 POST 请求。比如在购买一件 $100 的产品时,应当用 POST 请求。如果给付款页面加了书签,由于 URL 中没有变量,所以什么也不会发生,也不会意外地购买了不想购买的东西,或者在已经拥有的情况下又买了一次。

GET 和 POST 的意义在 Ajax 中有同样的作用。在构建本文的应用程序和未来的应用程序时,理解 GET 和 POST 请求的差异是很重要的。这会有助于避免 Web 应用程序开发的常见缺陷。

编码方法

对于通过 HTTP 传输的数据进行编码,有多种方法,而 XML 只接受其中少数几种。互操作性最大的一种编码方法是 UTF-8,因为它向后兼容美国信息交换标准码(ASCII)。有许多在其他国家使用的国际字符的编码方式不向后兼容 ASCII ,如果不进行恰当的编码,就不适合放在 XML 文件中。

例如,把字符串 ”Internationalization” 放在浏览器中,用 UTF-8 编码的话,会把它变成 I%F1t%EBrn%E2ti%F4n%E0liz%E6ti%F8n。经典 ASCII 字符的 UTF-8 编码与相同字符的 7 位 ASCII 码对应,这使 UTF-8 是一个理想的编码方法选择。

了解这一点是很重要的,因为在通过 HTTP 传输和接收文档的过程中都要处理编码问题,在使用 Ajax 时也是一样。使用 Ajax 进行传输时,也应当使用 UTF-8 编码,因为标准化可以提高互操作性。

php

基于PHP和AJAX创建RSS聚合器

Tags: , , ,

想象使用一个简单HTML文件来把一个请求发送到一个服务器端脚本,收到一个基于该请求的定制XML文件,然后把它显示给用户而几乎不需要刷新浏览器!本文作者将同你一起探讨怎样在普通Web应用程序中联合javascript:;” onClick=”javascript:tagshow(event, ‘PHP’);” target=”_self”>PHP和AJAX技术来创建实时的数据传输而不需要进行浏览器刷新。

尽管本文所使用的是PHP语言,但是请记住任何服务器端语言都会正常工作。为了理解本文,我假定你基本理解JavaScript和PHP或一类似服务器端语言。

本文示例使用AJAX来把一请求从一个RSS馈送发送到一定制的PHP对象。该PHP对象复制一份在本地服务器上的该馈送并返回这一路径。该请求对象收到这一路径,分析它,并且把数据以HTML形式显示给用户。这听起来涉及很多步骤,其实它仅由4个小文件组成。之所以使用了4个小文件,是为了平衡它们各自特定的力量而使整个系统的处理极富效率性。

有些读者可能会问,为什么你要创建在本地服务器上的馈送的一个副本而不是简单分析最原始的馈送。原因是,这样以来可以允许绕过XML HTTP Request对象所强加的跨域限制。后面,我还会解释怎样创建这个定制的PHP对象;但是首先,让我们从表单创建开始。

创建发出请求的表单

你要做的第一事情是,在你的HTML的head标签之间包括你可能想使用的JavaScript和任何CSS文件。我包括了一个式样表来实现该聚合器的最后布局并用一个JavaScript文件来发出请求和进行馈送分析:

<link href=”css/layout.css” rel=”stylesheet” type=”text/css” />
<script src=”js/request.js”></script>

下一步,创建一个表单,它针对你所选择的一个RSS馈送发出请求。我创建的表单只包括一个输入字段和一个提交该请求的按钮。该请求的查询是一个字符串,它由馈送输入值和一个将在服务器端被校验的口令字组成;作为一个示例,我使用了下面形式:
“password=mypassword

该代码在每次页面加载之时发出一次请求;因此,如果页面被刷新,现有的在该输入域中的馈送串将在页面加载时被请求。下面是一个表单数据的示例,连同一些div标签用来显示已分析的馈送的特定结点:

<body onload=”javascript:makeRequest(’request.php?request=’ + document.feedForm.feed.value + ‘”password=mypassword’);”>
<form name=”feedForm” method=”post” action=”javascript:makeRequest(’request.php?request=’ + document.feedForm.feed.value + ‘”password=mypassword’);”>
Enter a feed: <input type=”text” name=”feed” id=”feed” size=”20″>
<input type=”submit” name=”submit” value=”Add Feed”>
</form>
<div id=”logo”></div>
<hr/>
<div id=”copy”></div>
<div id=”details”></div>
</body>

我所创建的这三个div标签是logo,copy和details,其中每一个都在布局样式表中有一个与之相关联的样式。当我们分析馈送时将会用到它们,但是我们首先需要能够存取我们所请求的馈送。这可以使用我前面所提到的PHP对象来完成。

创建定制的PHP对象

我用PHP创建了一个小型RSS类,它在本地服务器上创建一个请求馈送的副本,这样它可以为我们稍后要创建的XML HTTP Request对象所存取。典型地,你不能跨域请求一个文件,这意味着你要请求的文件需要位于本地服务器上。这个类是一种解决跨域问题的办法,因为它创建该馈送的一个副本,这个副本在本地服务器上被请求并且把本地路径返回到该馈送,然后它由该Request对象来存取。

这个类中唯一的方法是一个请求方法,它仅有一个指向所请求的RSS 馈送的URL的参数。然后,它通过rss的名字来检查是否一目录位于本地服务器上。如果不存在,就创建一个并把其权限模式设置为0666,这意味着该目录 可读写。当被设置为可读的时,该目录就可以在以后被存取;而当被设置为可写的时,就可以把该馈送的一个副本写向本地服务器上的目录:

//如果不存在目录就创建一个
$dir = “rss”;
if(!is_dir($dir))
{
mkdir($dir, 0666);
}

注意

在一台Windows机器上,对于PHP 4.2.0及以上版本中模式设置是不被要求的。但是,如果它存在的话,它将被忽略;因此,我保留了它,以备该工程被迁移到一台UNIX或Linux服务器上。

在把馈送复制到该服务器前,我们需要一个唯一的文件名。我对这个完整的URL使用了md5加密方法以确保所有馈送的名字是唯一的。通过这个新的文件名,它可以连接一个描述指向该文件的目录的字符串;这将在创建该馈送的副本时使用:

//创建唯一的命名
$file=md5($rss_url);
$path=”$dir/$file.xml”;

通过使用被定义在上面的路径和到原始的被请求的馈送的URL的参考,现在我们能创建该文件的一个副本。最后,把该路径返回到该新文件,作为对该请求的响应:

//复制馈送到本地服务器
copy($rss_url,”$path”);
return $path;
Following is the small, yet powerful RSS class in its entirety:
<?php
class RSS
{
function get($rss_url)
{
if($rss_url != “”)
{
//如果不存在目录就创建一个
$dir = “rss”;
if(!is_dir($dir))
{
mkdir($dir, 0666);
}
// 创建一个唯一的名字
$file = md5($rss_url);
$path = “$dir/$file.xml”;
//复制馈送到本地服务器
copy($rss_url, “$path”);
return $path;
}
}
}
?>

为了存取该PHP类中的方法,需要有一个请求文件来担当到该类的一个接口,这也正是我们正在请求的文件。这个文件首先验证从该请求查询的一口令变量,或 者返回一条指定该请求者不是一名经授权的用户的消息,或者用指向RSS馈送(该馈送在由请求方法处理后被复制到本地服务器)的路径作出响应。为了响应该 RSS馈送,需要包含这个RSS对象并把它实例化,并且需要通过使用被请求的馈送的URL作为一参数来激活请求方法:

<?
if($password == “mypassword”)
{
require_once(’classes/RSS.class.php’);
$rss = new RSS();
echo $rss->get($request);
}
else
{
echo “You are an unauthorized user”;
}
?>

GET/POST与AJAX相结合

为了POST请求,我们首先需要创建该请求对象。如果你没有创建请求对象的经验,那么可以读一下我的文章《How To Use AJAX》或简单地研究一下本文的示例源代码。一旦创建该请求对象,就可以调用sendFeed方法并传递由表单所创建的URL:

function sendFeed(url){
post.onreadystatechange = sendRequest;
post.open(”POST”, url, true);
post.send(url);
}

一旦收到来自于PHP对象的响应并被正确加载,则对与该响应相应的本地文件发出另一个请求。在这种情况中,post.responseText提供给我们该新文件的路径:

function sendRequest(){
if(checkReadyState(post)){
request = createRequestObject();
request.onreadystatechange = onResponse;
request.open(”GET”, post.responseText, true);
request.send(null);
}
}

分析响应

由于RSS馈送之间的区别,分析响应具有一定的挑战性。一些含有包含标题和描述结点的图像,而其它则没有。因此,当我们分析回馈时,我们需要做一点检查来译解它是否包括一图像。如果它包括一图像,我们就可以,与该馈送的标题和链接一起,在image div标签中显示该图像:

var _logo = “”;
var _title = response.getElementsByTagName(’title’)[0].firstChild.data;
var _link = response.getElementsByTagName(’link’)[0].firstChild.data;;
_logo += “<a href=’” + _link + “‘ target=’_blank’>” + _title + “</a><br/>”;
if(checkForTag(response.getElementsByTagName(’image’)[0]))
{
var _url = response.getElementsByTagName(’url’)[0].firstChild.data;
_logo += “<img src=’” + _url + “‘ border=’0′><br/>”
}
document.getElementById(’logo’).innerHTML = _logo;

我们不仅必须检查每个图像以显示它,当遍历馈送中所有的项时我们还需要对之进行检查。因为如果存在一个图像,那么所有另外的标题和链接结点索引都将无法正常工作。因此,当发现图像标签时,我们应该通过在每一次遍历中增加索引值(+1)来调整标题和链接结点的索引:

if(checkForTag(response.getElementsByTagName(’image’)[0]) “” i>0){
var _title=response.getElementsByTagName(’title’)[i+1].firstChild.data;
var _link=response.getElementsByTagName(’link’)[i+1].firstChild.data;
}
else{
var _title =response.getElementsByTagName(’title’)[i].firstChild.data;
var _link = response.getElementsByTagName(’link’)[i].firstChild.data;
}

你可以使用checkForTag方法来检查是否存在特定的标签:

function checkForTag(tag){
if(tag != undefined) {
return true;
}
else{
return false;
}
}

存在许多种进行馈送分析的可能性。例如,你可以把项赋到类别上并使得该类别可折迭,这样用户就可以对其想观看的内容进行选择。作为一个示例,我使用日期 来对项进行分类-这可以通过译解是否针对一个特定项的pubDate不同于前一个项的pubDate并且相应地显示一新的日期来实现:

if(i>1){
var previousPubDate = response.getElementsByTagName(’pubDate’)[i-1].firstChild.data;
}
if(pubDate != previousPubDate || previousPubDate == undefined){
_copy += “<div id=’detail’>” + pubDate + “</div><hr align=’left’ width=’90%’/>”;
}
_copy += “<a href=\”javascript:showDetails(’” + i + “‘);\”>” + _title + “</a><br/><br/>”;
document.getElementById(’copy’).innerHTML += _copy;

注意,上面的最后一部分是showDetails方法,它用于当一用户从一个馈送中选择一特定的项时进行细节显示。这个方法有一个参数(项索引值),这个索引用于发现在该馈送中details结点的索引:

function showDetails(index){
document.getElementById(’details’).innerHTML = response.getElementsByTagName(’description’)[index].firstChild.data;
}

结论

使用AJAX发送查询字符串到一个服务器端脚本并检索一个基于该串的定制响应,这对于任何web开发者都有实现的可能。这样以来,你的下一个web应用程序也将会充满了新的可能性。

php

2009/11/15

AJAX 技术在 PHP 中的简单使用(1 )

Tags: , ,

AJAX无疑是2005年炒的最热的Web开发技术之一,当然,这个功劳离不开Google。我只是一个普通开发者,使用AJAX的地方不是特别多,我就简单的把我使用的心得说一下。(本文假设用户已经具有JavaScript、HTML、CSS等基本的Web开发能力)

[AJAX介绍]

Ajax是使用客户端脚本与Web服务器交换数据的Web应用开发方法。Web页面不用打断交互流程进行重新加裁,就可以动态地更新。使用Ajax,用户可以创建接近本地桌面应用的直接、高可用、更丰富、更动态的Web用户界面。

异步JavaScript和XML(AJAX)不是什么新技术,而是使用几种现有技术——包括级联样式表(CSS)、JavaScript、XHTML、XML和可扩展样式语言转换(XSLT),开发外观及操作类似桌面软件的Web应用软件。

[AJAX执行原理]

一个Ajax交互从一个称为XMLHttpRequest的JavaScript对象开始。如同名字所暗示的,它允许一个客户端脚本来执行HTTP请 求,并且将会解析一个XML格式的服务器响应。Ajax处理过程中的第一步是创建一个XMLHttpRequest实例。使用HTTP方法(GET或 POST)来处理请求,并将目标URL设置到XMLHttpRequest对象上。

当你发送HTTP请求,你不希望浏览器挂起并等待 服务器的响应,取而代之的是,你希望通过页面继续响应用户的界面交互,并在服务器响应真正到达后处理它们。要完成它,你可以向 XMLHttpRequest注册一个回调函数,并异步地派发XMLHttpRequest请求。控制权马上就被返回到浏览器,当服务器响应到达时,回调 函数将会被调用。

[AJAX实际应用]

1. 初始化Ajax

Ajax实际上就是调用了XMLHttpRequest对象,那么首先我们的就必须调用这个对象,我们构建一个初始化Ajax的函数:

/**
* 初始化一个xmlhttp对象
*/
function InitAjax()
{
var ajax=false;
try {
ajax = new ActiveXObject(”Msxml2.XMLHTTP”);
} catch (e) {
try {
ajax = new ActiveXObject(”Microsoft.XMLHTTP”);
} catch (E) {
ajax = false;
}
}
if (!ajax && typeof XMLHttpRequest!=’undefined’) {
ajax = new XMLHttpRequest();
}
return ajax;
}

你也许会说,这个代码因为要调用XMLHTTP组件,是不是只有IE浏览器能使,不是的经我试验,Firefox也是能使用的。

那么我们在执行任何Ajax操作之前,都必须先调用我们的InitAjax()函数来实例化一个Ajax对象。

2. 使用Get方式

现在我们第一步来执行一个Get请求,加入我们需要获取 /show.php?id=1的数据,那么我们应该怎么做呢?

假设有一个链接:<a href=”/show.php?id=1″>新闻1</a>,我点该链接的时候,不想任何刷新就能够看到链接的内容,那么我们该怎么做呢?

//将链接改为:
<a href=”#” onClick=”getNews(1)”>新闻1</a>
//并且设置一个接收新闻的层,并且设置为不显示:
<div id=”show_news”></div>

同时构造相应的JavaScript函数:

function getNews(newsID)
{
//如果没有把参数newsID传进来
if (typeof(newsID) == ‘undefined’)
{
return false;
}
//需要进行Ajax的URL地址
var url = “/show.php?id=”+ newsID;
//获取新闻显示层的位置
var show = document.getElementById(”show_news”);
//实例化Ajax对象
var ajax = InitAjax();
//使用Get方式进行请求
ajax.open(”GET”, url, true);
//获取执行状态
ajax.onreadystatechange = function() {
//如果执行是状态正常,那么就把返回的内容赋值给上面指定的层
if (ajax.readyState == 4 && ajax.status == 200) {
show.innerHTML = ajax.responseText;
}
}
//发送空
ajax.send(null);
}

那么当,当用户点击“新闻1”这个链接的时候,在下面对应的层将显示获取的内容,而且页面没有任何刷新。当然,我们上面省略了show.php这个文件,我们只是假设show.php文件存在,并且能够正常工作的从数据库中把id为1的新闻提取出来。

这种方式适应于页面中任何元素,包括表单等等,其实在应用中,对表单的操作是比较多的,针对表单,更多使用的是POST方式,这个下面将讲述。

php

利用PHP+JavaScript打造AJAX搜索窗(4)

Tags: , ,

八、 定制Web搜索框

借助于CSS,你可以容易地为你的现有站点定制搜索框,并且使你以后的任何重新设计都变得非常容易。

首先要讨论的CSS类是ajaxWebSearchBox(该类实现搜索框)。因为搜索框要确定位置,所以它必须要有一个绝对位置:

.ajaxWebSearchBox
{
position: absolute;
background-color: #0d1e4a;
width: 500px;
padding: 1px;
}

在此,绝对位置是唯一的要求。所有的其它属性都是根据你的口味可选的。在这个示例中,该框有一个微蓝色的背景,一个500像素的宽度,以及在四边上各有1个像素的填充。这个填充导致围绕该框的内容的是1个像素宽的边界。

下一个类是ajaxWebSearchHeading,它包含该框的头部文本和关闭链接。为了把关闭链接放在右上角,它使用绝对位置。因为这个原因,它要求ajaxWebSearchHeading使用一个相对位置:

.ajaxWebSearchHeading
{
position: relative;
background-color: #1162cc;
font: bold 14px tahoma;
height: 21px;
color: white;
padding: 3px 0px 0px 2px;
}

在此,唯一要求的属性也是position属性。其它的属性有助于给出该元素一个好看的外观。其背景颜色是浅蓝色,而文本部分是白色,14像素高且是Tahoma字体。该元素的高度是21个像素并且在顶部和左边都填充以边界。

如前面所述,该关闭链接的位置是绝对的:

a.ajaxWebSearchCloseLink
{
position: absolute;
right: 5px;
top: 3px;
text-decoration: none;
color: white;
}
a:hover.ajaxWebSearchCloseLink
{
color: red;
}

该元素被放置在距右边5个像素,距顶部3个像素的位置(该元素被放在右上角)。这个链接没有任何文本修饰并且颜色呈白色。当用户的鼠标停在该链接上时,文本颜色变红。

注意,这里没有使用访问过的或活动的”假”类。这是因为该窗口总是忽略这个链接的href属性(它已经在它的事件处理器中返回了false)。因此,该链接从来不会真正处于活动或被访问状态。

然后,ajaxWebSearchResults类使结果栏的风格如下:

.ajaxWebSearchResults
{
background-color: #d3e5fa;
padding: 5px;
}

这个元素不要求使用CSS属性。现有属性仅用于定义结果栏并且使它比较容易读取。背景颜色是一个浅蓝色并且围绕边缘有5个像素的填充。当然,你能够定制加载消息的风格:

.ajaxWebSearchResults div
{
text-align: center;
font: bold 14px tahoma;
color:#0a246a;
}

这个元素没有一个类名,但是你仍然能够通过使用前面的示例中展示的parent child标志控制它的风格。这个示例把文本放置在<div/>元素的中央,并且给它一个加粗蓝色的字体,且有14个像素高。

最后一个你需要风格化的元素是结果链接。这些链接有一个类名叫ajaxWebSearchLink:

a.ajaxWebSearchLink
{
font: 12px tahoma;
padding: 2px;
display: block;
color: #0a246a;
}
a:hover.ajaxWebSearchLink
{
color: white;
background-color: #316ac5;
}
a:visited.ajaxWebSearchLink
{
color: purple;
}

唯一要求的属性是display属性(被设置为block)。这使每一个链接都能够在它自己的行上显示。填充空白部分大约有两个像素宽,使各个链接之间 分开一些,从而使它们更易于读取。字体名为Tahoma并且有12像素高。它们的颜色是暗蓝色,与ajaxWebSearchResults的浅蓝色背景 形成对照。

当用户在这些链接上移动鼠标时,背景颜色被设置为蓝色,而文本颜色改变为白色。

在前面的代码的最后一条规则中访问过的”假”类被设置。这是为了给用户提供用户接口暗示-它们已经被使用过。通过把访问过的”假”类设置为显示一种紫色,用户就可以知道它们已经访问过那个链接,从而节省他们的时间-不必再访问一个他们可能不想看的页面。

现在,让我们来看一下如何实现搜索框。

九、 实现Web搜索搜索框

实现这个搜索框是很简单的。首先,你必须把websearch.php文件上传到你的web服务器(当然,必须安装PHP)。然后,你需要一个HTML 文档来引用所有的组件。msnWebSearch对象依赖于XParser类,这个类又依赖于zXml库(可从 www.nczonline.net/downloads/下载)。你必须引用下面这些文件:

<!DOCTYPE html PUBLIC “-//W3C//DTD XHTML 1.0 Transitional//EN”
“http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd”>
<html xml:lang=”en” lang=”en” xmlns=”http://www.w3.org/1999/xhtml”>
<head>
<meta http-equiv=”Content-Type” content=”text/html; charset=utf-8″ />
<title>Ajax WebSearch</title>
<link rel=”stylesheet” type=”text/css” href=”css/websearch.css” />
<script type=”text/javascript” src=”js/zxml.js”></script>
<script type=”text/javascript” src=”js/xparser.js”></script>
<script type=”text/javascript” src=”js/websearch.js”></script>
</head><body>
</body>
</html>

为了执行搜索,应该把msnWebSearch.search()方法设置为该元素的onclick处理器:

<!DOCTYPE html PUBLIC “-//W3C//DTD XHTML 1.0 Transitional//EN”
“http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd”>
<html xml:lang=”en” lang=”en” xmlns=”http://www.w3.org/1999/xhtml”>
<head>
<meta http-equiv=”Content-Type” content=”text/html; charset=utf-8″ />
<title>Ajax WebSearch</title>
<link rel=”stylesheet” type=”text/css” href=”css/websearch.css” />
<script type=”text/javascript” src=”js/zxml.js”></script>
<script type=”text/javascript” src=”js/xparser.js”></script>
<script type=”text/javascript” src=”js/websearch.js”></script>
</head><body>
<a href=”#” onclick=’msnWebSearch.search(event,”\”Professional Ajax\”");
return false;’>Search for “Professional Ajax”</a>
<br /><br /><br /><br />
<a href=”#” onclick=’msnWebSearch.search(event,”Professional Ajax”);
return false;’>Search for Professional Ajax</a>
</body>
</html>

第一个新的链接执行一个针对准确词组”Professional Ajax”的搜索,而第二个链接将搜索这其中的各个单词。还要注意,在onclick事件中返回的是false-这强迫浏览器忽略掉href属性。点击这 些链接将在光标位置绘制搜索框,并且就在此处显示你的搜索结果。

php

利用PHP+JavaScript打造AJAX搜索窗(3)

Tags: , ,

五、 显示结果

populateResults()方法负责使用搜索结果填充结果栏。它接受两个参数:包含该结果的元 素和一个XParser对象(XParser是一个基于Javascript的RSS读取器,可从 www.wdonline.com/javascript/xparser/自由下载):

msnWebSearch.populateResults = function(divResultsPane,oParser){
var oFragment = document.createDocumentFragment();

divResultsPane.removeChild(divResultsPane.firstChild);

这个方法以编程方式并通过DOM方法生成<a/>元素;这样以来,这些元素将被添加到一个在第一行创建的文档片断中。下一行删除添加在drawResultBox()中的正加载的<div/>元素。

下一步是创建这个链接:

for (var i = 0; i < oParser.items.length; i++) {
var oItem = oParser.items[i];

var aResultLink = document.createElement(”a”);
aResultLink.href = oItem.link.value;
aResultLink.className = “ajaxWebSearchLink”;
aResultLink.target = “_new”;
aResultLink.appendChild(document.createTextNode(oItem.title.value));

oFragment.appendChild(aResultLink);
}

这段代码遍历回馈的各个项,并且由该数据生成链接并把<a/>元素添加到文档片断最后。

当退出循环时,该文档片断被添加到divResultsPane以显示搜索结果:

divResultsPane.appendChild(oFragment);

六、 关闭结果框

为了关闭搜索结果框,msnWebSearch对象提供了close()方法。close()方法负责处理该链接的onclick事件(关闭该小框):

msnWebSearch.close = function () {
var divSearchBox = this.parentNode.parentNode;
document.body.removeChild(divSearchBox);

return false;
};

该搜索框其实并没有关闭;事实上,它被从该文档中删除了。为此,需要检索divSearchBox元素。第一行代码完成这一任务-通过检索这个元素的父 结点的父结点实现。因为close()负责处理onclick事件,所以this引用这一链接。下一行从文档中删除divSearchBox元素。最后一 行,返回false,从而强迫浏览器不要沿用一个链接的缺省行为(转到在href属性中标注的位置)。

七、 构建搜索接口

msnWebSearch对象的最后一个方法是search(),它提供执行一个搜索的接口。你可以使用一个元素的onclick事件来调用search()。它接受两个方法:一个事件对象和搜索术语:

msnWebSearch.search = function (e,sSearchTerm) {
var divSearchBox = this.drawResultBox(e);
var url = encodeURI(”websearch.php?search=” + sSearchTerm);
var oParser = new XParser(url);
oParser.onload = function () {
msnWebSearch.populateResults(divSearchBox.childNodes[1],oParser);
};
};

第一行调用drawResultBox()方法并且把事件e传递给它。下一行编码该URL以实现合适的转换。这个URL被传递给XParser构造器以 创建一个新的分析器。当搜索回馈完成加载并使用结果填充搜索框时,该分析器的onload事件处理器调用populateResult()方法。
当然,构建这个搜索框的一个理由是,使其更适合于你自己的站点的外观。

php

利用PHP+JavaScript打造AJAX搜索窗(2)

Tags: , ,

四、 绘制结果用户接口

生成这个HTML的代码相当长,因为其中的元素都是使用DOM方法生成的。drawResultBox()方法接受一个参数(一个事件对象):

msnWebSearch.drawResultBox = function (e) {
var divSearchBox= document.createElement(”div”);
var divHeading = document.createElement(”div”);
var divResultsPane = document.createElement(”div”);
var aCloseLink = document.createElement(”a”);

前面这些代码经由createElement()方法创建HTML元素。在创建这些元素后,你就能够开始赋予它们属性。上面完成终结(封尾)的两个元素分别是aCloseLink和divHeading:

aCloseLink.href = “#”;
aCloseLink.className = “ajaxWebSearchCloseLink”;
aCloseLink.onclick = this.close;
aCloseLink.appendChild(document.createTextNode(”X”));
divHeading.className = “ajaxWebSearchHeading”;
divHeading.appendChild(document.createTextNode(”MSN Search Results”));
divHeading.appendChild(aCloseLink);

前四行完成关闭结果框的链接。其中,方法close()成为链接的onclick事件的处理器。后面的几行代码负责使用文本和关闭链接填充头部的<div/>。

当这个结果框被绘制到页面上时,还没有接收到来自于一个服务器应用程序的响应。为了向用户展示已经发生了什么,可以向用户展示一个消息提示数据正在加载中(这种方式更友好些)(见图2)。为此,创建另一个元素并且把它添加到divResultsPane元素:

var divLoading = document.createElement(”div”);
divLoading.appendChild(document.createTextNode(”Loading Search Feed”));

divResultsPane.className = “ajaxWebSearchResults”;
divResultsPane.appendChild(divLoading);

这个代码创建加载消息并且把它添加到divResultsPane,同时还把类名赋给divResultsPane。


图2.向用户提示数据正在加载中

完成这些元素之后,剩下的就是把它们添加到divSearchBox元素中:

divSearchBox.className = “ajaxWebSearchBox”;
divSearchBox.appendChild(divHeading);
divSearchBox.appendChild(divResultsPane);
document.body.appendChild(divSearchBox);

这段代码负责把divHeading和divResultsPane元素添加到搜索窗,并且把搜索窗添加到页面。

在drawResultBox()中的最后一步是确定新绘制的小框的位置并且把divSearchBox返回到它的调用者:

msnWebSearch.drawResultBox = function (e) {
var divSearchBox= document.createElement(”div”);
var divHeading = document.createElement(”div”);
var divResultsPane = document.createElement(”div”);
var aCloseLink = document.createElement(”a”);
aCloseLink.href = “#”;
aCloseLink.className = “ajaxWebSearchCloseLink”;
aCloseLink.onclick = this.close;
aCloseLink.appendChild(document.createTextNode(”X”));
divHeading.className = “ajaxWebSearchHeading”;
divHeading.appendChild(document.createTextNode(”MSN Search Results”));
divHeading.appendChild(aCloseLink);
var divLoading = document.createElement(”div”);
divLoading.appendChild(document.createTextNode(”Loading Search Feed”));
divResultsPane.className = “ajaxWebSearchResults”;
divResultsPane.appendChild(divLoading);
divSearchBox.className = “ajaxWebSearchBox”;
divSearchBox.appendChild(divHeading);
divSearchBox.appendChild(divResultsPane);
document.body.appendChild(divSearchBox);
this.position(e, divSearchBox);
return divSearchBox;
};

通过这种方式建立msnWebSearch对象后,必须把divSearchBox返回到它的调用者以便进行其它操作。你可以已经猜出, position()方法负责放置该搜索框。它接受两个参数:传递到drawResultBox()的事件对象和divSearchBox元素:

msnWebSearch.position = function (e, divSearchBox) {
var x = e.clientX + document.documentElement.scrollLeft;
var y = e.clientY + document.documentElement.scrollTop;
divSearchBox.style.left = x + “px”;
divSearchBox.style.top = y + “px”;
};

前两行代码得到左边和顶部位置,用于放置搜索结果框。执行这个操作要求使用两种信息。首先是鼠标的x和y坐标(这些信息被存储在clientX和clientY属性)。

然而,这些坐标还不足以正确定位结果框,因为clientX和clientY属性返回相对于浏览器窗口客户区的鼠标位置,而不是页面中的实际坐标。考虑 到这一点,我们可以使用文档元素的scrollLeft和scrollTop属性。计算出最后的坐标后,你能够最后确定用户点击鼠标的框中的位置。

php

利用PHP+JavaScript打造AJAX搜索窗(1)

Tags: , , ,

一、 引言

Web世界中一项广为使用的功能就是搜索。随着Web技术的日益发展,为了更好地满足客户的需求,常规搜索 引擎开始对更多的非常规方式”敞开了大门”。在这方面,Yahoo!率先提供出其Y!Q服务。这个新的服务能够使你搜索任何web页面,前提是该页面的作 者必须包括在他们的web页面中。是服务技术实现了把相关的搜索结果呈现到读者眼前,从而向读者展示更多的信息而不必离开他们的当前页面。

Yahoo!的Y!Q服务的确是一个伟大的想法,但是它的出现也受到了一些批评。原因何在?首先,它要求客户端必须使用Yahoo!的 Javascript并且你必须添加一个<form/>元素,以便满足Yahoo!的搜索要求。对于许多网站作者来说,提供该服务要求付出太多的努力。并 且,在所有这些条件满足之后,搜索结果将以Yahoo!风格展现,从而破坏了用户网站的外观感觉。

幸运的是,Yahoo!并非唯一的 提供”从你的网站提供搜索结果”服务的搜索引擎。MSN Search也提供一种类似服务,除了它能够使web开发者控制外观感觉以外。这种能力来自于MSN Search提供它的搜索结果的RSS版本,从而使得订阅一个特定的搜索或使用Ajax方法把该结果添加到你的页面成为可能。

尽管Google已经率先实现了这种新的”从你的站点进行搜索”技术;但是,在本文成文之时,与Google相关的Google BlogSearch Beta也已经能够以RSS或者Atom格式提供返回的结果。

二、 服务器端组件

使用MSN Search执行搜索时,你会看到一个桔黄色的XML图像出现在结果页面的底部。点击这个图像将会把你带到一个新的页面,并提供给你订阅该搜索的URL。

这样以来,你就可以编写服务器端代码以检索远程馈送。对于本文中的搜索窗,你将使用PHP检索搜索馈送。来自于服务器应用程序请求信息的URL看起来如下所示:

websearch.php?search=[SEARCHTERM]

查询字符串只有一个变量:”search”;因此,应用程序应该查找此查询项。在服务器端,你需要创建一个页面来”拉动”这个数据:

<?php
header(”Content-Type: text/xml”);
header(”Cache-Control: no-cache”);if ( isset($_GET["search"]) )
{
$searchTerm = urlencode( stripslashes($_GET["search"]) );
$url = “http://search.msn.com/results.aspx?q=$searchTerm&format=rss”;
$xml = file_get_contents($url);
echo $xml;
}
?>

前两行设置要求的头部以便浏览器能够正确处理数据(XML形式,并且没有对结果予以缓冲)。下一行代码使用isset()函数来决定是否搜索键存在于查询字符串中。

为了把一个合适的请求发送到远程主机,搜索术语应该通过许多函数的”过滤”。首先,它被传递给stripslashes()函数。如果在PHP配置中启 动了”magic quotes”(缺省情况下是支持的),那么,到达PHP引擎的任何引号都会被使用一个斜杠(如,\”search query\”)自动地脱掉。该stripslashes()函数负责删除这些符号,仅留下”search query”。在删除斜杠后,转到urlencode()函数,由这个函数负责编码字符以用于查询字符串。空格、引号、”&”等符号都被编码。

注意 如果搜索术语不能通过这些函数的”过滤”,那么,MSN服务器将返回一个代码400-”Bad Request”。

当为转换搜索术语准备好后,它被包括到URL中并存储于$url变量中。最后,file_get_contents()函数负责打开远程文件,读取其内容并以一个字符串形式把它返回到$xml变量中,然后使用echo命令把它打印到页面中。

三、 客户端组件

本文中的搜索窗口(Widget)的客户端代码基于一个静态对象msnWebSearch创建-它被定义为一个没有任何属性(现在)的对象

var msnWebSearch ={};

这个对象用于一个HTMLElement的onclick事件中执行搜索:

<a href=”#”
onclick=’msnWebSearch.search(event,”Professional Ajax”); return false;’>
Professional Ajax
</a>

这个msnWebSearch对象提供若干取得搜索结果的方法,并负责绘制和放置包含这些数据的HTML。第一个方法是drawResultBox(),它负责绘制HTML。这个方法形成的HTML如下所示:

<divclass=”ajaxWebSearchBox”>
<div>MSN Search Results
<a href=”#”>X</a>
</div>
<div>
<a target=”_new” />
<a target=”_new” />
</div>
</div>

该结果框分为两部分:一个头部和一个结果栏(见图1)。头部告诉用户这个新的搜索窗包含来自一个MSN搜索的结果。它还包含一个”X”用于关闭该小窗口。其结果栏包含块风格的链接,当点击这些链接时将打一个新的窗口。


图1.结果框分为两部分:一个头部和一个结果栏