mirror of
https://github.com/RicterZ/nhentai.git
synced 2025-07-01 16:09:28 +02:00
Compare commits
15 Commits
Author | SHA1 | Date | |
---|---|---|---|
16e8ce6f45 | |||
0632826827 | |||
8d2cd1974b | |||
8c176cd2ad | |||
f2c88e8ade | |||
2300744c5c | |||
7f30c84eff | |||
dda849b770 | |||
14b3c82248 | |||
4577e9df9a | |||
de157ccb7f | |||
126bbe8d49 | |||
8546b9e759 | |||
6ff9751c30 | |||
f316c3243b |
@ -5,9 +5,7 @@ language: python
|
|||||||
python:
|
python:
|
||||||
- 2.7
|
- 2.7
|
||||||
- 2.6
|
- 2.6
|
||||||
- 3.3
|
- 3.6
|
||||||
- 3.4
|
|
||||||
- 3.5.2
|
|
||||||
|
|
||||||
install:
|
install:
|
||||||
- python setup.py install
|
- python setup.py install
|
||||||
@ -15,3 +13,5 @@ install:
|
|||||||
script:
|
script:
|
||||||
- NHENTAI=https://nhentai.net nhentai --search umaru
|
- NHENTAI=https://nhentai.net nhentai --search umaru
|
||||||
- NHENTAI=https://nhentai.net nhentai --id=152503,146134 -t 10 --output=/tmp/
|
- NHENTAI=https://nhentai.net nhentai --id=152503,146134 -t 10 --output=/tmp/
|
||||||
|
- NHENTAI=https://nhentai.net nhentai -l nhentai_test:nhentai --output=/tmp/
|
||||||
|
- NHENTAI=https://nhentai.net nhentai --tag lolicon
|
10
MANIFEST.in
10
MANIFEST.in
@ -1,5 +1,5 @@
|
|||||||
include README.md
|
include README.md
|
||||||
include requirements.txt
|
include requirements.txt
|
||||||
include nhentai/viewer/index.html
|
include nhentai/viewer/index.html
|
||||||
include nhentai/viewer/styles.css
|
include nhentai/viewer/styles.css
|
||||||
include nhentai/viewer/scripts.js
|
include nhentai/viewer/scripts.js
|
||||||
|
148
README.md
148
README.md
@ -1,70 +1,78 @@
|
|||||||
nhentai
|
nhentai
|
||||||
=======
|
=======
|
||||||
_ _ _ _
|
_ _ _ _
|
||||||
_ __ | | | | ___ _ __ | |_ __ _(_)
|
_ __ | | | | ___ _ __ | |_ __ _(_)
|
||||||
| '_ \| |_| |/ _ \ '_ \| __/ _` | |
|
| '_ \| |_| |/ _ \ '_ \| __/ _` | |
|
||||||
| | | | _ | __/ | | | || (_| | |
|
| | | | _ | __/ | | | || (_| | |
|
||||||
|_| |_|_| |_|\___|_| |_|\__\__,_|_|
|
|_| |_|_| |_|\___|_| |_|\__\__,_|_|
|
||||||
|
|
||||||
あなたも変態。 いいね?
|
あなたも変態。 いいね?
|
||||||
[](https://travis-ci.org/RicterZ/nhentai)
|
[](https://travis-ci.org/RicterZ/nhentai) 
|
||||||
|
|
||||||
🎉🎉 nhentai 现在支持 Windows 啦!
|
|
||||||
|
nHentai is a CLI tool for downloading doujinshi from [nhentai.net](http://nhentai.net).
|
||||||
由于 [http://nhentai.net](http://nhentai.net) 下载下来的种子速度很慢,而且官方也提供在线观看本子的功能,所以可以利用本脚本下载本子。
|
|
||||||
|
### Installation
|
||||||
### Installation
|
|
||||||
|
git clone https://github.com/RicterZ/nhentai
|
||||||
git clone https://github.com/RicterZ/nhentai
|
cd nhentai
|
||||||
cd nhentai
|
python setup.py install
|
||||||
python setup.py install
|
|
||||||
|
### Gentoo
|
||||||
### Gentoo
|
|
||||||
|
layman -fa glicOne
|
||||||
layman -fa glicOne
|
sudo emerge net-misc/nhentai
|
||||||
sudo emerge net-misc/nhentai
|
|
||||||
|
### Usage
|
||||||
### Usage
|
Download specified doujinshi:
|
||||||
下载指定 id 列表的本子:
|
```bash
|
||||||
```bash
|
nhentai --id=123855,123866
|
||||||
nhentai --id=123855,123866
|
```
|
||||||
```
|
|
||||||
|
Search a keyword and download the first page:
|
||||||
下载某关键词第一页的本子:
|
```bash
|
||||||
```bash
|
nhentai --search="tomori" --page=1 --download
|
||||||
nhentai --search="tomori" --page=1 --download
|
```
|
||||||
```
|
|
||||||
|
Download your favourite doujinshi (login required):
|
||||||
下载用户 favorites 内容:
|
```bash
|
||||||
```bash
|
nhentai --login "username:password" --download
|
||||||
nhentai --login "username:password" --download
|
```
|
||||||
```
|
|
||||||
|
Download by tag name:
|
||||||
### Options
|
```bash
|
||||||
|
nhentai --tag lolicon --download
|
||||||
`-t, --thread`:指定下载的线程数,最多为 10 线程。
|
```
|
||||||
`--path`:指定下载文件的输出路径,默认为当前目录。
|
|
||||||
`--timeout`:指定下载图片的超时时间,默认为 30 秒。
|
### Options
|
||||||
`--proxy`:指定下载的代理,例如: http://127.0.0.1:8080/
|
|
||||||
`--login`:nhentai 账号的“用户名:密码”组合
|
+ `-t, --thread`: Download threads, max: 10
|
||||||
`--nohtml`:nhentai Don't generate HTML
|
+ `--output`:Output dir of saving doujinshi
|
||||||
`--cbz`:nhentai Generate Comic Book CBZ file
|
+ `--tag`:Download by tag name
|
||||||
|
+ `--timeout`: Timeout of downloading each image
|
||||||
### nHentai Mirror
|
+ `--proxy`: Use proxy, example: http://127.0.0.1:8080/
|
||||||
如果想用自建镜像下载 nhentai 的本子,需要搭建 nhentai.net 和 i.nhentai.net 的反向代理。
|
+ `--login`: username:password pair of your nhentai account
|
||||||
例如用 h.loli.club 来做反向代理的话,需要 h.loli.club 反代 nhentai.net,i.h.loli.club 反带 i.nhentai.net。
|
+ `--nohtml`: Do not generate HTML
|
||||||
然后利用环境变量来下载:
|
+ `--cbz`: Generate Comic Book CBZ File
|
||||||
|
|
||||||
```bash
|
### nHentai Mirror
|
||||||
NHENTAI=http://h.loli.club nhentai --id 123456
|
If you want to use a mirror, you should set up a reverse proxy of `nhentai.net` and `i.nhentai.net`.
|
||||||
```
|
For example:
|
||||||
|
|
||||||

|
i.h.loli.club -> i.nhentai.net
|
||||||

|
h.loli.club -> nhentai.net
|
||||||

|
|
||||||
|
Set `NHENTAI` env var to your nhentai mirror.
|
||||||
### License
|
```bash
|
||||||
MIT
|
NHENTAI=http://h.loli.club nhentai --id 123456
|
||||||
|
```
|
||||||
### あなたも変態
|
|
||||||

|

|
||||||
|

|
||||||
|

|
||||||
|
|
||||||
|
### License
|
||||||
|
MIT
|
||||||
|
|
||||||
|
### あなたも変態
|
||||||
|

|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
__version__ = '0.2.12'
|
__version__ = '0.2.15'
|
||||||
__author__ = 'RicterZ'
|
__author__ = 'RicterZ'
|
||||||
__email__ = 'ricterzheng@gmail.com'
|
__email__ = 'ricterzheng@gmail.com'
|
||||||
|
@ -42,7 +42,7 @@ def cmd_parser():
|
|||||||
parser.add_option('--search', type='string', dest='keyword', action='store', help='search doujinshi by keyword')
|
parser.add_option('--search', type='string', dest='keyword', action='store', help='search doujinshi by keyword')
|
||||||
parser.add_option('--page', type='int', dest='page', action='store', default=1,
|
parser.add_option('--page', type='int', dest='page', action='store', default=1,
|
||||||
help='page number of search result')
|
help='page number of search result')
|
||||||
parser.add_option('--tags', type='string', dest='tags', action='store', help='download doujinshi by tags')
|
parser.add_option('--tag', type='string', dest='tag', action='store', help='download doujinshi by tag')
|
||||||
parser.add_option('--output', type='string', dest='output_dir', action='store', default='',
|
parser.add_option('--output', type='string', dest='output_dir', action='store', default='',
|
||||||
help='output dir')
|
help='output dir')
|
||||||
parser.add_option('--threads', '-t', type='int', dest='threads', action='store', default=5,
|
parser.add_option('--threads', '-t', type='int', dest='threads', action='store', default=5,
|
||||||
@ -86,20 +86,17 @@ def cmd_parser():
|
|||||||
if not args.is_download:
|
if not args.is_download:
|
||||||
logger.warning('YOU DO NOT SPECIFY `--download` OPTION !!!')
|
logger.warning('YOU DO NOT SPECIFY `--download` OPTION !!!')
|
||||||
|
|
||||||
if args.tags:
|
|
||||||
logger.warning('`--tags` is under construction')
|
|
||||||
exit(1)
|
|
||||||
|
|
||||||
if args.id:
|
if args.id:
|
||||||
_ = map(lambda id: id.strip(), args.id.split(','))
|
_ = map(lambda id: id.strip(), args.id.split(','))
|
||||||
args.id = set(map(int, filter(lambda id_: id_.isdigit(), _)))
|
args.id = set(map(int, filter(lambda id_: id_.isdigit(), _)))
|
||||||
|
|
||||||
if (args.is_download or args.is_show) and not args.id and not args.keyword and not args.login:
|
if (args.is_download or args.is_show) and not args.id and not args.keyword and \
|
||||||
|
not args.login and not args.tag:
|
||||||
logger.critical('Doujinshi id(s) are required for downloading')
|
logger.critical('Doujinshi id(s) are required for downloading')
|
||||||
parser.print_help()
|
parser.print_help()
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
if not args.keyword and not args.id and not args.login:
|
if not args.keyword and not args.id and not args.login and not args.tag:
|
||||||
parser.print_help()
|
parser.print_help()
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@ import signal
|
|||||||
import platform
|
import platform
|
||||||
|
|
||||||
from nhentai.cmdline import cmd_parser, banner
|
from nhentai.cmdline import cmd_parser, banner
|
||||||
from nhentai.parser import doujinshi_parser, search_parser, print_doujinshi, login_parser
|
from nhentai.parser import doujinshi_parser, search_parser, print_doujinshi, login_parser, tag_guessing, tag_parser
|
||||||
from nhentai.doujinshi import Doujinshi
|
from nhentai.doujinshi import Doujinshi
|
||||||
from nhentai.downloader import Downloader
|
from nhentai.downloader import Downloader
|
||||||
from nhentai.logger import logger
|
from nhentai.logger import logger
|
||||||
@ -27,13 +27,19 @@ def main():
|
|||||||
for doujinshi_info in login_parser(username=username, password=password):
|
for doujinshi_info in login_parser(username=username, password=password):
|
||||||
doujinshi_list.append(Doujinshi(**doujinshi_info))
|
doujinshi_list.append(Doujinshi(**doujinshi_info))
|
||||||
|
|
||||||
|
if options.tag:
|
||||||
|
tag_id = tag_guessing(options.tag)
|
||||||
|
if tag_id:
|
||||||
|
doujinshis = tag_parser(tag_id)
|
||||||
|
print_doujinshi(doujinshis)
|
||||||
|
if options.is_download:
|
||||||
|
doujinshi_ids = map(lambda d: d['id'], doujinshis)
|
||||||
|
|
||||||
if options.keyword:
|
if options.keyword:
|
||||||
doujinshis = search_parser(options.keyword, options.page)
|
doujinshis = search_parser(options.keyword, options.page)
|
||||||
print_doujinshi(doujinshis)
|
print_doujinshi(doujinshis)
|
||||||
if options.is_download:
|
if options.is_download:
|
||||||
doujinshi_ids = map(lambda d: d['id'], doujinshis)
|
doujinshi_ids = map(lambda d: d['id'], doujinshis)
|
||||||
else:
|
|
||||||
doujinshi_ids = options.id
|
|
||||||
|
|
||||||
if doujinshi_ids:
|
if doujinshi_ids:
|
||||||
for id_ in doujinshi_ids:
|
for id_ in doujinshi_ids:
|
||||||
|
@ -7,6 +7,8 @@ BASE_URL = os.getenv('NHENTAI', 'https://nhentai.net')
|
|||||||
|
|
||||||
DETAIL_URL = '%s/api/gallery' % BASE_URL
|
DETAIL_URL = '%s/api/gallery' % BASE_URL
|
||||||
SEARCH_URL = '%s/api/galleries/search' % BASE_URL
|
SEARCH_URL = '%s/api/galleries/search' % BASE_URL
|
||||||
|
TAG_URL = '%s/tag' % BASE_URL
|
||||||
|
TAG_API_URL = '%s/api/galleries/tagged' % BASE_URL
|
||||||
LOGIN_URL = '%s/login/' % BASE_URL
|
LOGIN_URL = '%s/login/' % BASE_URL
|
||||||
FAV_URL = '%s/favorites/' % BASE_URL
|
FAV_URL = '%s/favorites/' % BASE_URL
|
||||||
|
|
||||||
|
@ -27,7 +27,7 @@ def login_parser(username, password):
|
|||||||
|
|
||||||
s.get(constant.LOGIN_URL)
|
s.get(constant.LOGIN_URL)
|
||||||
content = s.get(constant.LOGIN_URL).content
|
content = s.get(constant.LOGIN_URL).content
|
||||||
html = BeautifulSoup(content, 'html.parser').encode("ascii")
|
html = BeautifulSoup(content, 'html.parser')
|
||||||
csrf_token_elem = html.find('input', attrs={'name': 'csrfmiddlewaretoken'})
|
csrf_token_elem = html.find('input', attrs={'name': 'csrfmiddlewaretoken'})
|
||||||
|
|
||||||
if not csrf_token_elem:
|
if not csrf_token_elem:
|
||||||
@ -44,14 +44,22 @@ def login_parser(username, password):
|
|||||||
logger.error('Login failed, please check your username and password')
|
logger.error('Login failed, please check your username and password')
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
html = BeautifulSoup(s.get(constant.FAV_URL).content, 'html.parser').encode("ascii")
|
html = BeautifulSoup(s.get(constant.FAV_URL).content, 'html.parser')
|
||||||
count = html.find('span', attrs={'class': 'count'})
|
count = html.find('span', attrs={'class': 'count'})
|
||||||
if not count:
|
if not count:
|
||||||
logger.error('Cannot get count of your favorites, maybe login failed.')
|
logger.error('Cannot get count of your favorites, maybe login failed.')
|
||||||
|
|
||||||
count = int(count.text.strip('(').strip(')'))
|
count = int(count.text.strip('(').strip(')'))
|
||||||
pages = count / 25
|
if count == 0:
|
||||||
pages += 1 if count % (25 * pages) else 0
|
logger.warning('No favorites found')
|
||||||
|
return []
|
||||||
|
pages = int(count / 25)
|
||||||
|
|
||||||
|
if pages:
|
||||||
|
pages += 1 if count % (25 * pages) else 0
|
||||||
|
else:
|
||||||
|
pages = 1
|
||||||
|
|
||||||
logger.info('Your have %d favorites in %d pages.' % (count, pages))
|
logger.info('Your have %d favorites in %d pages.' % (count, pages))
|
||||||
|
|
||||||
if os.getenv('DEBUG'):
|
if os.getenv('DEBUG'):
|
||||||
@ -68,7 +76,7 @@ def login_parser(username, password):
|
|||||||
for page in range(1, pages+1):
|
for page in range(1, pages+1):
|
||||||
try:
|
try:
|
||||||
logger.info('Getting doujinshi id of page %d' % page)
|
logger.info('Getting doujinshi id of page %d' % page)
|
||||||
resp = s.get(constant.FAV_URL + '?page=%d' % page).content
|
resp = s.get(constant.FAV_URL + '?page=%d' % page).text
|
||||||
ids = doujinshi_id.findall(resp)
|
ids = doujinshi_id.findall(resp)
|
||||||
requests_ = threadpool.makeRequests(doujinshi_parser, ids, _callback)
|
requests_ = threadpool.makeRequests(doujinshi_parser, ids, _callback)
|
||||||
[thread_pool.putRequest(req) for req in requests_]
|
[thread_pool.putRequest(req) for req in requests_]
|
||||||
@ -102,11 +110,18 @@ def doujinshi_parser(id_):
|
|||||||
doujinshi['pages'] = len(response['images']['pages'])
|
doujinshi['pages'] = len(response['images']['pages'])
|
||||||
|
|
||||||
# gain information of the doujinshi
|
# gain information of the doujinshi
|
||||||
needed_fields = ['character', 'artist', 'language']
|
needed_fields = ['character', 'artist', 'language', 'tag']
|
||||||
for tag in response['tags']:
|
for tag in response['tags']:
|
||||||
tag_type = tag['type']
|
tag_type = tag['type']
|
||||||
if tag_type in needed_fields:
|
if tag_type in needed_fields:
|
||||||
if tag_type not in doujinshi:
|
if tag_type == 'tag':
|
||||||
|
if tag_type not in doujinshi:
|
||||||
|
doujinshi[tag_type] = {}
|
||||||
|
|
||||||
|
tag['name'] = tag['name'].replace(' ', '-')
|
||||||
|
tag['name'] = tag['name'].lower()
|
||||||
|
doujinshi[tag_type][tag['name']] = tag['id']
|
||||||
|
elif tag_type not in doujinshi:
|
||||||
doujinshi[tag_type] = tag['name']
|
doujinshi[tag_type] = tag['name']
|
||||||
else:
|
else:
|
||||||
doujinshi[tag_type] += tag['name']
|
doujinshi[tag_type] += tag['name']
|
||||||
@ -146,5 +161,48 @@ def print_doujinshi(doujinshi_list):
|
|||||||
tabulate(tabular_data=doujinshi_list, headers=headers, tablefmt='rst'))
|
tabulate(tabular_data=doujinshi_list, headers=headers, tablefmt='rst'))
|
||||||
|
|
||||||
|
|
||||||
|
def tag_parser(tag_id):
|
||||||
|
logger.info('Get doujinshi of tag id: {0}'.format(tag_id))
|
||||||
|
result = []
|
||||||
|
response = request('get', url=constant.TAG_API_URL, params={'sort': 'popular', 'tag_id': tag_id}).json()
|
||||||
|
|
||||||
|
for row in response['result']:
|
||||||
|
title = row['title']['english']
|
||||||
|
title = title[:85] + '..' if len(title) > 85 else title
|
||||||
|
result.append({'id': row['id'], 'title': title})
|
||||||
|
|
||||||
|
if not result:
|
||||||
|
logger.warn('Not found anything of tag id {}'.format(tag_id))
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def tag_guessing(tag_name):
|
||||||
|
tag_name = tag_name.lower()
|
||||||
|
tag_name = tag_name.replace(' ', '-')
|
||||||
|
logger.info('Trying to get tag_id of tag \'{0}\''.format(tag_name))
|
||||||
|
response = request('get', url='%s/%s' % (constant.TAG_URL, tag_name)).content
|
||||||
|
html = BeautifulSoup(response, 'html.parser')
|
||||||
|
first_item = html.find('div', attrs={'class': 'gallery'})
|
||||||
|
if not first_item:
|
||||||
|
logger.error('Cannot find doujinshi id of tag \'{0}\''.format(tag_name))
|
||||||
|
return
|
||||||
|
|
||||||
|
doujinshi_id = re.findall('(\d+)', first_item.a.attrs['href'])
|
||||||
|
if not doujinshi_id:
|
||||||
|
logger.error('Cannot find doujinshi id of tag \'{0}\''.format(tag_name))
|
||||||
|
return
|
||||||
|
|
||||||
|
ret = doujinshi_parser(doujinshi_id[0])
|
||||||
|
if 'tag' in ret and tag_name in ret['tag']:
|
||||||
|
tag_id = ret['tag'][tag_name]
|
||||||
|
logger.info('Tag id of tag \'{0}\' is {1}'.format(tag_name, tag_id))
|
||||||
|
else:
|
||||||
|
logger.error('Cannot find doujinshi id of tag \'{0}\''.format(tag_name))
|
||||||
|
return
|
||||||
|
|
||||||
|
return tag_id
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
print(doujinshi_parser("32271"))
|
print(doujinshi_parser("32271"))
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
from __future__ import unicode_literals, print_function
|
from __future__ import unicode_literals, print_function
|
||||||
|
|
||||||
|
import sys
|
||||||
import os
|
import os
|
||||||
import string
|
import string
|
||||||
import zipfile
|
import zipfile
|
||||||
@ -30,12 +31,14 @@ def urlparse(url):
|
|||||||
|
|
||||||
return urlparse(url)
|
return urlparse(url)
|
||||||
|
|
||||||
|
|
||||||
def readfile(path):
|
def readfile(path):
|
||||||
loc = os.path.dirname(__file__)
|
loc = os.path.dirname(__file__)
|
||||||
|
|
||||||
with open(os.path.join(loc, path), 'r') as file:
|
with open(os.path.join(loc, path), 'r') as file:
|
||||||
return file.read()
|
return file.read()
|
||||||
|
|
||||||
|
|
||||||
def generate_html(output_dir='.', doujinshi_obj=None):
|
def generate_html(output_dir='.', doujinshi_obj=None):
|
||||||
image_html = ''
|
image_html = ''
|
||||||
|
|
||||||
@ -65,10 +68,17 @@ def generate_html(output_dir='.', doujinshi_obj=None):
|
|||||||
title = 'nHentai HTML Viewer'
|
title = 'nHentai HTML Viewer'
|
||||||
|
|
||||||
data = html.format(TITLE=title, IMAGES=image_html, SCRIPTS=js, STYLES=css)
|
data = html.format(TITLE=title, IMAGES=image_html, SCRIPTS=js, STYLES=css)
|
||||||
with open(os.path.join(doujinshi_dir, 'index.html'), 'w') as f:
|
try:
|
||||||
f.write(data)
|
if sys.version_info < (3, 0):
|
||||||
|
with open(os.path.join(doujinshi_dir, 'index.html'), 'w') as f:
|
||||||
|
f.write(data)
|
||||||
|
else:
|
||||||
|
with open(os.path.join(doujinshi_dir, 'index.html'), 'wb') as f:
|
||||||
|
f.write(data.encode('utf-8'))
|
||||||
|
|
||||||
logger.log(15, 'HTML Viewer has been write to \'{0}\''.format(os.path.join(doujinshi_dir, 'index.html')))
|
logger.log(15, 'HTML Viewer has been write to \'{0}\''.format(os.path.join(doujinshi_dir, 'index.html')))
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning('Writen HTML Viewer failed ({})'.format(str(e)))
|
||||||
|
|
||||||
|
|
||||||
def generate_cbz(output_dir='.', doujinshi_obj=None):
|
def generate_cbz(output_dir='.', doujinshi_obj=None):
|
||||||
|
@ -1,24 +1,24 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<title>{TITLE}</title>
|
<title>{TITLE}</title>
|
||||||
<style>
|
<style>
|
||||||
{STYLES}
|
{STYLES}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
|
||||||
<nav id="list">
|
<nav id="list">
|
||||||
{IMAGES}</nav>
|
{IMAGES}</nav>
|
||||||
|
|
||||||
<div id="image-container">
|
<div id="image-container">
|
||||||
<span id="page-num"></span>
|
<span id="page-num"></span>
|
||||||
<div id="dest"></div>
|
<div id="dest"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
{SCRIPTS}
|
{SCRIPTS}
|
||||||
</script>
|
</script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
@ -1,62 +1,62 @@
|
|||||||
const pages = Array.from(document.querySelectorAll('img.image-item'));
|
const pages = Array.from(document.querySelectorAll('img.image-item'));
|
||||||
let currentPage = 0;
|
let currentPage = 0;
|
||||||
|
|
||||||
function changePage(pageNum) {
|
function changePage(pageNum) {
|
||||||
const previous = pages[currentPage];
|
const previous = pages[currentPage];
|
||||||
const current = pages[pageNum];
|
const current = pages[pageNum];
|
||||||
|
|
||||||
if (current == null) {
|
if (current == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
previous.classList.remove('current');
|
previous.classList.remove('current');
|
||||||
current.classList.add('current');
|
current.classList.add('current');
|
||||||
|
|
||||||
currentPage = pageNum;
|
currentPage = pageNum;
|
||||||
|
|
||||||
const display = document.getElementById('dest');
|
const display = document.getElementById('dest');
|
||||||
display.style.backgroundImage = `url("${current.src}")`;
|
display.style.backgroundImage = `url("${current.src}")`;
|
||||||
|
|
||||||
document.getElementById('page-num')
|
document.getElementById('page-num')
|
||||||
.innerText = [
|
.innerText = [
|
||||||
(pageNum + 1).toLocaleString(),
|
(pageNum + 1).toLocaleString(),
|
||||||
pages.length.toLocaleString()
|
pages.length.toLocaleString()
|
||||||
].join('\u200a/\u200a');
|
].join('\u200a/\u200a');
|
||||||
}
|
}
|
||||||
|
|
||||||
changePage(0);
|
changePage(0);
|
||||||
|
|
||||||
document.getElementById('list').onclick = event => {
|
document.getElementById('list').onclick = event => {
|
||||||
if (pages.includes(event.target)) {
|
if (pages.includes(event.target)) {
|
||||||
changePage(pages.indexOf(event.target));
|
changePage(pages.indexOf(event.target));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
document.getElementById('image-container').onclick = event => {
|
document.getElementById('image-container').onclick = event => {
|
||||||
const width = document.getElementById('image-container').clientWidth;
|
const width = document.getElementById('image-container').clientWidth;
|
||||||
const clickPos = event.clientX / width;
|
const clickPos = event.clientX / width;
|
||||||
|
|
||||||
if (clickPos < 0.5) {
|
if (clickPos < 0.5) {
|
||||||
changePage(currentPage - 1);
|
changePage(currentPage - 1);
|
||||||
} else {
|
} else {
|
||||||
changePage(currentPage + 1);
|
changePage(currentPage + 1);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
document.onkeypress = event => {
|
document.onkeypress = event => {
|
||||||
switch (event.key.toLowerCase()) {
|
switch (event.key.toLowerCase()) {
|
||||||
// Previous Image
|
// Previous Image
|
||||||
case 'arrowleft':
|
case 'arrowleft':
|
||||||
case 'a':
|
case 'a':
|
||||||
changePage(currentPage - 1);
|
changePage(currentPage - 1);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
// Next Image
|
// Next Image
|
||||||
case ' ':
|
case ' ':
|
||||||
case 'enter':
|
case 'enter':
|
||||||
case 'arrowright':
|
case 'arrowright':
|
||||||
case 'd':
|
case 'd':
|
||||||
changePage(currentPage + 1);
|
changePage(currentPage + 1);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
};
|
};
|
@ -1,69 +1,69 @@
|
|||||||
*, *::after, *::before {
|
*, *::after, *::before {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
img {
|
img {
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
html, body {
|
html, body {
|
||||||
display: flex;
|
display: flex;
|
||||||
background-color: #e8e6e6;
|
background-color: #e8e6e6;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-family: sans-serif;
|
font-family: sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
#list {
|
#list {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
width: 260px;
|
width: 260px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
#list img {
|
#list img {
|
||||||
width: 200px;
|
width: 200px;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
margin: 15px 0;
|
margin: 15px 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
#list img.current {
|
#list img.current {
|
||||||
background: #0003;
|
background: #0003;
|
||||||
}
|
}
|
||||||
|
|
||||||
#image-container {
|
#image-container {
|
||||||
flex: auto;
|
flex: auto;
|
||||||
height: 100vh;
|
height: 100vh;
|
||||||
background: #222;
|
background: #222;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
#image-container #dest {
|
#image-container #dest {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
background-size: contain;
|
background-size: contain;
|
||||||
background-repeat: no-repeat;
|
background-repeat: no-repeat;
|
||||||
background-position: center;
|
background-position: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
#image-container #page-num {
|
#image-container #page-num {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
font-size: 18pt;
|
font-size: 18pt;
|
||||||
left: 10px;
|
left: 10px;
|
||||||
bottom: 5px;
|
bottom: 5px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
opacity: 0.75;
|
opacity: 0.75;
|
||||||
text-shadow: /* Duplicate the same shadow to make it very strong */
|
text-shadow: /* Duplicate the same shadow to make it very strong */
|
||||||
0 0 2px #222,
|
0 0 2px #222,
|
||||||
0 0 2px #222,
|
0 0 2px #222,
|
||||||
0 0 2px #222;
|
0 0 2px #222;
|
||||||
}
|
}
|
@ -2,4 +2,4 @@ requests>=2.5.0
|
|||||||
BeautifulSoup4>=4.0.0
|
BeautifulSoup4>=4.0.0
|
||||||
threadpool>=1.2.7
|
threadpool>=1.2.7
|
||||||
tabulate>=0.7.5
|
tabulate>=0.7.5
|
||||||
future>=0.15.2
|
future>=0.15.2threadpool==1.3.2
|
||||||
|
Reference in New Issue
Block a user