728x90

부트스트랩에서 사용하는 테이블은 모두 반응형(responsive) 테이블이다.

따라서 브라우저의 길이에 맞춰 테이블의 크기 및 내용 부분 또한 가변적으로 변하게 된다.

테이블 태그 부분에 class="table" 을 적용해주면 width 가 100%인 테이블이 생성된다.


<table class="table">

각 행의 아래에 테두리(border)가 있는 테이블이 만들어진다.


<table class="table table-bordered">

모든 셀에 테두리가 만들어 진다.


<table class="table table-striped">

한 줄 건너 배경색이 달라지는 스트라이프 형태의 테이블이 된다.


<table class="table table-hover">

테이블에 마우스를 올렸을 때 마우스 커서가 있는 행이 다른 색으로 변한다.


<table class="table table-condensed">

셀 패딩이 반으로 감소돼 작은 크기의 테이블이 된다.


테이블에 적용된 클래스 선택자의 경우 혼합해서 사용 가능하다.

<table class="table table-bordered table-hover">


테이블 행 또는 셀에 색상 지정

클래스 설명
.active 특정 행이나 셀에 마우스 오버 색상 적용
색상은 회색
.success 성공 또는 긍정적인 작업 표시
색상은 연두색 계열 #d0e9c6
.info

중립적인 정보 변경이나 작업 표시

.warning 주의를 필요로 하는 경고를 표시
색상은 겨자색 계열 #faf2cc
.danger 위험 또는 잠재적으로 부정적인 작업 표시
색상은 붉은색 계열 #ebcccc


<div class="table-responsive"></div>

부트스트랩에서는 화면 폭이 좁을 때(768px 이하) 하단에 스크롤바가 나타나며 수평으로 스크롤해서 테이블을 볼 수 있게 했다. <table> 태그 외곽에 <div> 태그로 감싸서 table-responsive 클래스를 추가하면 반응형 테이블로 사용할 수 있다.

728x90
블로그 이미지

Link2Me

,
728x90

부트스트랩으로 버튼을 꾸미는 방법



기본 버튼과 부트스트랩 버튼의 차이는 class="btn btn-default" 과 같이 부트스트랩에서 제공하는 class 속성을 추가해주는 것이다.


<div>
    <button type="button">기본</button>
    <button type="button" class="btn btn-default">기본</button>
    <button type="button" class="btn btn-primary">Primary</button>
    <button type="button" class="btn btn-success">성공</button>
    <button type="button" class="btn btn-info">정보</button>
    <button type="button" class="btn btn-warning">경고</button>
    <button type="button" class="btn btn-danger">위험</button>
</div>
<div>
    <button type="button">기본</button>
    <button type="button" class="btn btn-default btn-sm">기본</button>
    <button type="button" class="btn btn-primary btn-sm">Primary</button>
    <button type="button" class="btn btn-success btn-sm">성공</button>
    <button type="button" class="btn btn-info btn-sm">정보</button>
    <button type="button" class="btn btn-warning btn-sm">경고</button>
    <button type="button" class="btn btn-danger btn-sm">위험</button>
</div>
<div>
    <button type="button">기본</button>
    <button type="button" class="btn btn-default btn-xs">기본</button>
    <button type="button" class="btn btn-primary btn-xs">Primary</button>
    <button type="button" class="btn btn-success btn-xs">성공</button>
    <button type="button" class="btn btn-info btn-xs">정보</button>
    <button type="button" class="btn btn-warning btn-xs">경고</button>
    <button type="button" class="btn btn-danger btn-xs">위험</button>
</div>
<div>
    <button type="button">기본</button>
    <button type="button" class="btn btn-default btn-lg">기본</button>
    <button type="button" class="btn btn-primary btn-lg">Primary</button>
    <button type="button" class="btn btn-success btn-lg">성공</button>
    <button type="button" class="btn btn-info btn-lg">정보</button>
    <button type="button" class="btn btn-warning btn-lg">경고</button>
    <button type="button" class="btn btn-danger btn-lg">위험</button>
</div>


728x90
블로그 이미지

Link2Me

,
728x90

DB 암호를 강력하게 만들기 위해서 SHA384를 적용하는 코드를 작성해봤다.


<?php
$userpw = "admin1234"; // 사용자 입력 암호
function createSalt()
{
    $text = md5(uniqid(rand(), TRUE));
    return substr($text, 0, 34);
}

$salt = createSalt(); // 랜덤하게 생성
$password = hash('sha384', $userpw.$salt);
echo $salt.'<br />';
echo $password.'<br />';
echo strlen($password);
?>


DB 패스워드를 다른 키($salt)와 조합하여 저장하도록 하는 로직이다.

여기서 $salt 값이 계속 변하므로 이 키값도 같이 DB에 저장하지 않으면 다음번에 로그인할 때 문제가 생긴다.

따라서 테이블에 key 값을 저장하는 테이블도 하나 추가하는게 좋다.

728x90
블로그 이미지

Link2Me

,
728x90

자바스크립트로 하면 여러줄을 적어줘야 하는데 jQuery 로 하면 심플해진다.

// Assumes element with id='button'
var button = document.getElementById('button');

button.onclick = function() {
    var mydiv = document.getElementById('tabName');
    if (mydiv.style.display === 'block' || mydiv.style.display === '') {
        mydiv.style.display = 'none';
    }
    else {
        mydiv.style.display = 'block';
    }
};

$("#button").click(function() {
    // assumes element with id='button'
    $("#tabName").toggle();
});


메뉴만들기

열심히 이 자료 저자료 테스트해보면서 이해하고 내것으로 만들어가고 있는 중이다.

아래 코드를 연습한 소스코드 파일 첨부한다.

이미지 파일은 http://blog.naver.com/hyoyeol/70177051041 블로그에 올려진 파일 이미지 파일을 활용했다.

하지만 코드 내용은 완전 다르다.


topmenu-01.zip

=== index.php ===

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="robots" content="noindex,nofollow"/>
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <title>메뉴만들기</title>
    <link rel="stylesheet" href="css/bootstyle.css" />
    <link rel="stylesheet" href="css/topmenu.css" />
    <link href="bootstrap/css/bootstrap.min.css" rel="stylesheet">
    <script src="http://code.jquery.com/jquery.min.js" ></script>
    <script src="bootstrap/js/bootstrap.min.js"></script>
    <script type="text/javascript">
        var currentName = null;
        $(document).ready(function(){
            tabSetting(); // 탭 초기화 및 설정
        });
       
        function tabSetting() {
            $('.tabPage').hide();
            $('.topmenu a').on('click', function() {
                var tabName = $(this).attr('href');
                if(currentName != tabName){
                    $('.tabPage').hide();
                    currentName = tabName;
                }
                $(tabName).toggle();
            });
        }
    </script>
</head>
<body>
<header>
     <nav class="navbar navbar-fixed-top navbar-full navbar-light">
        <div class="topmenu">
            <a href="#home">Home</a>
            <a href="#tab01">News</a>
            <a href="#tab02">Contact</a>
            <a href="#tab03">About</a>
            <a href="#tab04">Support</a>
            <a href="#tab05">Blog</a>
            <a href="#tools">Tools</a>
            <a href="#base">Base</a>
            <a href="#custom">Custom</a>
            <a href="#more">More</a>
            <a href="#logo">Logo</a>
            <a href="#friends">Friends</a>
            <a href="#partners">Partners</a>
            <a href="#people">People</a>
            <a href="#work">Work</a>
        </div>
     </nav>
</header>
<main class="container-fluid">
     <div id="contents">
        <div id="tab01" class="tabPage">
            <img src="./images/img_01.jpg">
        </div>
        <div id="tab02" class="tabPage">
            <img src="./images/img_02.jpg">
        </div>
        <div id="tab03" class="tabPage">
            <img src="./images/img_03.jpg">
        </div>
        <div id="tab04" class="tabPage">
            <img src="./images/img_04.jpg">
        </div>
        <div id="tab05" class="tabPage">
            <img src="./images/img_05.jpg">
        </div>
    </div>

</main>

</body>
</html>


/// nav 설명

<nav class="navbar navbar-fixed-top navbar-full navbar-light">

.navbar-fixed-top : 상단에 고정

.navbar-fixed-bottom : 하단에 고정


/// 설명 : 화면 부드럽게 나오고 사라지게 하기

$(tabName).toggle(); 는 화면이 부드럽게 나오고 부드럽게 사라지지 않는다.

화면을 부드럽게 나오고 부드럽게 사라지게 하고 싶으면

$(element).is(":visible"); // 화면이 보이는 상태인지 체크

를 사용해서 코드를 변경해주면 된다.


if($(tabName).is(":visible")){
    $(tabName).slideUp();
}else{
    $(tabName).slideDown();
}



=== topmenu.css ===

charset "utf-8";
@import url(http://fonts.googleapis.com/earlyaccess/nanumgothic.css);

body, div, ul, li, table, tr, td, th{margin:0px; padding:0px;}

ul, li{list-style:none;}

body {
    font-family: NanumGothic, '나눔고딕', NanumGothicWeb, "Malgun Gothic",Gulim,sans-serif;
    background: #ddd;
    font-size:10px;
}

div.topmenu {
    border: 1px solid #ccc;
    background-color: #f1f1f1;
    margin:auto;
    overflow: auto;
    white-space: nowrap;
}

div.topmenu a {
    display: inline-block; /* 자동으로 줄바꿈되지 않고 크기를 지정할 수 있음 */
    color: black;
    text-align: center;
    padding: 14px;
    text-decoration: none;
    transition: 0.1s;
}

div.topmenu a:hover {
    background-color: #ddd;
    color:#0000ff;
    cursor:pointer;
    font-weight:bold;
}

/*컨텐츠*/
#contents{
    width:900px;
    margin:auto;
}

#contents div{
    margin:0 0px;
}

/* 화면 해상도에 따른 글자 크기 변경 */
@media only screen and (max-width: 768px) {
    div.topmenu a {
        padding: 12px;
    }
}
@media only screen and (min-width: 768px) and (max-width: 1024px) {
    div.topmenu a {
        padding: 13px;
    }
}
@media only screen and (min-width: 1025px) {
        body { background-color: blue; }
        h1 { color: white; }
}

728x90
블로그 이미지

Link2Me

,
728x90

foreach문은 배열에서만 동작한다.
키(key)와 값(value)로 분리, 변수로 지정 가능하다.
배열이 아니면 php코드 내에 오류메시지를 출력하고, 다차원 배열이나 중첩도 가능하다

<?php
$A = array (
    '2017-01-25' => '가가가',
    '2017-01-26' => '나나나'
);

// 배열에만 동작하는 반복문 foreach문
foreach($A as $key => $value){
    if($key !== date("Y-m-d")) continue;
    echo $value.'<br />';
}

$colors = array("red","green","blue","yellow");
foreach($colors as $key => $value){
    //echo $value.'<br />';
    echo $value.'-'.$colors[1].'<br />';
}


$numItems = count($colors); // 배열 개수 카운트
$i = 0;
foreach($colors as $key=>$value) {
  if(++$i === $numItems) {
    echo "last index!";
  }
}

?>


// 중첩 foreach 문

foreach($samplearr as $key => $item){
   echo $key;
    foreach($item as $detail){
       echo $detail['value1'] . " " . $detail['value2']
     }
 }

728x90

'Web 프로그램 > PHP 문법' 카테고리의 다른 글

HTML Entity List (특수문자 용어)  (0) 2021.11.27
[PHP] PHP 문자열 찾기  (0) 2017.11.14
PHP Class 개념 이해 ★★  (0) 2017.01.10
PHP isset 과 empty 설명  (1) 2016.04.27
PHP 대체 문법  (0) 2016.04.05
블로그 이미지

Link2Me

,
728x90

설치환경

AutoSet9 on Windows10 64bit


설치파일 구하기

https://xdebug.org/download.php  에서 AutoSet9 설치된 버전을 다운로드 한다.


php.ini 환경설정 파일에서 아래 코드를 추가한다.


[xdebug]
zend_extension = "C:\AutoSet9\server\bin\ext\php_xdebug-2.5.0-5.6-vc11-nts-x86_64.dll"
;zend_extension = "C:\AutoSet9\server\bin\ext\php_xdebug-2.5.0-5.6-vc11-nts.dll"


zend_extension 의 윗줄은 64bit 용으로, 아래 줄은 32bit 용으로 받은 거다.

윗줄로 설정을 했을 때는 에러없이 동작한다.


phpstorm과 xdebug 연동하기 URL http://codeigniter-kr.org/bbs/view/tip?idx=8346 에 예시가 잘 나와있다.

breaking point 지정해서 하는 걸 코드를 하나 선택해서 동작 여부를 테스트해봐야겠다.


https://opentutorials.org/course/692/3758 동영상도 보면서 파악을 해봐야겠다.

728x90
블로그 이미지

Link2Me

,
728x90

Aptana Studio 3 에서 선택한 파일을 실행하는 테스트를 하려고 한다.

외부 서버 설정 등을 해서 해당 파일을 찾아서 실행하는 것이 생각보다 쉽지 않다.




위와 같이 설정을 하고 파일을 선택해서 실행하니까 해당 파일이 실행이 바로 바로 된다.

EditPlus 로 코딩을 하는 것은 에러가 날 확률도 높은데다가 매번 파일명을 일일이 지정하는게 너무 너무 귀찮아서 이런 편리한 방법을 좀 찾고 있었다.



위 화면처럼 파일 선택하고 실행하면 바로 해당 파일이 Firefox 브라우저에서 실행되어 결과를 보여준다.


http://localhost/workspace 로 경로를 지정한 이유는 이미 http://localhost 디렉토리 하단에 여러 디렉토리를 만들어서 테스트를 하던 상태라서 Aptana Studio 3 에서 workspace 경로를 localhost 경로로 지정했더니 에러가 발생해서다.

하지만 처음부터 Aptana Studio 3를 설치하고 테스트를 했다면 이런 문제는 없었을 것이다.



http://localhost/workspace 를 바로 http://localhost 로 만드는 방법은

http.conf 환경설정에서 경로를 변경하면 된다.

DocumentRoot "C:/AutoSet9/public_html"
<Directory "C:/AutoSet9/public_html">
    Options FollowSymLinks
    AllowOverride All
    Require all granted
</Directory>

를 찾아서

DocumentRoot "C:/AutoSet9/public_html/workspace"
<Directory "C:/AutoSet9/public_html/workspace">
    Options FollowSymLinks
    AllowOverride All
    Require all granted
</Directory>
로 변경하고 Apache를 재시작하면 홈디렉토리 경로는 변경된다.


이렇게 변경하면 다시 Run Configuration 을 다시 설정해줘야 한다.


이렇게 변경해주면 문제없이 동작된다.


728x90
블로그 이미지

Link2Me

,
728x90

PHP 코드를 입력하고 제대로 동작하는지 테스트를 해보자.


<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<script>
    window.open("http://google.co.kr");
</script>
</head>   
<body>
   
</body>
</html>


이 코드는 PHP는 한줄도 없다. 네이버지식인에 올라온 질문을 보고서 그대로 연습해서 작성한 코드다.

코드 자동완성 기능 연습을 했다.


이 코드를 웹서버에서 제대로 실행되는지 테스트 해보자.


Run As 를 한번만 하고 나면 다음부터는 버튼만 누르면 이 파일이 실행된다.


테스트를 하고 보니 eclipse 에서 Android 폰 개발할 때와 같은 개념으로 동작한다.

단지 환경만 Web Server 환경으로 변경되었다는 것만 다르다.


연습으로 실행을 한 것이라서 ex-01.php 파일로 생성했지만

실제로는 디렉토리 단위로 프로젝트를 생성한다고 가정해야 하므로

index.php 파일이 있어야 하고 연관된 파일들을 생성해서 코드를 작성해야 한다.


그런 환경이라고 가정하면...



처럼 생성해야 맞다.


PHP Web Page 에 프로젝트 단위로 계속 추가되고, 프로젝트 단위로 inital Script 를 설정하면 된다.


728x90
블로그 이미지

Link2Me

,
728x90

Aptan Studio 3 기본 설치와 workspace 경로를 변경했다.

이제 본격적으로 PHP Project 를  생성해보려고 한다.


프로젝트 그룹 생성



생성된 그룹에 파일 추가




PHP Interpreters 설정



내장된 Web 서버 설정



Web Server 를 설정해보자.



Base URL : 마지막에 / 는 절대 붙이지 말라. 테스트 해보니 붙이면 안된다.



Aptana 라는 이름으로는 workspace 경로까지 포함해서 추가했다.




화면 글꼴 변경

화면 글꼴이 너무 작아보여서 16으로 수정했다.



PHP Server 환경 설정




기본적인 환경설정은 끝났다.


728x90
블로그 이미지

Link2Me

,
728x90

Aptana 를 설치하고 나서 보니 AutoSet9 Document Root 디렉토리 밖에 있다보니 테스트를 하는데 애로사항이 계속 발생한다.

그래서 Workspace를 변경했다.




httpd.conf 파일에 설정된

DocumentRoot "C:/AutoSet9/public_html"

하위 디렉토리에 workspace를 변경했다.


다음에는 실제 PHP Project 를 생성하고 환경설정하는 법에 대해 기술하련다.


728x90
블로그 이미지

Link2Me

,
728x90

윈도우버전 Aptana Studio 3.6.1 에서는 아래 과정은 시도하지 않아도 된다는 걸 확인했다.


aptana eclipse plugin installation

http://www.aptana.com/products/studio3/success_plugin.html 에서

http://download.aptana.com/studio3/plugin/install 를 복사(Ctrl + C) 한다.



최신버전이 설치되었다고 나오면서 설치가 안된다.

유투브 동영상 보면서 이런 과정으로 설치된다는 걸 확인해봤다.

728x90
블로그 이미지

Link2Me

,
728x90

책을 다시 보면서 기본 지식을 다시 정리해보고 있다.

하다가 막히면 다시 원점에서 흘려버린 걸 다시 되짚어 가면서 시작하는 것이 도움이 될 수도 있기 때문이다.


모든 언어의 기본은 데이터 타입을 파악하는 것부터 시작한다.

자바스크립트에서는 기본 타입을 제외한 모든 값은 객체다.


자바스크립트는 변수를 선언할 때 var 라는 한가지 키워드를 사용한다.

 - 선언부에 Data Type을 기재하지 않는다.

 - var 선언문이 위치한 영역에 따라 전역/지역변수 결정

 - var 를 사용하지 않으면 전역(global)변수로 선언

var 로 변수를 선언하기 때문에 데이터 타입이 무엇인지 몰라서 헤매는 경우도 생기더라.

+ 는 숫자연산과 문자열 연결에 사용되므로 잘못된 결과가 나올 수 있으니 주의해서 봐야 한다.


숫자

 다른 언어와 달리 하나의 숫자형만 존재한다.

 모든 숫자를 64비트 부동 소수점 형태로 저장한다.

 모든 숫자를 실수로 처리하므로, 나눗셈 연산을 할 때는 주의해야 한다.

  ex) 1/2 = 0.5

 정수만 구하고 싶다면 Math.floor() 메서드를 사용한다.

 + 연산자는 숫자일 경우에는 더하기 연산이 수행된다.

 Number 형 변환 실패 결과는 NaN  

  ex) "abc" - 0 = NaN

 문자열

 작은 따옴표나 큰 따옴표로 생성한다.

 문자열은 문자 배열처럼 인덱스를 이용해서 접근할 수 있다.

 + 연산자는 문자열 연결 연산을 수행한다.

 boolean

 자바스크립트는 true 와 false를 나타내는 boolean 타입을 가진다.

 undefined

 자바스크립트에서 '값이 비어있음'을 나타낸다.

 typeof 연산자 결과는 undefined 이다. 즉, Data Type 이 정해지지 않은 타입

 null

 자바스크립트에서 '값이 비어있음'을 나타낸다.

 typeof 연산자 결과는 object 이다. null 이 아니다. 아무것도 참조하지 않는 값이다.

 객체

 자바스크립트 객체는 key:value 형태의 property들은 저장하는 컨테이너다.

  - 객체 내에서 변수는 속성(property)이라 부른다.

  - 객체 내에서 함수는 메서드(method)라 부른다.

  - 이름은 key라고 불린다.

 객체는 여러 개의 property들을 포함할 수 있다.

 객체 생성

   - 생성자 문법을 이용하여 객체 생성 :
     new 연산자 사용 var hotel = new Object(); // 객체를 위한 공간 생성

     this 키워드 -> window 객체를 나타냄

   - 객체 생성자 표기법을 이용하여 객체를 생성

 객체 property 삭제는 delete 연산자를 이용해서 삭제할 수 있지만 객체 자체는 삭제 못한다.

 상속이란 기존의 생성자 함수나 객체를 기반으로 새로운 생성자 함수나 객체를 쉽게 만드는 것

 비교

 기본 타입은 동등 연산자(==)를 이용해서 비교할 때 값을 비교한다.

 동등 연산자(==)는 피연산자의 타입이 다를 경우 타입 변환을 거친 다음 비교한다.
 참조 타입인 객체 비교는 참조값이 같아야 true 가 된다.

 일치 연산자(===)는 피연산자의 타입이 다를 경우 타입을 변경하지 않고 비교한다.

 배열

 자바스크립트에서 배열은 크기를 지정하지 않아도 된다.
 배열 리터럴은 []를 사용한다.
 동적으로 배열 원소를 추가할 수 있다.
 length 프로퍼티는 배열 내에서 가장 큰 인덱스에 1을 더한 값이다.

 실제 메모리는 length 크기처럼 할당되지 않는다.

 push() 메서드는 인수로 넘어온 항목을 배열의 끝에 추가한다.

 arr.length = 5;

 arr.push('english');

 생성자 함수로 배열을 생성할 때 new 연산자를 사용한다.

 delete arr[3]; // 배열 요소 삭제하라는 의미지만 실제는 해당 요소값을 undefined로 설정한다.


 배열도 객체이므로

  for(var i in arr) {

    console.log(i, arr[i]);

  }

  문을 사용해서 배열 내의 모든 프로퍼티를 열거할 수 있다. --> 불필요한 프로퍼티가 출력될 수 있다.

  for(var i=0; i<arr.length; i++) {

   console.log(i, arr[i]);

  }

  문을 사용하면 불필요한 프로퍼티가 출력되지 않는다.

 함수

 function 키워드로 함수를 선언한다.

 선언부에 Return Type 을 기재하지 않는다.

 자바스크립트에서는 함수도 하나의 값처럼 취급한다.

 함수도 숫자나 문자열처럼 변수에 할당하는 것이 가능하다.

 매개변수로 넘기는 값에 변수 타입을 기술하지 않는다.

 자바스크립트 함수 표현식에서 함수 이름은 꼭 붙이지 않아도 된다. (익명함수)

 함수 내부에서 정의된 매개변수는 var를 사용하여 정의한다.

 함수에 정의된 인자가 3개라고 가정하자.

   - 2개는 정의하나 나머지 하나를 정의하지 않으면 undefined 값이 할당된다.

   - 정의된 개수보다 많게 4개를 정의하면, 초과된 인수는 무시된다.


728x90
블로그 이미지

Link2Me

,
728x90

구글 아이디로 로그인하는 걸 해보려고 구글링해서 몇가지를 테스트해봤는데 이것은 동작된다.


https://www.sanwebe.com/2012/11/login-with-google-api-php

에 가면 파일을 데모를 실행해보고 다운로드 할 수 있다.


테이블 스키마는 변경해야 한다.

사진 경로 길이가 너무 길어서 위 사이트에 나온 스키마 구조로는 에러가 발생한다.

CREATE TABLE IF NOT EXISTS `google_users` (
  `uid` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `google_id` varchar(60) NOT NULL,
  `google_name` varchar(60) NOT NULL,
  `google_email` varchar(60) NOT NULL,
  `google_link` varchar(150) NOT NULL,
  `google_picture_link` varchar(200) NOT NULL,
  PRIMARY KEY (`uid`)
) ENGINE=MyISAM  DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;


테이블 스키마는 회원테이블 정보에 가져올 정보를 표시하는 것이므로 굳이 이 테이블이 아니어도 된다.

이 테이블은 구글 회원정보를 가져온다는 걸 보여주는 것이라고 보면 된다.


첨부파일을 받으면 index.php 파일 내용을 수정해야 한다.

// https://console.developers.google.com/ 에서 client ID and secret 생성
$client_id = 'xxxxxxxxxxxxxxxxxx';  // 구글에서 생성한 id 입력
$client_secret = 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxx'; // 비밀키 입력
$redirect_uri = 'http://abc.com/oauth/google/'; // 본인 도메인으로 수정하세요


$redirect_uri = 'http://localhost/oauth/google/';

localhost 로 설정하면 에러가 발생한다.

Error: redirect_uri_mismatch

The redirect URI in the request, http://localhost/oauth/google/, does not match the ones authorized for the OAuth client.

그러므로 반드시 도메인명을 기록해야 한다. (IP주소 입력한 것은 테스트 안해봤음)

Google OAuth 2.0 Client API Keys 생성방법

https://console.developers.google.com/

https://www.sanwebe.com/2012/10/creating-google-oauth-api-key 에도 설명은 나오고

http://phppot.com/php/php-google-oauth-login/ 에도 나온다.

내가 등록을 참조한 곳은 http://phppot.com/php/php-google-oauth-login/ 이다.

하지만 이 사이트에서 제공하는 소스로는 연동이 잘 안되었다.


생성된 결과 화면이다.


테스트에 사용했던 파일입니다.

MySQLi 가 아닌 MySQL 연동 테스트 입니다. MySQLi 연동방식은 블로그에 올려진 걸 찾아서 변경하면 됩니다.

받아서 도움되신 분은 공감 꾸욱 눌러주세요~~


google_oauth.zip



728x90
블로그 이미지

Link2Me

,
728x90

Class 란 객체(object)를 만들기 위한 설계도이다.

객체지향에 대한 개념을 잘 이해하고자 한다면, 유투브 강좌 "자바의 정석"을 추천한다.


PHP Class 를 이용하면 함수 이상으로 코드를 재사용할 수 있다.


PHP의 모든 함수와 클래스는 전역이다.
함수가 내부에서 정의되었더라도 외부에서 호출할 수 있으며, 반대도 성립한다.
PHP는 함수 오버로딩(overloading)을 지원하지 않으며, 함수 정의를 해제하거나 이미 선언된 함수를 다시 선언할 수 없다.


이중 밑줄(__) 로 시작하는 PHP 함수를 매직 메소드(Magic Method)라 부른다.

Magic Method 에는 __construct(), __destruct(), __call(), __callStatic(), __get(), __set(), __isset(), __unset(), __sleep(), __wakeup(), __toString(), __invoke(), __set_state() 그리고 __clone() 이 있다.


  __construct() : 클래스의 생성자
     새로운 오브젝트가 생성(인스턴스화)될때마다 이 메소드를 호출

  __destruct() : 클래스의 소멸자
     특정 오브젝트에의 모든 참조(레퍼런스)가 삭제된 직후 도는 오브젝트가 명시적으로 파기된 직후 호출
  __set() – 엑세스 할 속성에 데이터를 쓸 때 사용 (저장)
  __get() – 엑세스 할 속성에서 데이터를 읽을 때 사용 (출력)
  __isset() – 엑세스할 속성에 isset() 또는 empty() 를 호출하여 트리거 됨 (확인)
  __unset() – 엑세스 할 속성에 unset() 를 사용할 경우 호출 (삭제)


클래스 정의

class 를 정의하기 위해서는 class 키워드를 사용하여 정의하여야 한다.

Login 으로 클래스를 정의할 수도 있지만 클래스 라는 걸 인지하기 쉽도록 LoginClass 로 정의했다.


<?php

// 클래스명은 첫자는 대문자로 시작하고, 파일명과 동일하게 한다.

class LoginClass { // LoginClass 클래스 선언
    public $email, $name;  //public으로 변수 선언

    function __construct(){       // 클래스의 생성자 선언
        $this->name = "Link2me"; // 객체가 생성되면서 name 속성에 Link2me 대입
    }

    function __destruct(){
        echo "Login Class 객체 소멸";
    }

    function UserIDJoinCheck($userID) { // 함수(메소드) 선언
        if(!isset($userID) || empty($userID)) {
            return '-1';
        } else {
            global $db; //global 키워드를 사용하여 변수에 선언할 경우 함수 밖의 변수를 참조할 수 있다.          
            $userID = preg_replace("/[\s\t]+/","", $userID); // 공백이나 탭 제거(사용자 실수 방지)

            $sql = "select id, count(id)
from member ";
            $sql.= "where id= '".$userID."' ";
            if($result = mysql_query($sql,$db)) { //성공
                $row = mysql_fetch_row($result);
                if($row[1] == '0') return 0; // userID 미가입
                return $row[1];
            } else {
                return '-1';
            }
        }
    }


    static function MyName(){
        echo "My name is Link2me.";
    }

}
?>


접근 제한자

public  
- 클래스 안, 밖에서 접근이 가능하다.
- 키워드를 아무것도 적지 않았을 경우에는 public 이 기본 값으로 적용된다.

protected
- 클래스 안에서만 접근이 가능하지만 하위 클래스로 상속이 가능하다.
  하지만 하위 클래스 외부에선 접근이 불가하다.

private
- 클래스 안에서만 접근이 가능하며, 상속이 안된다.


객체 생성

객체를 생성하려면 new 키워드를 사용한다.

$object = new LoginClass;  // new 키워드로 LoginClass 객체 생성(객체화, 인스턴스화)


객체 접근

클래스 메서드 내에서 비정적 객체(object)의 속성(property)에 접근하는 방법은 객체연산자( -> )를 사용한다.

$data = $object->UserIDJoinCheck($id);

속성이나 함수(메소드)에 접근할 때에는 $ 를 붙이지 않도록 하라. $를 붙이면 해당 변수를 가리키기 때문에 엉뚱한 결과가 발생할 수 있음을 명심하자.


Class 내에서 다른 메소드(함수)에 접근할 때에는 $this 를 사용한다.

$this는 해당 클래스를 가리킨다.

$this->val 는  해당 클래스의 $val 변수에 접근한다는 것을 의미한다.


정적(static) 메소드의 호출은 객체 연산자(->) 가 아닌 더블콜론(::)으로 호출된다.

더블콜론은 static 이나 constant와, 클래스의 재정의된 프로퍼티나 메서드에 접근 할 수 있도록 해주는 토큰이다.

LoginClass::MyName();



객체 복제

new로 객체를 생성하여 매개 변수로 넘길 때는 참조로 넘어간다.
따라서 새로운 메모리를 할당하는 것이아닌 객체의 주소값을 전달하는 것이다.

$object1 = new LoginClass;

$boject2 = $object1;


객체 복사를 해서 적용하거나 테스트를 안해봐서 세부적인 사항은 나중에 추가할 생각이다.


접근 제한자 : 멤버변수나 메소드 앞에 지정하여 클래스 외부에서의 참조나 실행을 제한 할 때 사용한다.

 - 접근 제한자가 정의되어 있지 않으면, 기본으로 public 으로 가정한다.

 - public : 외부에서 참조할 수 있음.

 - private : 변수를 선언한 클래스 안에서만 참조할 수 있다.

 - protected : 변수를 선언한 클래스와 상속 받은 클래스에서 참조할 수 있다.

 - final 메소드 : 서브 클래스가 부모 클래스의 메소드를 대체하는 걸 원치 않을 경우에는 final 키워드를 사용한다.

   final function copyright(){

   echo "Copyright Link2me";

   }


상속(inheritance)
: 클래스를 정의할 때 다른 클래스의 내부 데이터(멤버변수)나 메소드를 이어받어 새로운 클래스를 작성할 수 있는 기능(코드의 재활용성)


extends 키워드를 사용하여 다른 클래스의 메소드와 멤버변수(속성)을 상속할 수 있다.

class 자식 클래스명 extends 부모클래스명 {

   처리 코드

}


class ParentClass {
    var $name = "개발자";
}

class SubClass extends ParentClass{
    function info(){
        echo "name : ".$this->name; // 부모 클래스의 변수를 상속 받는다.
    }
}

$childObject = new SubClass();
$childObject -> info();


// 부모 클래스의 public 키워드로 선언된 변수와 메소드를 final로 변경하면

// 자식클래스에서 오버라이딩이 제한된다.


부모 클래스의 접근 제한자가 public, protected, private 에 따라 출력 결과가 달라짐을 확인해보라.

<?php
    class ParentClass{
        public $name = "홍길동";
        
        protected function showInfo(){
            echo "name : ".$this->name;
        }
        
        public function output(){
            echo "ParentClass ";
            $this->showInfo();
        }
    }
    
    
    class SubClass extends ParentClass{
        public function output(){
            echo "subClass ";
            $this->showInfo();
        }
    }
    
    $parentObject = new ParentClass();
    $parentObjec->output();
    
    echo '<br />';
    
    $childObjec = new SubClass();
    $childObjec->output();
?>

생성자

 - 자식 클래스가 생성자를 가지고 있지 않은 경우, 자동으로 부모 클래스의 생성자를 호출한다.

 - 자식 클래스가 생성자를 가지고 있는 경우 부모 클래스의 생성자가 암묵적으로 호출되지 않는다.
 - 부모클래스의 생성자를 실행하기위해서는 자식 클래스내에서 parent::__construct() 를 호출하여야 한다.

 - 생성자는 2개(ex, 인자 있는것과 없는 것) 생성하면 에러가 발생한다.

<?php
class people {
    public $height;
    public $weight;
    public function __construct($height, $weight) {
        $this -> height = $height;
        $this -> weight = $weight;
    }
}

class subPeople extends people {
    public function __construct($height, $weight) {
        parent::__construct($height, $weight);
    }
}

$people = new people(170, 50);
$subPeople = new subPeople(180, 60);

print_r($people);
print_r($subPeople);
?>


728x90

'Web 프로그램 > PHP 문법' 카테고리의 다른 글

[PHP] PHP 문자열 찾기  (0) 2017.11.14
[PHP] 배열에만 동작하는 foreach 문  (0) 2017.01.26
PHP isset 과 empty 설명  (1) 2016.04.27
PHP 대체 문법  (0) 2016.04.05
[PHP] switch 문  (0) 2015.06.01
블로그 이미지

Link2Me

,
728x90

생활코딩 사이트 동영상 강좌를 보다가 Aptana 를 알게되어 설치를 시도했다.

Aptana Studio 3 를 알고 있을 때와 모르고 있을 때의 차이점은

Aptana Studio 3 를 알면 파워풀한 기능, 편리한 기능으로 코딩하기가 정말 쉬워진다.


http://www.aptana.com/products/studio3/download.html

에서 윈도우버전 3.6.1을 받아서 설치를 했다. (경고성 메시지 떠서 강제로 실행했음, 2019.8.27)

https://github.com/aptana/studio3/releases 에 최신 버전이 올라와 있다.


위 사이트에 가면 최신버전이 나온다.


알아야 할 사항

JDK 파일은 반드시 32비트 버전 8버전대를 받아야 한다.

64비트 용으로 받았다가 지우고 다시 32비트 버전으로 설정을 했다.



에러가 발생하면서 설치가 중단된다.

nodejs 파일을 설치해주면 된다고 해서 받아서 설치해도 안된다.

검색했더니 아래 파일을 설치해주라고 되어 있어서 받아서 설치를 했더니 설치도중에 나오는 에러메시지는 없어지고 잘 설치가 되었다.


node-v0.10.13-x86.msi


http://go.appcelerator.com/installer_git_windows.exe 파일을 받아서 설치하라고 구글 검색하면 나온다.


Aptana 프로그램을 실행했더니 또 에러가 나면서 실행이 안된다.

Failed to load the JNI shared library "C:\Program Files......\jvm.dll"

Aptana 3.6.1 이 기본적으로 x86 으로 설치가 되는거 같다.

eclipse 는 64비트로 잘 동작되는데도 불구하고 설치를 하면 에러가 발생한다.


http://www.oracle.com/technetwork/java/javase/downloads/jre8-downloads-2133155.html



받아서 설치하니까 자동으로 인식된다.

위 사이트에서 받지 않고 첨부한 이 파일을 받아서 설치해도 된다.


jre-8u111-windows-i586-iftw.exe



Aptana 도 eclipse 처럼 workspace 경로를 잡아주어야 한다.

그리고 AutoSet9 과 연동하여 Apache 환경설정 경로를 잡아주어야 제대로 사용하는 거다.

- 프로젝트 단위로  생성하여 관리한다. (eclipse 기능하고 동일한 개념)

- 파일에서 내용 검색이 쉽다.

- 코드를 입력하면 자동으로 검색되는 키워드를 보여줘 무척 편하다.

- 코드를 만들어 테스트할 때, Firefox 등 웹브라우저로 바로 연결되어 결과를 확인할 수 있어 편하다.



EditPlus

- Total Commander 에서 파일을 선택하면 자동으로 EditPlus 창이 뜨므로 코드 수정이 편하다.
-
리눅스 APM에 코드를 올려야 한다면 WinSCP 를 이용하면 편하다.
-
EditPlus 매크로 기능을 사용하면 단순작업이 편하다.

- 인터넷에 있는 코드를 복사해서 붙여넣기 한다음 간단하게 수정하기가 편하다.

728x90
블로그 이미지

Link2Me

,
728x90

Form 데이터를 jQuery submit 하면서 기초가 부족해서 삽질을 많이 한다.

http://hayageek.com/jquery-submit-form/ 사이트 예제, http://www.sqler.com/387921 를 보면 도움된다.


Form 전송시 필드(input box) 를 동적으로 추가해야 할 경우가 있다.

http://www.codexworld.com/add-remove-input-fields-dynamically-using-jquery/ 에서 소스를 받아서 코드를 보강하면서 테스트를 했다.


수정/추가한 부분 및 내역

- javascript:void(0); 대신에 #으로 수정하고 jQuery 코드 부분 수정(e.preventDefault() 추가)

- 용어 내 스타일로 변경

- form submit 부분 추가하고 ajax 형태로 처리하는 코드 추가

- 입력값이 없을 때 체크하는 코드 추가

※ DB에 저장하고 나서 다시 수정/삭제/추가할 경우까지 고려되어 있지는 않다.


=== index.php ===

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<title>Add more fields using jQuery</title>
<style type="text/css">
input[type="text"]{height:18px;width:220px;vertical-align:top;}
.field_wrapper div{ margin-bottom:10px;}
.add_button{ margin-top:10px; margin-left:10px;vertical-align: text-bottom;}
.removeBtn{ margin-top:10px; margin-left:10px;vertical-align: text-bottom;}
</style>
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<script type="text/javascript">
$(document).ready(function(){
    var maxCount = 5;
    var addButton = $('.add_button');
    var wrapper = $('.field_wrapper');
    var AddBox = '<div><input type="text" name="kordiv[]" value=""/><a href="#" class="removeBtn" title="제거"><img src="remove-icon.png"/></a></div>';
    var x = 1;
    $(addButton).click(function(e){
        e.preventDefault();
        if(x < maxCount){
            x++;
            $(wrapper).append(AddBox);
        }else{
            alert('최대 '+maxCount+'개까지 허용합니다');
            return false;
        }
    });
    $(wrapper).on('click', '.removeBtn', function(e){
        e.preventDefault();
        $(this).parent('div').remove();
        x--;
    });

    $("#submit1").click(function(){
        // 폼 요소 내부의 입력값만을 찾기 위해서 폼 요소에서 find()를 호출
        var xyz = $("#myform").find('[name=xyz]').val();
        if(xyz =='') {
            alert('xyz 값이 입력되지 않았습니다.');
            return false;
        }

        // 배열로 된 값이 제대로 입력되었는지 체크
        var cnt = 0;
        $('input[name^="kordiv"]').each(function() {
            //alert($(this).val());
            if($(this).val() === '') {
                cnt++;
            }
        });
        if(cnt > 0){
            alert('kordiv 에 입력되지 않는 필드가 '+cnt+'개 있습니다');
            return false;
        }

        var formData = $("#myform").serialize();
        var URL = $("#myform").attr("action");
        $.post(URL, formData, function(data, textStatus, jqXHR){
            alert(data);
            var data = jQuery.parseJSON(data);
            $.each(data, function(index, value) {
                alert(value);
            });

        }).fail(function(jqXHR, textStatus, errorThrown) {
            console(errorThrown);
        });

    });

});
</script>
</head>
<body>
<form name="frm" id="myform" action="frm_result.php" method="POST">
<input type="hidden" name="xyz" value="123" />
<div class="field_wrapper">
    <div>
        <input type="text" name="kordiv[]" value=""/>
        <a href="#" class="add_button" title="추가"><img src="add-icon.png"/></a>
    </div>
</div>
<input type="hidden" name="submit" value="true" />
</form>
<br/>
<input type="button" id="submit1" value="전송" />
</body>
</html>


=== frm_result.php ===

<?php
if(isset($_POST['submit'])){
    $kordiv_array = $_POST['kordiv']; // 배열

    /*
    print '<pre>';
    print_r($kordiv_array);
    print_r($_POST['xyz']);
    print '</pre>';
    // */

    $R = array();
    foreach($kordiv_array as $value){ // 배열로 받은 걸 분리 처리
        //echo $value;
        array_push($R,$value);
    }
    $Content = 'Hello world';
    echo json_encode(array('rs'=>$R,'Content' => $Content));
}
?>


728x90
블로그 이미지

Link2Me

,
728x90

<th><a href="#" onfocus=blur()  onClick="window.open('Stats.php','','width=1600, height=680,top=200,left=50,location=no, directories=no,resizable=no,status=no,toolbar=no,menubar=no,titlebar=no,scrollbars=no, resizable=no,copyhistory=no');return false;">일별통계</a></th>



location=no 를 설정해도 보안 문제 때문에 무조건 보이게 한단다.

그래서 윈도우 창을 새로 띄운 경우에는 URL 정보가 노출된다.


jQuery popup 하는 걸 테스트해보면

<div></div> 가 있는 상태에서 화면에 보이지 않게 하다가 새로운 창으로 정보를 보이게 해주는 역할이다.

그러다보니 기존 파일의 jQuery 정보와의 충돌문제가 있을 수도 있어서 원하지 않은 결과가 나오기도 한다.


팝업창 띄우는 걸 검색해서 하나 찾았다.

http://inspirationalpixels.com/tutorials/custom-popup-modal


이것만으로 쉽게 해결될 수도 있지만 해결이 잘 안될때가 있다.

그걸 극복하기 위해서 편법을 구상했다. 바로 <iframe> 이다.

iframe 에 대한 사항은 http://www.homejjang.com/05/iframe.php 을 참조하면 도움된다.


아래 코드와 같이 하면 해결된다.


<!DOCTYPE html>
<head>
<meta charset="UTF-8" />
<meta name="robots" content="noindex,nofollow"/>
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no"/>
<meta http-equiv="X-UA Compatible" />
<link href="http://www.jqueryscript.net/css/jquerysctipttop.css" rel="stylesheet" type="text/css">
<link href="popup_layer.css" rel="stylesheet" type="text/css">
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<script type="text/javascript">
$(function() {
    // 팝업 OPEN
    $('#popup-open-1').on('click', function(e)  {
        var targeted_popup_class = jQuery(this).attr('data-popup-open');
        $('[data-popup="' + targeted_popup_class + '"]').fadeIn(350);
        e.preventDefault();
    });
 
    // 팝업 CLOSE
    $('#popup-colse-1').on('click', function(e)  {
        var targeted_popup_class = jQuery(this).attr('data-popup-close');
        $('[data-popup="' + targeted_popup_class + '"]').fadeOut(350);
        e.preventDefault();
    });
});
</script>
</head>
<body>
<a href="#" id="popup-open-1" class="btn" data-popup-open="popup-1">Open Popup #1</a>
<div class="popup" data-popup="popup-1">
    <div class="popup-inner">
        <iframe src="http://www.daum.net" frameborder="0" width="1200" height="500" marginwidth="0" marginheight="0"></iframe>
        <a href="#" id="popup-colse-1" class="popup-close" data-popup-close="popup-1">x</a>
    </div>
</div>

</body>
</html>




728x90
블로그 이미지

Link2Me

,
728x90

PHP 배열로 넘겨주는 함수에 따라서 처리를 다르게 해야 동작된다는 걸 확인하고 코드를 다시 적어둔다.


http://link2me.tistory.com/1134 게시글에서 가져오는 함수에서 넘겨주는 값이

문자열로 넘겨줄 경우와 배열로 넘겨줄 경우 각각 상황에 맞게 구현해야 한다.


<?php
require_once 'connect.php'; // db접속 성공
require_once 'statsClass.php'; // 통계 클래스
$b = new statsClass();
$R = $b->extract_YM();
date_default_timezone_set('Asia/Seoul');
if(isset($_GET['ym'])){
    $ym = $_GET['ym'];
} else {
    $ym = date("Ym");
}
$ticks = json_encode($b->maxdate_YM($ym));
$line1 = json_encode($b->dailyCnt_value($ym));
$line2 = json_encode($b->userCnt_value($ym));
?>
<!DOCTYPE html>
<head>
<meta charset="UTF-8" />
<meta name="robots" content="noindex,nofollow"/>
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no"/>
<meta http-equiv="X-UA Compatible" />
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<!-- http://www.jqplot.com 제공 자바스크립트 include -->
<script type="text/javascript" src="../js/jqplot/jquery.jqplot.js"></script>
<script type="text/javascript" src="../js/jqplot/plugins/jqplot.barRenderer.js"></script>
<script type="text/javascript" src="../js/jqplot/plugins/jqplot.pointLabels.js"></script>
<script type="text/javascript" src="../js/plugins/jqplot.categoryAxisRenderer.js"></script>
<link rel="stylesheet" type="text/css" href="../js/jqplot/jquery.jqplot.css" />
<script type="text/javascript">
$(document).ready(function(){
    var plot1;

    var obj1 = <?php echo $line1; ?>;
    var line1 = $.map(obj1, function(el) { return el });
    var obj2 = <?php echo $line2; ?>;
    var line2 = $.map(obj2, function(el) { return el });
    var ticks = JSON.parse('<?php echo $ticks; ?>');

    renderChart(line1,line2,ticks);

    $('#selectmonth').on('change',function(){
        if(this.value !== ""){
            var optVal=$(this).find(":selected").val();
            //var pageURL = "<?php echo $_SERVER['PHP_SELF'];?>?ym=" + optVal;
            //$(location).attr('href', pageURL);

            var year = optVal.substr(0,4); // 년
            var month = optVal.substr(4,2); // 월
            $.post('stats_db.php',{optVal:optVal},function(msg){
                //alert(msg); // 배열 형태로 넘어오는지 확인 목적
                var jsonObj = $.parseJSON(msg); // JSON 문자열을 JavaScript object 로 반환
                //alert(typeof jsonObj.ticks);
                //alert(jsonObj.data1);
// string 인지 object 인지 확인 목적
                var data1 = $.map(jsonObj.data1, function(el) { return el });
                var data2 = $.map(jsonObj.data2, function(el) { return el });
                //alert(jsonObj.ticks);
                var xAxis = jsonObj.ticks;
                updatePlot(data1,data2,xAxis);
            });
        }
    });

});


function renderChart(line1,line2,xAxis){
    plot1 = $.jqplot('chartDiv', [line1,line2], CreateBarChartOptions(xAxis));
}

function updatePlot(line1,line2,xAxis){
    plot1.destroy();
    plot1 = $.jqplot('chartDiv', [line1,line2], CreateBarChartOptions(xAxis));
}

function CreateBarChartOptions(xAxis) {
    var optionsObj = {
        title: '일자별 접속 통계',
        seriesDefaults:{
            renderer:$.jqplot.BarRenderer, // 막대 그래프
            rendererOptions: { barWidth: 10 }, // bar의 너비 수동으로 설정
            pointLabels: { show: true } // 레이블 값
        },
        axes: {
            xaxis: {
                renderer: $.jqplot.CategoryAxisRenderer,
                ticks: xAxis,
                tickOptions: {
                    formatString: '%d' // 정수형으로 표시
                }
                //label:'일자'
            },
            yaxis:{
                //renderer:$.jqplot.DateAxisRenderer,
                min:0,
                //max:80
            }
        },
        highlighter: { show: false }
    };
    return optionsObj;
}
</script>

</head>
<body>
<div id="chartDiv" style="height:400px;width:1500px; "></div>
<div style="height:300px;width:1490px;margin-top:10px;text-align:right;">
<select name="month" id="selectmonth">
<option value="">년월 선택</option>
<?php
    while ($row = mysql_fetch_array($R)){
        echo "<option value='".$row[0]."' ";
        if($row[0] === $ym) echo "selected='selected'";
        echo ">".$row[0]."</option>\n";
    }
?>
</select>
</div>
</body>
</html>


=== stats_db.php ===

<?php
if(isset($_POST['optVal'])){
    require_once 'connect.php'; // db접속 성공
    require_once 'statsClass.php'; // 통계 클래스
    $b = new statsClass;
    $ym = $_POST['optVal'];
    $data1 = $b->dailyCnt_value($ym); // 배열로 받은 결과
    $data2 = $b->userCnt_value($ym); // 배열로 받은 결과
    $ticks = $b->maxdate_YM($ym);  // 배열로 받은 결과
    $R = array('data1'=>$data1,'data2'=>$data2, 'ticks'=>$ticks);
    echo json_encode($R);
}
?>

728x90
블로그 이미지

Link2Me

,
728x90

Array 변수 처리에 대한 이해가 부족하여 기능 구현보다 변환하는데 시간이 휠씬 더 걸리고 있다.

다 기초가 부족한 탓이다. 테스트한 결과를 무조건 기록해둔다.


PHP 함수에서 return json_encode($R); 배열로 반환 결과가

{"1":"36","2":"23","3":"14","4":"7","5":"12","6":"6","7":"5","8":"9","9":"17","10":"7"}

일 경우


자바스크립트에서 배열 변수로 받기 위해서는

var obj1 = JSON.parse('<?php echo $line1; ?>');

var line1 = new Array();

for(var i in obj1){

    line1.push(obj1[i]);

}

로 하면 값만 추출된 배열변수가 만들어진다.

또다른 방법으로 테스트를 했다.

var obj1 = <?php echo $line1; ?>;

var line1 = $.map(obj1, function(el) { return el });

로 해도 동일한 결과가 나온다.



배열 변환 결과가

[0,1,2,3,4,5,6,7,8,9,10]

로 화면 출력될 경우에는

var ticks = JSON.parse('<?php echo json_encode($ticks); ?>');

또는

var ticks = <?php echo '["' . implode('","', $ticks) . '"]' ?>;

둘 중에 하나를 사용하면 배열 변수가 된다.


PHP 배열로 반환된 결과를

<?php

$Line2 = array();
foreach($line2 as $key => $value){
    array_push($Line2,$value);
}

?>

로 값만 다시 배열로 만든다음에

var line2 = <?php echo '["' . implode('","', $Line2) . '"]' ?>;

로 자바스크립트 배열 변수를 만들 수 있다.


테스트한 파일을 그대로 적어둔다.

<?php
require_once 'connect.php'; // db접속 성공
require_once 'statsClass.php'; // 통계 클래스
$b = new statsClass();
$R = $b->extract_YM();
date_default_timezone_set('Asia/Seoul');
if(isset($_GET['ym'])){
    $ym = $_GET['ym'];
} else {
    $ym = date("Ym");
}
$ticks = $b->maxdate_YM($ym);
$line1 = $b->dailyCnt_value($ym);
$line2 = $b->userCnt_value($ym);
//echo $line1.'<br />';
//echo json_encode($line2).'<br />';
//echo json_encode($ticks).'<br />';
$Line2 = array();
foreach($line2 as $key => $value){
    array_push($Line2,$value);
}
?>
<!DOCTYPE html>
<head>
<meta charset="UTF-8" />
<meta name="robots" content="noindex,nofollow"/>
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no"/>
<meta http-equiv="X-UA Compatible" />
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<!-- http://www.jqplot.com 제공 자바스크립트 include -->
<script type="text/javascript" src="../js/jqplot/jquery.jqplot.js"></script>
<script type="text/javascript" src="../js/jqplot/plugins/jqplot.barRenderer.js"></script>
<script type="text/javascript" src="../js/jqplot/plugins/jqplot.pointLabels.js"></script>
<script type="text/javascript" src="../js/plugins/jqplot.categoryAxisRenderer.js"></script>
<link rel="stylesheet" type="text/css" href="../js/jqplot/jquery.jqplot.css" />
<script type="text/javascript">
$(document).ready(function(){
    var plot1;
    /*
    var obj1 = JSON.parse('<?php echo $line1; ?>');
    var line1 = new Array();
    for(var i in obj1){
        line1.push(obj1[i]);
    }
    */
    var obj1 = <?php echo $line1; ?>;
    var line1 = $.map(obj1, function(el) { return el });

    //var obj2 = JSON.parse('<?php echo json_encode($line2); ?>');
    /*
    var line2 = new Array();
    for(var i in obj2){
        line2.push(obj2[i]);
    }
    */
    var line2 = <?php echo '["' . implode('","', $Line2) . '"]' ?>;

    //var ticks = JSON.parse('<?php echo json_encode($ticks); ?>');
    var ticks = <?php echo '["' . implode('","', $ticks) . '"]' ?>;

    renderChart(line1,line2,ticks);

    $('#selectmonth').on('change',function(){
        if(this.value !== ""){
            var optVal=$(this).find(":selected").val();
            var pageURL = "<?php echo $_SERVER['PHP_SELF'];?>?ym=" + optVal;
            $(location).attr('href', pageURL);
        }
    });

});


function renderChart(line1,line2,xAxis){
    plot1 = $.jqplot('chartDiv', [line1,line2], CreateBarChartOptions(xAxis));
}

function updatePlot(line1,line2,xAxis){
    plot1.destroy();
    plot1 = $.jqplot('chartDiv', [line1,line2], CreateBarChartOptions(xAxis));
}

function CreateBarChartOptions(xAxis) {
    var optionsObj = {
        title: '일자별 접속 통계',
        seriesDefaults:{
            renderer:$.jqplot.BarRenderer, // 막대 그래프
            rendererOptions: { barWidth: 10 }, // bar의 너비 수동으로 설정
            pointLabels: { show: true } // 레이블 값
        },
        axes: {
            xaxis: {
                renderer: $.jqplot.CategoryAxisRenderer,
                ticks: xAxis,
                tickOptions: {
                    formatString: '%d' // 정수형으로 표시
                }
                //label:'일자'
            },
            yaxis:{
                //renderer:$.jqplot.DateAxisRenderer,
                min:0,
                //max:80
            }
        },
        highlighter: { show: false }
    };
    return optionsObj;
}
</script>

</head>
<body>
<div id="chartDiv" style="height:400px;width:1500px; "></div>
<div style="height:300px;width:1490px;margin-top:10px;text-align:right;">
<select name="month" id="selectmonth">
<option value="">년월 선택</option>
<?php
    while ($row = mysql_fetch_array($R)){
        echo "<option value='".$row[0]."' ";
        if($row[0] === $ym) echo "selected='selected'";
        echo ">".$row[0]."</option>\n";
    }
?>
</select>
</div>
</body>
</html>


728x90
블로그 이미지

Link2Me

,
728x90

어제 고객사 사이트에 autocomplete 기능을 적용 테스트를 했다.

내가 사용하는 환경인 윈도우10 Explorer 11 에서는 autocomplete 동작이 잘 된다.

하지만 고객사 키맨으로부터 전혀 검색을 할 수가 없어 기존 사용하던 것도 안된다는 민원이 발생해서 환경을 확인해보니 윈도우 7 에 Explorer 버전이라고 나온다. Explorer 세부 버전정보까지는 확인 불가

그래서 결국에

$('#searchValue').autocomplete({
    source: "path/autosearch.php",
    minLength: 1
});

는 주석처리 할 수 밖에 없었다.


// === autosearch.php ====
<?php
require_once "connect.php";

if (isset($_GET['term'])){
    $return_arr = array();

    $keyword = $_GET['term'];
    $sql = "SELECT distinct(name) FROM member WHERE name LIKE '%".$keyword."%'";
    $result = mysql_query($sql);
    while($row = mysql_fetch_assoc($result)){
        array_push($return_arr,$row['name']);
    }

    echo json_encode($return_arr);
}
?>


중복없이 이름 검색이 되도록 하기 위해서 distinct 처리를 했다.

원하는 결과가 잘 나옴에도 불구하고 적용은 할 수가 없었다.


http://jqueryui.com/autocomplete/ 에서 샘플로 나온 것이 아래 스크립트를 포함하고 있어서 이걸로 테스트를 했는데 윈도우 10 Firefox 브라우저에서는 잘 되는걸 확인했다.

<link rel="stylesheet" href="http://code.jquery.com/ui/1.12.1/themes/base/jquery-ui.css" type="text/css" />
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<script type="text/javascript" src="http://code.jquery.com/ui/1.12.1/jquery-ui.js"></script>


일단 접속하는 브라우저의 정확한 정보부터 수집하는 코드를 구현해야 할 거 같다.

지금은 Chrome,Firefox, Explorer 여부만 수집되는데 버전 정보까지 수집되도록 코드를 세분화해야 할 거 같다.

그래야 새로운 코드 적용시 발생하는 민원을 정확하게 감지할 수 있을 거 같다.


http://www.computerhope.com/issues/ch000604.htm 사이트 자료에 Explorer 세팅 정보를 수정하면 동작되는 것 처럼 되어 있기는 하다.


멀티부팅되는 환경을 만들어서 다양한 접속을 해야만 하나보다.

728x90
블로그 이미지

Link2Me

,