1. 최초커밋

This commit is contained in:
2021-08-03 08:58:29 +09:00
commit 3a53a27240
10 changed files with 1562 additions and 0 deletions

398
.gitignore vendored Normal file
View File

@@ -0,0 +1,398 @@
# Created by https://www.toptal.com/developers/gitignore/api/visualstudio
# Edit at https://www.toptal.com/developers/gitignore?templates=visualstudio
### VisualStudio ###
## Ignore Visual Studio temporary files, build results, and
## files generated by popular Visual Studio add-ons.
##
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
# User-specific files
*.rsuser
*.suo
*.user
*.userosscache
*.sln.docstates
# User-specific files (MonoDevelop/Xamarin Studio)
*.userprefs
# Mono auto generated files
mono_crash.*
# Build results
[Dd]ebug/
[Dd]ebugPublic/
[Rr]elease/
[Rr]eleases/
x64/
x86/
[Ww][Ii][Nn]32/
[Aa][Rr][Mm]/
[Aa][Rr][Mm]64/
bld/
[Bb]in/
[Oo]bj/
[Ll]og/
[Ll]ogs/
# Visual Studio 2015/2017 cache/options directory
.vs/
# Uncomment if you have tasks that create the project's static files in wwwroot
#wwwroot/
# Visual Studio 2017 auto generated files
Generated\ Files/
# MSTest test Results
[Tt]est[Rr]esult*/
[Bb]uild[Ll]og.*
# NUnit
*.VisualState.xml
TestResult.xml
nunit-*.xml
# Build Results of an ATL Project
[Dd]ebugPS/
[Rr]eleasePS/
dlldata.c
# Benchmark Results
BenchmarkDotNet.Artifacts/
# .NET Core
project.lock.json
project.fragment.lock.json
artifacts/
# ASP.NET Scaffolding
ScaffoldingReadMe.txt
# StyleCop
StyleCopReport.xml
# Files built by Visual Studio
*_i.c
*_p.c
*_h.h
*.ilk
*.meta
*.obj
*.iobj
*.pch
*.pdb
*.ipdb
*.pgc
*.pgd
*.rsp
*.sbr
*.tlb
*.tli
*.tlh
*.tmp
*.tmp_proj
*_wpftmp.csproj
*.log
*.tlog
*.vspscc
*.vssscc
.builds
*.pidb
*.svclog
*.scc
# Chutzpah Test files
_Chutzpah*
# Visual C++ cache files
ipch/
*.aps
*.ncb
*.opendb
*.opensdf
*.sdf
*.cachefile
*.VC.db
*.VC.VC.opendb
# Visual Studio profiler
*.psess
*.vsp
*.vspx
*.sap
# Visual Studio Trace Files
*.e2e
# TFS 2012 Local Workspace
$tf/
# Guidance Automation Toolkit
*.gpState
# ReSharper is a .NET coding add-in
_ReSharper*/
*.[Rr]e[Ss]harper
*.DotSettings.user
# TeamCity is a build add-in
_TeamCity*
# DotCover is a Code Coverage Tool
*.dotCover
# AxoCover is a Code Coverage Tool
.axoCover/*
!.axoCover/settings.json
# Coverlet is a free, cross platform Code Coverage Tool
coverage*.json
coverage*.xml
coverage*.info
# Visual Studio code coverage results
*.coverage
*.coveragexml
# NCrunch
_NCrunch_*
.*crunch*.local.xml
nCrunchTemp_*
# MightyMoose
*.mm.*
AutoTest.Net/
# Web workbench (sass)
.sass-cache/
# Installshield output folder
[Ee]xpress/
# DocProject is a documentation generator add-in
DocProject/buildhelp/
DocProject/Help/*.HxT
DocProject/Help/*.HxC
DocProject/Help/*.hhc
DocProject/Help/*.hhk
DocProject/Help/*.hhp
DocProject/Help/Html2
DocProject/Help/html
# Click-Once directory
publish/
# Publish Web Output
*.[Pp]ublish.xml
*.azurePubxml
# Note: Comment the next line if you want to checkin your web deploy settings,
# but database connection strings (with potential passwords) will be unencrypted
*.pubxml
*.publishproj
# Microsoft Azure Web App publish settings. Comment the next line if you want to
# checkin your Azure Web App publish settings, but sensitive information contained
# in these scripts will be unencrypted
PublishScripts/
# NuGet Packages
*.nupkg
# NuGet Symbol Packages
*.snupkg
# The packages folder can be ignored because of Package Restore
**/[Pp]ackages/*
# except build/, which is used as an MSBuild target.
!**/[Pp]ackages/build/
# Uncomment if necessary however generally it will be regenerated when needed
#!**/[Pp]ackages/repositories.config
# NuGet v3's project.json files produces more ignorable files
*.nuget.props
*.nuget.targets
# Nuget personal access tokens and Credentials
nuget.config
# Microsoft Azure Build Output
csx/
*.build.csdef
# Microsoft Azure Emulator
ecf/
rcf/
# Windows Store app package directories and files
AppPackages/
BundleArtifacts/
Package.StoreAssociation.xml
_pkginfo.txt
*.appx
*.appxbundle
*.appxupload
# Visual Studio cache files
# files ending in .cache can be ignored
*.[Cc]ache
# but keep track of directories ending in .cache
!?*.[Cc]ache/
# Others
ClientBin/
~$*
*~
*.dbmdl
*.dbproj.schemaview
*.jfm
*.pfx
*.publishsettings
orleans.codegen.cs
# Including strong name files can present a security risk
# (https://github.com/github/gitignore/pull/2483#issue-259490424)
#*.snk
# Since there are multiple workflows, uncomment next line to ignore bower_components
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
#bower_components/
# RIA/Silverlight projects
Generated_Code/
# Backup & report files from converting an old project file
# to a newer Visual Studio version. Backup files are not needed,
# because we have git ;-)
_UpgradeReport_Files/
Backup*/
UpgradeLog*.XML
UpgradeLog*.htm
ServiceFabricBackup/
*.rptproj.bak
# SQL Server files
*.mdf
*.ldf
*.ndf
# Business Intelligence projects
*.rdl.data
*.bim.layout
*.bim_*.settings
*.rptproj.rsuser
*- [Bb]ackup.rdl
*- [Bb]ackup ([0-9]).rdl
*- [Bb]ackup ([0-9][0-9]).rdl
# Microsoft Fakes
FakesAssemblies/
# GhostDoc plugin setting file
*.GhostDoc.xml
# Node.js Tools for Visual Studio
.ntvs_analysis.dat
node_modules/
# Visual Studio 6 build log
*.plg
# Visual Studio 6 workspace options file
*.opt
# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)
*.vbw
# Visual Studio LightSwitch build output
**/*.HTMLClient/GeneratedArtifacts
**/*.DesktopClient/GeneratedArtifacts
**/*.DesktopClient/ModelManifest.xml
**/*.Server/GeneratedArtifacts
**/*.Server/ModelManifest.xml
_Pvt_Extensions
# Paket dependency manager
.paket/paket.exe
paket-files/
# FAKE - F# Make
.fake/
# CodeRush personal settings
.cr/personal
# Python Tools for Visual Studio (PTVS)
__pycache__/
*.pyc
# Cake - Uncomment if you are using it
# tools/**
# !tools/packages.config
# Tabs Studio
*.tss
# Telerik's JustMock configuration file
*.jmconfig
# BizTalk build output
*.btp.cs
*.btm.cs
*.odx.cs
*.xsd.cs
# OpenCover UI analysis results
OpenCover/
# Azure Stream Analytics local run output
ASALocalRun/
# MSBuild Binary and Structured Log
*.binlog
# NVidia Nsight GPU debugger configuration file
*.nvuser
# MFractors (Xamarin productivity tool) working folder
.mfractor/
# Local History for Visual Studio
.localhistory/
# BeatPulse healthcheck temp database
healthchecksdb
# Backup folder for Package Reference Convert tool in Visual Studio 2017
MigrationBackup/
# Ionide (cross platform F# VS Code tools) working folder
.ionide/
# Fody - auto-generated XML schema
FodyWeavers.xsd
# VS Code files for those working on multiple tools
.vscode/*
!.vscode/settings.json
!.vscode/tasks.json
!.vscode/launch.json
!.vscode/extensions.json
*.code-workspace
# Local History for Visual Studio Code
.history/
# Windows Installer files from build outputs
*.cab
*.msi
*.msix
*.msm
*.msp
# JetBrains Rider
.idea/
*.sln.iml
### VisualStudio Patch ###
# Additional files built by Visual Studio
# End of https://www.toptal.com/developers/gitignore/api/visualstudio

25
NaverSearcher.sln Normal file
View File

@@ -0,0 +1,25 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 16
VisualStudioVersion = 16.0.31424.327
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NaverSearcher", "NaverSearcher\NaverSearcher.csproj", "{54262938-66CB-4A5F-926B-48C3ACDCE9CF}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{54262938-66CB-4A5F-926B-48C3ACDCE9CF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{54262938-66CB-4A5F-926B-48C3ACDCE9CF}.Debug|Any CPU.Build.0 = Debug|Any CPU
{54262938-66CB-4A5F-926B-48C3ACDCE9CF}.Release|Any CPU.ActiveCfg = Release|Any CPU
{54262938-66CB-4A5F-926B-48C3ACDCE9CF}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {D85D7D5D-9754-4B34-806A-EC1878B37547}
EndGlobalSection
EndGlobal

64
NaverSearcher/Form1.Designer.cs generated Normal file
View File

@@ -0,0 +1,64 @@

namespace NaverSearcher
{
partial class Form1
{
/// <summary>
/// Required designer variable.
/// </summary>
private System.ComponentModel.IContainer components = null;
/// <summary>
/// Clean up any resources being used.
/// </summary>
/// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param>
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
#region Windows Form Designer generated code
/// <summary>
/// Required method for Designer support - do not modify
/// the contents of this method with the code editor.
/// </summary>
private void InitializeComponent()
{
this.button1 = new System.Windows.Forms.Button();
this.SuspendLayout();
//
// button1
//
this.button1.Location = new System.Drawing.Point(562, 269);
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(75, 23);
this.button1.TabIndex = 0;
this.button1.Text = "button1";
this.button1.UseVisualStyleBackColor = true;
this.button1.Click += new System.EventHandler(this.button1_Click);
//
// Form1
//
this.AutoScaleDimensions = new System.Drawing.SizeF(7F, 15F);
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
this.ClientSize = new System.Drawing.Size(800, 450);
this.Controls.Add(this.button1);
this.Name = "Form1";
this.Text = "Form1";
this.FormClosing += new System.Windows.Forms.FormClosingEventHandler(this.Form1_FormClosing);
this.Load += new System.EventHandler(this.Form1_Load);
this.ResumeLayout(false);
}
#endregion
private System.Windows.Forms.Button button1;
}
}

285
NaverSearcher/Form1.cs Normal file
View File

@@ -0,0 +1,285 @@
using OpenQA.Selenium;
using OpenQA.Selenium.Chrome;
using OpenQA.Selenium.Support.UI;
using SeleniumExtras.WaitHelpers;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace NaverSearcher
{
public partial class Form1 : Form
{
ChromeDriver _ChromeDriver;
public Form1()
{
InitializeComponent();
ChromeOptions _ChromeOptions = new ChromeOptions();
_ChromeOptions.AddArguments("disable-infobars");
_ChromeOptions.AddArguments("--js-flags=--expose-gc");
_ChromeOptions.AddArguments("--enable-precise-memory-info");
_ChromeOptions.AddArguments("--disable-popup-blocking");
_ChromeOptions.AddArguments("--disable-default-apps");
_ChromeOptions.AddArguments("--headless");
_ChromeOptions.AddArguments("user - agent = Mozilla / 5.0(Macintosh; Intel Mac OS X 10_12_6) AppleWebKit / 537.36(KHTML, like Gecko) Chrome / 61.0.3163.100 Safari / 537.36");
// 프록시 설정
//Proxy proxy = new Proxy();
//proxy.Kind = ProxyKind.Manual;
//proxy.IsAutoDetect = false;
//proxy.HttpProxy =
//proxy.SslProxy = ip;
//_ChromeOptions.Proxy = proxy;
//_ChromeOptions.AddArgument("ignore-certificate-errors");
ChromeDriverService _ChromeDriverService = ChromeDriverService.CreateDefaultService();
_ChromeDriverService.HideCommandPromptWindow = true;
_ChromeDriver = new ChromeDriver(_ChromeDriverService, _ChromeOptions);
//_ChromeDriver = new ChromeDriver();
//_ChromeOptions.add_argument("user-agent=Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; rv:11.0) like Gecko")
// 하루에 두번 검색
// {{바나나우유 라떼, 초코우유 라떼}, {민트초코바, 민트색}}
// 검색어 1페어
// 로그인 여부 확인
// 로그아웃 실행
// 네이버 메인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 검색어 입력 (검색어 1)
// 검색어 확인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 아무 게시글 클릭
// 이동된 페이지 작업
// 스크롤 내리기
// 20 ~ 60초 대기
// 새창 닫기
// 검색어 입력 (검색어 2, 기존 대기창)
// 검색어 확인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 아무 게시글 클릭
// 이동된 페이지 작업
// 스크롤 내리기
// 20 ~ 60초 대기
// 새창 닫기
// 아무거나 검색 (비선형 검색어)
// 검색어 확인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 아무 게시글 클릭
// 이동된 페이지 작업
// 스크롤 내리기
// 20 ~ 60초 대기
// 새창 닫기
// 검색어 2페어
// 로그인 여부 확인
// 로그아웃 실행
// 네이버 메인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 검색어 입력 (검색어 1)
// 검색어 확인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 아무 게시글 클릭
// 이동된 페이지 작업
// 스크롤 내리기
// 20 ~ 60초 대기
// 전부 닫기
// 로그인 여부 확인
// 로그아웃 실행
// 네이버 메인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 검색어 입력 (검색어 2, 기존 대기창)
// 검색어 확인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 아무 게시글 클릭
// 이동된 페이지 작업
// 스크롤 내리기
// 20 ~ 60초 대기
// 새창 닫기
// 아무거나 검색 (비선형 검색어)
// 검색어 확인
// 스크롤 내린다.
// 20 ~ 60초 대기
// 아무 게시글 클릭
// 이동된 페이지 작업
// 스크롤 내리기
// 20 ~ 60초 대기
// 새창 닫기
}
private void button1_Click(object sender, EventArgs e)
{
try
{
/*
_ChromeDriver.execute_script('window.open("about:blank", "_blank");')
_ChromeDriver.execute_script('window.open("about:blank", "_blank");')
tabs = _ChromeDriver.window_handles
# TAB_1
_ChromeDriver.switch_to_window(tabs[0])
_ChromeDriver.get('http://www.naver.com/')
# TAB_2
_ChromeDriver.switch_to_window(tabs[1])
_ChromeDriver.get('http://www.google.com/')
*/
_ChromeDriver.Navigate().GoToUrl(@"https://naver.com");
WebDriverWait _WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
var elem = _ChromeDriver.FindElementByXPath("//*");
var source_code = elem.GetAttribute("outerHTML");
((IJavaScriptExecutor)_ChromeDriver).ExecuteScript("window.open();");
_ChromeDriver.SwitchTo().Window(_ChromeDriver.WindowHandles[1]);
string winHandleBefore = _ChromeDriver.CurrentWindowHandle;
//_ChromeDriver.SwitchTo().Window(winHandleBefore);
_WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
elem = _ChromeDriver.FindElementByXPath("//*");
source_code = elem.GetAttribute("outerHTML");
_ChromeDriver.SwitchTo().Window(_ChromeDriver.WindowHandles[1]);
winHandleBefore = _ChromeDriver.CurrentWindowHandle;
_WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
elem = _ChromeDriver.FindElementByXPath("//*");
source_code = elem.GetAttribute("outerHTML");
_ChromeDriver.Navigate().GoToUrl(@"https://inrose.com");
_WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
elem = _ChromeDriver.FindElementByXPath("//*");
source_code = elem.GetAttribute("outerHTML");
_ChromeDriver.SwitchTo().Window(_ChromeDriver.WindowHandles.First());
winHandleBefore = _ChromeDriver.CurrentWindowHandle;
_WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
elem = _ChromeDriver.FindElementByXPath("//*");
source_code = elem.GetAttribute("outerHTML");
_ChromeDriver.Navigate().GoToUrl(@"https://google.com");
_WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
elem = _ChromeDriver.FindElementByXPath("//*");
source_code = elem.GetAttribute("outerHTML");
return;
_ChromeDriver.Navigate().GoToUrl(@"https://www.instagram.com/");
_WebDriverWait = new WebDriverWait(_ChromeDriver, TimeSpan.FromSeconds(3));
//_WebDriverWait.IgnoreExceptionTypes(typeof(NoSuchElementException));
string _id_xpath = "//*[@id=\"loginForm\"]/div/div[1]/div/label/input";
string _pass_xpath = "//*[@id=\"loginForm\"]/div/div[2]/div/label/input";
string _login_xpath = "//*[@id=\"loginForm\"]/div/div[3]/button/div";
_WebDriverWait.Until(ExpectedConditions.ElementIsVisible(By.XPath(_login_xpath)));
_ChromeDriver.FindElement(By.XPath(_id_xpath)).SendKeys(@"leeumdkj@gmail.com");
_ChromeDriver.FindElement(By.XPath(_pass_xpath)).SendKeys(@"awdr1536!!");
_ChromeDriver.FindElement(By.XPath(_login_xpath)).Click();
// 알림 존재시
string _alert_xpath = "//*[@id=\"react-root\"]/section/main/div/div/div/div/button";
_WebDriverWait.Until(ExpectedConditions.ElementIsVisible(By.XPath(_alert_xpath)));
try
{
_ChromeDriver.FindElement(By.XPath(_alert_xpath)).Click();
}
catch (Exception)
{
}
_alert_xpath = "/html/body/div[4]/div/div/div/div[3]/button[2]";
_WebDriverWait.Until(ExpectedConditions.ElementIsVisible(By.XPath(_alert_xpath)));
try
{
_ChromeDriver.FindElement(By.XPath(_alert_xpath)).Click();
}
catch (Exception)
{
}
string _search_xpath = "//*[@id=\"react-root\"]/section/nav/div[2]/div/div/div[2]/input";
_WebDriverWait.Until(ExpectedConditions.ElementIsVisible(By.XPath(_search_xpath)));
_ChromeDriver.FindElement(By.XPath(_search_xpath)).SendKeys(@"인플 검색");
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
}
private void Form1_Load(object sender, EventArgs e)
{
#region
string kknd = HangulString.SplitToPhonemes("ㅄ");
MessageBox.Show(kknd);
char[] arr = kknd.ToCharArray();
foreach (var item in arr)
{
Trace.WriteLine(item);
}
#endregion
}
private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
_ChromeDriver.Quit();
}
}
}

60
NaverSearcher/Form1.resx Normal file
View File

@@ -0,0 +1,60 @@
<root>
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
<xsd:element name="root" msdata:IsDataSet="true">
<xsd:complexType>
<xsd:choice maxOccurs="unbounded">
<xsd:element name="metadata">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" />
</xsd:sequence>
<xsd:attribute name="name" use="required" type="xsd:string" />
<xsd:attribute name="type" type="xsd:string" />
<xsd:attribute name="mimetype" type="xsd:string" />
<xsd:attribute ref="xml:space" />
</xsd:complexType>
</xsd:element>
<xsd:element name="assembly">
<xsd:complexType>
<xsd:attribute name="alias" type="xsd:string" />
<xsd:attribute name="name" type="xsd:string" />
</xsd:complexType>
</xsd:element>
<xsd:element name="data">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
</xsd:sequence>
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
<xsd:attribute ref="xml:space" />
</xsd:complexType>
</xsd:element>
<xsd:element name="resheader">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
</xsd:sequence>
<xsd:attribute name="name" type="xsd:string" use="required" />
</xsd:complexType>
</xsd:element>
</xsd:choice>
</xsd:complexType>
</xsd:element>
</xsd:schema>
<resheader name="resmimetype">
<value>text/microsoft-resx</value>
</resheader>
<resheader name="version">
<value>2.0</value>
</resheader>
<resheader name="reader">
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader>
<resheader name="writer">
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader>
</root>

292
NaverSearcher/HangulChar.cs Normal file
View File

@@ -0,0 +1,292 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace NaverSearcher
{
/// <summary>
/// 한글 문자 클래스: 현대 한글 문자에 대해 다양한 속성과 메서드를 제공하는 클래스입니다.
/// </summary>
public class HangulChar
{
// 한글에 대한 유니코드 범위 [44032, 55215]
private const int BeginCode = 0xAC00;
private const int EndCode = 0xD7AF;
// 문자 구성: 초성, 중성, 종성
private static readonly char[] Onset = { 'ㄱ', 'ㄲ', 'ㄴ', 'ㄷ', 'ㄸ', 'ㄹ', 'ㅁ', 'ㅂ', 'ㅃ', 'ㅅ', 'ㅆ',
'ㅇ', 'ㅈ', 'ㅉ', 'ㅊ', 'ㅋ', 'ㅌ', 'ㅍ', 'ㅎ' };
private static readonly char[] Nucleus = { 'ㅏ', 'ㅐ', 'ㅑ', 'ㅒ', 'ㅓ', 'ㅔ', 'ㅕ', 'ㅖ', 'ㅗ', 'ㅘ', 'ㅙ',
'ㅚ', 'ㅛ', 'ㅜ', 'ㅝ', 'ㅞ', 'ㅟ', 'ㅠ', 'ㅡ', 'ㅢ', 'ㅣ' };
private static readonly char[] Coda = { (char)0x00, 'ㄱ', 'ㄲ', 'ㄳ', 'ㄴ', 'ㄵ', 'ㄶ', 'ㄷ', 'ㄹ', 'ㄺ', 'ㄻ',
'ㄼ', 'ㄽ', 'ㄾ', 'ㄿ', 'ㅀ', 'ㅁ', 'ㅂ', 'ㅄ', 'ㅅ', 'ㅆ', 'ㅇ', 'ㅈ', 'ㅊ', 'ㅋ', 'ㅌ', 'ㅍ', 'ㅎ' };
// 문자 획수: 초성, 중성, 종성
private static readonly int[] OnsetStrokes = { 1, 2, 1, 2, 4, 3, 3, 4, 8, 2, 4, 1, 2, 4, 3, 2, 3, 4, 3 };
private static readonly int[] NucleusStrokes =
{ 2, 3, 3, 4, 2, 3, 3, 4, 2, 4, 5, 3, 3, 2, 4, 5, 3, 3, 1, 2, 1 };
private static readonly int[] CodaStrokes =
{ 0, 1, 2, 3, 1, 3, 4, 2, 3, 4, 6, 7, 5, 6, 7, 6, 3, 4, 6, 2, 4, 1, 2, 3, 2, 3, 4, 3 };
/// <summary>
/// 문자로부터 한글 문자 클래스의 인스턴스를 생성합니다.
/// </summary>
/// <param name="character">한글 문자(char)</param>
public HangulChar(char character)
{
this.CurrentCharacter = character;
}
/// <summary>
/// 유니코드로부터 한글 문자 클래스의 인스턴스를 생성합니다.
/// </summary>
/// <param name="unicode">한글 문자(char)에 해당하는 유니코드</param>
public HangulChar(int unicode)
{
this.CurrentCharacter = Convert.ToChar(unicode);
}
/// <summary>
/// 현재 인스턴스의 문자입니다.
/// </summary>
public char CurrentCharacter { get; private set; }
/// <summary>
/// 현재 인스턴스 문자의 유니코드입니다.
/// </summary>
public int CurrentUnicode => (int)CurrentCharacter;
/// <summary>
/// 인자의 음소 배열을 한글 음절로 합성합니다.
/// 반환값은 합성의 성공(가능) 여부를 나타냅니다.
/// </summary>
/// <param name="phonemes">초성, 중성(, 종성) 순의 한글 음소 배열</param>
/// <param name="syllable">합성된 한글 음절</param>
/// <returns></returns>
/// <exception cref="ArgumentNullException"><paramref name="phonemes"/>이 null로 주어질 경우</exception>
public static bool TryJoinToSyllable(char[] phonemes, out char syllable)
{
if (phonemes == null) { throw new ArgumentNullException(); }
bool isSuccess = false;
// 초성, 중성(, 종성)으로 이루어진 문자 배열인지 확인
if (phonemes.Length >= 2 && phonemes.Length <= 3)
{
bool check = Onset.Contains(phonemes[0]) && Nucleus.Contains(phonemes[1]);
isSuccess = phonemes.Length == 3 ? check && Coda.Contains(phonemes[2]) : check;
}
syllable = (char)0x00;
// 한글 음절로의 합성이 불가능한 경우
if (!isSuccess)
{
return false;
}
// 한글 음절로의 합성이 가능한 경우
int onsetIndex = Array.IndexOf(Onset, phonemes[0]);
int nucleusIndex = Array.IndexOf(Nucleus, phonemes[1]);
int codaIndex = phonemes.Length == 3 ? Array.IndexOf(Coda, phonemes[2]) : 0;
int newCode = BeginCode + (onsetIndex * 588) + (nucleusIndex * 28) + codaIndex;
if (newCode < BeginCode || newCode > EndCode)
{
isSuccess = false;
}
else
{
syllable = Convert.ToChar(newCode);
}
return isSuccess;
}
/// <summary>
/// 인자의 음소 배열을 한글 음절로 합성합니다.
/// 합성이 불가능할 경우 예외를 발생시킵니다.
/// </summary>
/// <param name="phonemes">초성, 중성(, 종성) 순의 한글 음소 배열</param>
/// <exception cref="InvalidOperationException">인수의 배열이 합성 가능한 초성, 중성(, 종성)이 아닐 경우</exception>
/// <returns></returns>
public static char JoinToSyllable(char[] phonemes)
{
bool isSuccess = TryJoinToSyllable(phonemes, out char syllable);
if (!isSuccess)
{
throw new InvalidOperationException("인수의 배열은 합성 가능한 한글 초성, 중성(, 종성)이 아닙니다.");
}
return syllable;
}
/// <summary>
/// 검색문자를 대상문자에 대해 (초성) 비교 후 일치 여부를 반환합니다.
/// 검색문자에 초성이 주어질 경우 초성 일치, 그렇지 않은 경우 문자 완전 일치 여부를 반환합니다.
/// </summary>
/// <param name="searchChar">(초성) 비교할 문자</param>
/// <param name="targetChar">비교 대상 문자</param>
/// <returns></returns>
public static bool IsOnsetMatch(char searchChar, char targetChar)
{
// 1. 검색문자가 초성인 경우 대응하는 대상문자도 초성을 비교
// 2. 그렇지 않은 경우 대응하는 대상문자와 완전 일치 여부 비교
HangulChar shc = new HangulChar(searchChar);
HangulChar thc = new HangulChar(targetChar);
if (shc.IsOnset() && thc.TrySplitSyllable(out char[] phonemes))
{
targetChar = phonemes[0];
}
return searchChar == targetChar ? true : false;
}
/// <summary>
/// 검색문자를 현재 인스턴스의 문자에 대해 (초성) 비교 후 일치 여부를 반환합니다.
/// 검색문자에 초성이 주어질 경우 초성 일치, 그렇지 않은 경우 문자 완전 일치 여부를 반환합니다.
/// </summary>
/// <param name="searchChar">(초성) 비교할 문자</param>
/// <returns></returns>
public bool IsOnsetMatch(char searchChar)
{
return HangulChar.IsOnsetMatch(searchChar, this.CurrentCharacter);
}
/// <summary>
/// 초성으로 사용될 수 있는지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsOnset() => Onset.Contains(CurrentCharacter);
/// <summary>
/// 중성으로 사용될 수 있는지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsNucleus() => Nucleus.Contains(CurrentCharacter);
/// <summary>
/// 종성으로 사용될 수 있는지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsCoda() => Coda.Contains(CurrentCharacter);
/// <summary>
/// 자음인지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsConsonant() => Onset.Contains(CurrentCharacter) || Coda.Contains(CurrentCharacter);
/// <summary>
/// 모음인지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsVowel() => Nucleus.Contains(CurrentCharacter);
/// <summary>
/// 음소(낱소리)인지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsPhoneme() => IsConsonant() || IsVowel();
/// <summary>
/// 음소(낱소리)가 아닌 완전한 한글 음절인지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsSyllable() => BeginCode <= (int)CurrentCharacter && (int)CurrentCharacter <= EndCode;
/// <summary>
/// 한글 문자인지 판단합니다.
/// </summary>
/// <returns></returns>
public bool IsKoreanCharacter() => IsPhoneme() || IsSyllable();
/// <summary>
/// 한글 문자인지 판단합니다. (== IsKoreanCharacter())
/// </summary>
/// <returns></returns>
public bool IsHangul() => IsKoreanCharacter();
/// <summary>
/// 한글 음절을 초성, 중성, 종성 순으로 분리합니다.
/// 반환 결과는 분리의 성공(가능)여부를 나타냅니다.
/// </summary>
/// <param name="phonemes">초성, 중성, 종성 순으로 분리된 배열</param>
/// <returns></returns>
public bool TrySplitSyllable(out char[] phonemes)
{
// 분리 가능한 한글이 아닌 경우
if (!IsSyllable())
{
phonemes = new char[] { (char)0x00, (char)0x00, (char)0x00 };
return false;
}
int foo = (int)CurrentCharacter - BeginCode;
int onsetIndex = (int)foo / 588;
int nucleusIndex = (int)(foo - onsetIndex * 588) / 28;
int codaIndex = foo - onsetIndex * 588 - 28 * nucleusIndex;
phonemes = new char[] { Onset[onsetIndex], Nucleus[nucleusIndex], Coda[codaIndex] };
return true;
}
/// <summary>
/// 한글 음절을 초성, 중성, 종성 순으로 분리합니다.
/// 분리가 불가능한 경우 예외를 발생시킵니다.
/// </summary>
/// <exception cref="InvalidOperationException">인스턴스의 문자가 분리 가능한 한글이 아닌 경우</exception>
/// <returns></returns>
public char[] SplitSyllable()
{
bool isSuccess = TrySplitSyllable(out char[] phonemes);
if (!isSuccess)
{
throw new InvalidOperationException("인스턴스의 문자가 분리 가능한 한글이 아닙니다.");
}
return phonemes;
}
/// <summary>
/// 한글 획수를 반환합니다. 한글이 아닌 문자의 경우 0을 반환합니다.
/// </summary>
/// <returns></returns>
public int CountStrokes()
{
// 한글이 아닌 경우 0을 반환
if (!IsHangul())
{
return 0;
}
int count;
// 한글 - 음소인 경우
if (IsPhoneme())
{
if (IsOnset())
{
count = OnsetStrokes[Array.IndexOf(Onset, CurrentCharacter)];
}
else if (IsNucleus())
{
count = NucleusStrokes[Array.IndexOf(Nucleus, CurrentCharacter)];
}
else
{
count = CodaStrokes[Array.IndexOf(Coda, CurrentCharacter)];
}
}
// 한글 - 분리 가능한 음절인 경우
else
{
_ = TrySplitSyllable(out char[] phonemes);
count = OnsetStrokes[Array.IndexOf(Onset, phonemes[0])];
count += NucleusStrokes[Array.IndexOf(Nucleus, phonemes[1])];
count += CodaStrokes[Array.IndexOf(Coda, phonemes[2])];
}
return count;
}
}
}

View File

@@ -0,0 +1,390 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace NaverSearcher
{
/// <summary>
/// 한글 문자열 클래스: (한글을 포함한) 문자열에 대해 한글 처리에 관련된 다양한 속성과 메서드를 제공하는 클래스입니다.
/// </summary>
public class HangulString
{
/// <summary>
/// 문자열로부터 한글 문자열 클래스의 인스턴스를 생성합니다.
/// </summary>
/// <param name="aString">인스턴스를 생성할 문자열</param>
public HangulString(string aString) => CurrentString = aString;
/// <summary>
/// 현재 인스턴스의 문자열입니다.
/// </summary>
public string CurrentString { get; private set; }
/// <summary>
/// 문자열을 HangulChar 클래스 인스턴스의 배열로 변환하여 반환합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
/// <exception cref="ArgumentNullException">입력 인자가 null인 경우</exception>
public static HangulChar[] ToHangulCharArray(string aString)
{
if (aString is null)
{
throw new ArgumentNullException();
}
return aString.ToCharArray().Select(c => new HangulChar(c)).ToArray();
}
/// <summary>
/// 문자열을 한글 문자열 부분과 나머지 문자열 부분으로 구분하여 반환합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
/// <exception cref="ArgumentNullException">입력 인자가 null인 경우</exception>
public static string[] SeparateString(string aString)
{
if (aString is null)
{
throw new ArgumentNullException();
}
string[] result = new string[2];
StringBuilder sbKorean = new StringBuilder();
StringBuilder sbOthers = new StringBuilder();
foreach (char c in aString)
{
HangulChar hc = new HangulChar(c);
if (hc.IsKoreanCharacter())
{
sbKorean.Append(c);
}
else
{
sbOthers.Append(c);
}
}
result[0] = sbKorean.ToString();
result[1] = sbOthers.ToString();
return result;
}
/// <summary>
/// 문자열이 모두 한글로만 이루어져 있는지의 여부를 반환합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
public static bool IsAllHangul(string aString)
{
string checkPoint = HangulString.SeparateString(aString)[1];
if (checkPoint.Length > 0)
{
return false;
}
else
{
return true;
}
}
/// <summary>
/// 문자열에 대해 한글 음절을 초성, 중성, 종성으로 분리하여 반환합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
/// <exception cref="ArgumentNullException"><paramref name="aString"/>이 null인 경우</exception>
public static string SplitToPhonemes(string aString)
{
if (aString == null)
{
throw new ArgumentNullException();
}
StringBuilder sb = new StringBuilder();
foreach (char c in aString)
{
HangulChar hc = new HangulChar(c);
// 한글 음절로 판명되어 분리가 가능한 경우
if (hc.TrySplitSyllable(out char[] phonemes))
{
foreach (char pc in phonemes)
{
// 초성-중성으로만 이루어져 있는 경우 종성은 반환 문자열에 포함하지 않음
if (pc != (char)0x00)
{
sb.Append(pc);
}
}
}
// 한글 음절이 아닌 경우
else
{
sb.Append(c);
}
}
return sb.ToString();
}
/// <summary>
/// 문자열에 대해 한글 음절을 초성, 중성, 종성으로 분리하여 반환합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
/// <exception cref="ArgumentNullException"><paramref name="aString"/>이 null인 경우</exception>
public static string SplitToPhonemes2(string aString)
{
if (aString == null)
{
throw new ArgumentNullException();
}
StringBuilder sb = new StringBuilder();
foreach (char c in aString)
{
HangulChar hc = new HangulChar(c);
// 한글 음절로 판명되어 분리가 가능한 경우
if (hc.TrySplitSyllable(out char[] phonemes))
{
foreach (char pc in phonemes)
{
// 초성-중성으로만 이루어져 있는 경우 종성은 반환 문자열에 포함하지 않음
if (pc != (char)0x00)
{
sb.Append(pc);
}
}
}
// 한글 음절이 아닌 경우
else
{
sb.Append(c);
}
}
// 여기에서 한번더 수행한다.
StringBuilder sb2 = new StringBuilder();
foreach (char c in sb.ToString())
{
HangulChar hc = new HangulChar(c);
// 한글 음절로 판명되어 분리가 가능한 경우
if (hc.TrySplitSyllable(out char[] phonemes))
{
foreach (char pc in phonemes)
{
// 초성-중성으로만 이루어져 있는 경우 종성은 반환 문자열에 포함하지 않음
if (pc != (char)0x00)
{
sb.Append(pc);
}
}
}
// 한글 음절이 아닌 경우
else
{
sb.Append(c);
}
}
return sb2.ToString();
}
/// <summary>
/// 문자열 내의 초성, 중성, 종성 음소를 합성하여 반환합니다.
/// 의도하지 않은 반환 결과를 피하기 위해 일반적으로 사용하는 방식의
/// (완전한 한글 음절이 분리된) 자음/모음으로 구성된 문자열을 인수로 사용할 것을 권장합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
/// <exception cref="ArgumentNullException"><paramref name="aString"/>이 null인 경우</exception>
public static string JoinPhonemes(string aString)
{
if (aString == null) { throw new ArgumentNullException(); }
StringBuilder sb = new StringBuilder();
int curIdx = 0;
int lastIdx = aString.Length - 1;
// 마지막 인덱스 이후의 값 판단에 대한 예외를 피하기 위해 대상 문자열에 10개의 공백 임시 추가
string newString = aString + new string(' ', 10);
while (curIdx <= lastIdx)
{
// 현재 인덱스의 문자
HangulChar hc = new HangulChar(newString[curIdx]);
// 현재 인덱스의 문자가 음소가 아닌 경우 현재 문자를 더하고 다음 인덱스로 이동
if (!hc.IsPhoneme())
{
sb.Append(hc.CurrentCharacter);
curIdx++;
continue;
}
// 현재 인덱스의 문자가 음소인 경우에 대해
// (1) 현재 인덱스로부터 2글자 합성이 가능하고
// (2)-A 이후 첫 글자가 한글이 아니거나
// (2)-B 이후 두 글자(초성-중성)가 합성이 가능한 경우
// 현재 인덱스로부터 2글자를 합성하고, 인덱스를 2칸 전진
bool isCur2SpanOk = HangulChar.TryJoinToSyllable(
new char[] { newString[curIdx], newString[curIdx + 1] },
out char cur2SpanSyllable);
bool isCur2Next1Ok = !(new HangulChar(newString[curIdx + 2])).IsHangul();
bool isCur2Next2Ok = HangulChar.TryJoinToSyllable(
new char[] { newString[curIdx + 2], newString[curIdx + 3] },
out char _);
// (3) 현재 인덱스로부터 3글자 합성이 가능하고
// (4)-A 이후 첫 글자가 한글이 아니거나
// (4)-B 이후 두 글자(초성-중성)가 합성이 가능한 경우
// 현재 인덱스로부터 3글자를 합성하고, 인덱스를 3칸 전진
bool isCur3SpanOk = HangulChar.TryJoinToSyllable(
new char[] { newString[curIdx], newString[curIdx + 1], newString[curIdx + 2] },
out char cur3SpanSyllable);
bool isCur3Next1Ok = !(new HangulChar(newString[curIdx + 3])).IsHangul();
bool isCur3Next2Ok = HangulChar.TryJoinToSyllable(
new char[] { newString[curIdx + 3], newString[curIdx + 4] },
out char _);
// 상기 논리에 의한 판단부
// 2글자 기준 판단
if (isCur2SpanOk && (isCur2Next1Ok || isCur2Next2Ok))
{
sb.Append(cur2SpanSyllable);
curIdx += 2;
}
// 3글자 기준 판단
else if (isCur3SpanOk && (isCur3Next1Ok || isCur3Next2Ok))
{
sb.Append(cur3SpanSyllable);
curIdx += 3;
}
else
{
sb.Append(hc.CurrentCharacter);
curIdx++;
}
}
return sb.ToString();
}
/// <summary>
/// 문자열에 대해 초성 검색을 실시합니다. 반환값은 초성 검색에 대한 결과의 존재여부입니다.
/// 검색 문자열에 초성이 주어질 경우 초성 일치, 그렇지 않은 경우 문자 완전 일치 여부를 반환합니다.
/// </summary>
/// <param name="searchString">검색할 문자열</param>
/// <param name="targetString">대상 문자열</param>
/// <param name="indices">(초성)일치가 발견된 대상 문자열 내 인덱스 배열</param>
/// <returns></returns>
/// <exception cref="ArgumentNullException"><paramref name="searchString"/> 또는 <paramref name="targetString"/>이 null인 경우</exception>
public static bool GetOnsetMatches(string searchString, string targetString, out int[] indices)
{
if (searchString == null || targetString == null)
{
throw new ArgumentNullException();
}
List<int> idxs = new List<int>();
// 검색 문자열의 길이가 대상 문자열의 길이보다 긴 경우
if (searchString.Length > targetString.Length)
{
indices = idxs.ToArray();
return false;
}
// 대상 문자열 내의 인덱스를 증가시켜 가며
for (int curIdx = 0; curIdx < targetString.Length - searchString.Length + 1; curIdx++)
{
// 해당 인덱스로부터 검색 문자열의 길이만큼 초성일치 검색 수행
// 초성 일치가 될 경우 해당 인덱스 저장
bool isMatch = true;
for (int chkIdx = curIdx; chkIdx < curIdx + searchString.Length; chkIdx++)
{
if (!HangulChar.IsOnsetMatch(searchString[chkIdx - curIdx], targetString[chkIdx]))
{
isMatch = false;
break;
}
}
if (isMatch)
{
idxs.Add(curIdx);
}
}
indices = idxs.ToArray();
if (indices.Length > 0)
{
return true;
}
else
{
return false;
}
}
/// <summary>
/// 현재 인스턴스의 문자열을 HangulChar 클래스 인스턴스의 배열로 변환하여 반환합니다.
/// </summary>
/// <returns></returns>
public HangulChar[] ToHangulCharArray() => HangulString.ToHangulCharArray(CurrentString);
/// <summary>
/// 현재 인스턴스의 문자열을 한글 문자열 부분과 나머지 문자열 부분으로 구분하여 반환합니다.
/// </summary>
/// <returns></returns>
public string[] SeparateString() => HangulString.SeparateString(CurrentString);
/// <summary>
/// 현재 인스턴스의 문자열이 모두 한글로만 이루어져 있는지의 여부를 반환합니다.
/// </summary>
/// <returns></returns>
public bool IsAllHangul() => HangulString.IsAllHangul(CurrentString);
/// <summary>
/// 현재 인스턴스의 문자열에 대해 한글 음절을 초성, 중성, 종성으로 분리하여 반환합니다.
/// </summary>
/// <returns></returns>
public string SplitToPhonemes() => HangulString.SplitToPhonemes(CurrentString);
/// <summary>
/// 인스턴스 문자열 내의 초성, 중성, 종성 음소를 합성하여 반환합니다.
/// 의도하지 않은 반환 결과를 피하기 위해 일반적으로 사용하는 방식의
/// (완전한 한글 음절이 분리된) 자음/모음으로 구성된 문자열을 인수로 사용할 것을 권장합니다.
/// </summary>
/// <returns></returns>
public string JoinPhonemes() => HangulString.JoinPhonemes(CurrentString);
/// <summary>
/// 인스턴스 문자열의 길이(글자수)를 반환합니다.
/// </summary>
/// <remarks>
/// 윈도우 시스템에서는 개행문자가 2글자(2바이트)로 취급됨에 유의해야 합니다.
/// </remarks>
/// <param name="ignoreWhiteSpcaes">공백문자(whitespace) 무시 여부</param>
/// <returns></returns>
public int GetStringLength(bool ignoreWhiteSpcaes = false)
{
string inputString = ignoreWhiteSpcaes ? RemoveWhiteSpaces(CurrentString) : CurrentString;
return inputString.Length;
}
/// <summary>
/// 현재 인코딩에 대해 인스턴스 문자열의 길이(바이트)를 반환합니다.
/// </summary>
/// <remarks>
/// 윈도우 시스템에서는 개행문자가 2글자(2바이트)로 취급됨에 유의해야 합니다.
/// </remarks>
/// <param name="ignoreWhiteSpcaes">공백문자(whitespace) 무시 여부</param>
/// <returns></returns>
public int GetStringByteLength(bool ignoreWhiteSpcaes = false)
{
string inputString = ignoreWhiteSpcaes ? RemoveWhiteSpaces(CurrentString) : CurrentString;
return Encoding.Default.GetByteCount(inputString);
}
/// <summary>
/// 인수의 문자열에서 공백문자(whitespace)를 제거한 문자열을 반환합니다.
/// </summary>
/// <param name="aString"></param>
/// <returns></returns>
private static string RemoveWhiteSpaces(string aString) =>
new string(aString.Where(c => !char.IsWhiteSpace(c)).Select(c => c).ToArray());
}
}

View File

@@ -0,0 +1,15 @@
<Project Sdk="Microsoft.NET.Sdk.WindowsDesktop">
<PropertyGroup>
<OutputType>WinExe</OutputType>
<TargetFramework>netcoreapp3.1</TargetFramework>
<UseWindowsForms>true</UseWindowsForms>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="DotNetSeleniumExtras.WaitHelpers" Version="3.11.0" />
<PackageReference Include="Selenium.WebDriver" Version="3.141.0" />
<PackageReference Include="Selenium.WebDriver.ChromeDriver" Version="92.0.4515.10700" />
</ItemGroup>
</Project>

23
NaverSearcher/Program.cs Normal file
View File

@@ -0,0 +1,23 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace NaverSearcher
{
static class Program
{
/// <summary>
/// The main entry point for the application.
/// </summary>
[STAThread]
static void Main()
{
Application.SetHighDpiMode(HighDpiMode.SystemAware);
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
}
}

View File

@@ -0,0 +1,10 @@
using System;
using System.Collections.Generic;
using System.Text;
namespace NaverSearcher
{
class StringParser
{
}
}