<!--{{{-->
<link rel='alternate' type='application/rss+xml' title='RSS' href='index.xml' />
<!--}}}-->
Background: #fff
Foreground: #000
PrimaryPale: #8cf
PrimaryLight: #18f
PrimaryMid: #04b
PrimaryDark: #014
SecondaryPale: #ffc
SecondaryLight: #fe8
SecondaryMid: #db4
SecondaryDark: #841
TertiaryPale: #eee
TertiaryLight: #ccc
TertiaryMid: #999
TertiaryDark: #666
Error: #f88
/*{{{*/
body {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}

a {color:[[ColorPalette::PrimaryMid]];}
a:hover {background-color:[[ColorPalette::PrimaryMid]]; color:[[ColorPalette::Background]];}
a img {border:0;}

h1,h2,h3,h4,h5,h6 {color:[[ColorPalette::SecondaryDark]]; background:transparent;}
h1 {border-bottom:2px solid [[ColorPalette::TertiaryLight]];}
h2,h3 {border-bottom:1px solid [[ColorPalette::TertiaryLight]];}

.button {color:[[ColorPalette::PrimaryDark]]; border:1px solid [[ColorPalette::Background]];}
.button:hover {color:[[ColorPalette::PrimaryDark]]; background:[[ColorPalette::SecondaryLight]]; border-color:[[ColorPalette::SecondaryMid]];}
.button:active {color:[[ColorPalette::Background]]; background:[[ColorPalette::SecondaryMid]]; border:1px solid [[ColorPalette::SecondaryDark]];}

.header {background:[[ColorPalette::PrimaryMid]];}
.headerShadow {color:[[ColorPalette::Foreground]];}
.headerShadow a {font-weight:normal; color:[[ColorPalette::Foreground]];}
.headerForeground {color:[[ColorPalette::Background]];}
.headerForeground a {font-weight:normal; color:[[ColorPalette::PrimaryPale]];}

.tabSelected{color:[[ColorPalette::PrimaryDark]];
	background:[[ColorPalette::TertiaryPale]];
	border-left:1px solid [[ColorPalette::TertiaryLight]];
	border-top:1px solid [[ColorPalette::TertiaryLight]];
	border-right:1px solid [[ColorPalette::TertiaryLight]];
}
.tabUnselected {color:[[ColorPalette::Background]]; background:[[ColorPalette::TertiaryMid]];}
.tabContents {color:[[ColorPalette::PrimaryDark]]; background:[[ColorPalette::TertiaryPale]]; border:1px solid [[ColorPalette::TertiaryLight]];}
.tabContents .button {border:0;}

#sidebar {}
#sidebarOptions input {border:1px solid [[ColorPalette::PrimaryMid]];}
#sidebarOptions .sliderPanel {background:[[ColorPalette::PrimaryPale]];}
#sidebarOptions .sliderPanel a {border:none;color:[[ColorPalette::PrimaryMid]];}
#sidebarOptions .sliderPanel a:hover {color:[[ColorPalette::Background]]; background:[[ColorPalette::PrimaryMid]];}
#sidebarOptions .sliderPanel a:active {color:[[ColorPalette::PrimaryMid]]; background:[[ColorPalette::Background]];}

.wizard {background:[[ColorPalette::PrimaryPale]]; border:1px solid [[ColorPalette::PrimaryMid]];}
.wizard h1 {color:[[ColorPalette::PrimaryDark]]; border:none;}
.wizard h2 {color:[[ColorPalette::Foreground]]; border:none;}
.wizardStep {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];
	border:1px solid [[ColorPalette::PrimaryMid]];}
.wizardStep.wizardStepDone {background:[[ColorPalette::TertiaryLight]];}
.wizardFooter {background:[[ColorPalette::PrimaryPale]];}
.wizardFooter .status {background:[[ColorPalette::PrimaryDark]]; color:[[ColorPalette::Background]];}
.wizard .button {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::SecondaryLight]]; border: 1px solid;
	border-color:[[ColorPalette::SecondaryPale]] [[ColorPalette::SecondaryDark]] [[ColorPalette::SecondaryDark]] [[ColorPalette::SecondaryPale]];}
.wizard .button:hover {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::Background]];}
.wizard .button:active {color:[[ColorPalette::Background]]; background:[[ColorPalette::Foreground]]; border: 1px solid;
	border-color:[[ColorPalette::PrimaryDark]] [[ColorPalette::PrimaryPale]] [[ColorPalette::PrimaryPale]] [[ColorPalette::PrimaryDark]];}

.wizard .notChanged {background:transparent;}
.wizard .changedLocally {background:#80ff80;}
.wizard .changedServer {background:#8080ff;}
.wizard .changedBoth {background:#ff8080;}
.wizard .notFound {background:#ffff80;}
.wizard .putToServer {background:#ff80ff;}
.wizard .gotFromServer {background:#80ffff;}

#messageArea {border:1px solid [[ColorPalette::SecondaryMid]]; background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]];}
#messageArea .button {color:[[ColorPalette::PrimaryMid]]; background:[[ColorPalette::SecondaryPale]]; border:none;}

.popupTiddler {background:[[ColorPalette::TertiaryPale]]; border:2px solid [[ColorPalette::TertiaryMid]];}

.popup {background:[[ColorPalette::TertiaryPale]]; color:[[ColorPalette::TertiaryDark]]; border-left:1px solid [[ColorPalette::TertiaryMid]]; border-top:1px solid [[ColorPalette::TertiaryMid]]; border-right:2px solid [[ColorPalette::TertiaryDark]]; border-bottom:2px solid [[ColorPalette::TertiaryDark]];}
.popup hr {color:[[ColorPalette::PrimaryDark]]; background:[[ColorPalette::PrimaryDark]]; border-bottom:1px;}
.popup li.disabled {color:[[ColorPalette::TertiaryMid]];}
.popup li a, .popup li a:visited {color:[[ColorPalette::Foreground]]; border: none;}
.popup li a:hover {background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]]; border: none;}
.popup li a:active {background:[[ColorPalette::SecondaryPale]]; color:[[ColorPalette::Foreground]]; border: none;}
.popupHighlight {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
.listBreak div {border-bottom:1px solid [[ColorPalette::TertiaryDark]];}

.tiddler .defaultCommand {font-weight:bold;}

.shadow .title {color:[[ColorPalette::TertiaryDark]];}

.title {color:[[ColorPalette::SecondaryDark]];}
.subtitle {color:[[ColorPalette::TertiaryDark]];}

.toolbar {color:[[ColorPalette::PrimaryMid]];}
.toolbar a {color:[[ColorPalette::TertiaryLight]];}
.selected .toolbar a {color:[[ColorPalette::TertiaryMid]];}
.selected .toolbar a:hover {color:[[ColorPalette::Foreground]];}

.tagging, .tagged {border:1px solid [[ColorPalette::TertiaryPale]]; background-color:[[ColorPalette::TertiaryPale]];}
.selected .tagging, .selected .tagged {background-color:[[ColorPalette::TertiaryLight]]; border:1px solid [[ColorPalette::TertiaryMid]];}
.tagging .listTitle, .tagged .listTitle {color:[[ColorPalette::PrimaryDark]];}
.tagging .button, .tagged .button {border:none;}

.footer {color:[[ColorPalette::TertiaryLight]];}
.selected .footer {color:[[ColorPalette::TertiaryMid]];}

.sparkline {background:[[ColorPalette::PrimaryPale]]; border:0;}
.sparktick {background:[[ColorPalette::PrimaryDark]];}

.error, .errorButton {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::Error]];}
.warning {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::SecondaryPale]];}
.lowlight {background:[[ColorPalette::TertiaryLight]];}

.zoomer {background:none; color:[[ColorPalette::TertiaryMid]]; border:3px solid [[ColorPalette::TertiaryMid]];}

.imageLink, #displayArea .imageLink {background:transparent;}

.annotation {background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]]; border:2px solid [[ColorPalette::SecondaryMid]];}

.viewer .listTitle {list-style-type:none; margin-left:-2em;}
.viewer .button {border:1px solid [[ColorPalette::SecondaryMid]];}
.viewer blockquote {border-left:3px solid [[ColorPalette::TertiaryDark]];}

.viewer table, table.twtable {border:2px solid [[ColorPalette::TertiaryDark]];}
.viewer th, .viewer thead td, .twtable th, .twtable thead td {background:[[ColorPalette::SecondaryMid]]; border:1px solid [[ColorPalette::TertiaryDark]]; color:[[ColorPalette::Background]];}
.viewer td, .viewer tr, .twtable td, .twtable tr {border:1px solid [[ColorPalette::TertiaryDark]];}

.viewer pre {border:1px solid [[ColorPalette::SecondaryLight]]; background:[[ColorPalette::SecondaryPale]];}
.viewer code {color:[[ColorPalette::SecondaryDark]];}
.viewer hr {border:0; border-top:dashed 1px [[ColorPalette::TertiaryDark]]; color:[[ColorPalette::TertiaryDark]];}

.highlight, .marked {background:[[ColorPalette::SecondaryLight]];}

.editor input {border:1px solid [[ColorPalette::PrimaryMid]];}
.editor textarea {border:1px solid [[ColorPalette::PrimaryMid]]; width:100%;}
.editorFooter {color:[[ColorPalette::TertiaryMid]];}

#backstageArea {background:[[ColorPalette::Foreground]]; color:[[ColorPalette::TertiaryMid]];}
#backstageArea a {background:[[ColorPalette::Foreground]]; color:[[ColorPalette::Background]]; border:none;}
#backstageArea a:hover {background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]]; }
#backstageArea a.backstageSelTab {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
#backstageButton a {background:none; color:[[ColorPalette::Background]]; border:none;}
#backstageButton a:hover {background:[[ColorPalette::Foreground]]; color:[[ColorPalette::Background]]; border:none;}
#backstagePanel {background:[[ColorPalette::Background]]; border-color: [[ColorPalette::Background]] [[ColorPalette::TertiaryDark]] [[ColorPalette::TertiaryDark]] [[ColorPalette::TertiaryDark]];}
.backstagePanelFooter .button {border:none; color:[[ColorPalette::Background]];}
.backstagePanelFooter .button:hover {color:[[ColorPalette::Foreground]];}
#backstageCloak {background:[[ColorPalette::Foreground]]; opacity:0.6; filter:'alpha(opacity=60)';}
/*}}}*/
/*{{{*/
* html .tiddler {height:1%;}

body {font-size:.75em; font-family:arial,helvetica; margin:0; padding:0;}

h1,h2,h3,h4,h5,h6 {font-weight:bold; text-decoration:none;}
h1,h2,h3 {padding-bottom:1px; margin-top:1.2em;margin-bottom:0.3em;}
h4,h5,h6 {margin-top:1em;}
h1 {font-size:1.35em;}
h2 {font-size:1.25em;}
h3 {font-size:1.1em;}
h4 {font-size:1em;}
h5 {font-size:.9em;}

hr {height:1px;}

a {text-decoration:none;}

dt {font-weight:bold;}

ol {list-style-type:decimal;}
ol ol {list-style-type:lower-alpha;}
ol ol ol {list-style-type:lower-roman;}
ol ol ol ol {list-style-type:decimal;}
ol ol ol ol ol {list-style-type:lower-alpha;}
ol ol ol ol ol ol {list-style-type:lower-roman;}
ol ol ol ol ol ol ol {list-style-type:decimal;}

.txtOptionInput {width:11em;}

#contentWrapper .chkOptionInput {border:0;}

.externalLink {text-decoration:underline;}

.indent {margin-left:3em;}
.outdent {margin-left:3em; text-indent:-3em;}
code.escaped {white-space:nowrap;}

.tiddlyLinkExisting {font-weight:bold;}
.tiddlyLinkNonExisting {font-style:italic;}

/* the 'a' is required for IE, otherwise it renders the whole tiddler in bold */
a.tiddlyLinkNonExisting.shadow {font-weight:bold;}

#mainMenu .tiddlyLinkExisting,
	#mainMenu .tiddlyLinkNonExisting,
	#sidebarTabs .tiddlyLinkNonExisting {font-weight:normal; font-style:normal;}
#sidebarTabs .tiddlyLinkExisting {font-weight:bold; font-style:normal;}

.header {position:relative;}
.header a:hover {background:transparent;}
.headerShadow {position:relative; padding:4.5em 0 1em 1em; left:-1px; top:-1px;}
.headerForeground {position:absolute; padding:4.5em 0 1em 1em; left:0px; top:0px;}

.siteTitle {font-size:3em;}
.siteSubtitle {font-size:1.2em;}

#mainMenu {position:absolute; left:0; width:10em; text-align:right; line-height:1.6em; padding:1.5em 0.5em 0.5em 0.5em; font-size:1.1em;}

#sidebar {position:absolute; right:3px; width:16em; font-size:.9em;}
#sidebarOptions {padding-top:0.3em;}
#sidebarOptions a {margin:0 0.2em; padding:0.2em 0.3em; display:block;}
#sidebarOptions input {margin:0.4em 0.5em;}
#sidebarOptions .sliderPanel {margin-left:1em; padding:0.5em; font-size:.85em;}
#sidebarOptions .sliderPanel a {font-weight:bold; display:inline; padding:0;}
#sidebarOptions .sliderPanel input {margin:0 0 0.3em 0;}
#sidebarTabs .tabContents {width:15em; overflow:hidden;}

.wizard {padding:0.1em 1em 0 2em;}
.wizard h1 {font-size:2em; font-weight:bold; background:none; padding:0; margin:0.4em 0 0.2em;}
.wizard h2 {font-size:1.2em; font-weight:bold; background:none; padding:0; margin:0.4em 0 0.2em;}
.wizardStep {padding:1em 1em 1em 1em;}
.wizard .button {margin:0.5em 0 0; font-size:1.2em;}
.wizardFooter {padding:0.8em 0.4em 0.8em 0;}
.wizardFooter .status {padding:0 0.4em; margin-left:1em;}
.wizard .button {padding:0.1em 0.2em;}

#messageArea {position:fixed; top:2em; right:0; margin:0.5em; padding:0.5em; z-index:2000; _position:absolute;}
.messageToolbar {display:block; text-align:right; padding:0.2em;}
#messageArea a {text-decoration:underline;}

.tiddlerPopupButton {padding:0.2em;}
.popupTiddler {position: absolute; z-index:300; padding:1em; margin:0;}

.popup {position:absolute; z-index:300; font-size:.9em; padding:0; list-style:none; margin:0;}
.popup .popupMessage {padding:0.4em;}
.popup hr {display:block; height:1px; width:auto; padding:0; margin:0.2em 0;}
.popup li.disabled {padding:0.4em;}
.popup li a {display:block; padding:0.4em; font-weight:normal; cursor:pointer;}
.listBreak {font-size:1px; line-height:1px;}
.listBreak div {margin:2px 0;}

.tabset {padding:1em 0 0 0.5em;}
.tab {margin:0 0 0 0.25em; padding:2px;}
.tabContents {padding:0.5em;}
.tabContents ul, .tabContents ol {margin:0; padding:0;}
.txtMainTab .tabContents li {list-style:none;}
.tabContents li.listLink { margin-left:.75em;}

#contentWrapper {display:block;}
#splashScreen {display:none;}

#displayArea {margin:1em 17em 0 14em;}

.toolbar {text-align:right; font-size:.9em;}

.tiddler {padding:1em 1em 0;}

.missing .viewer,.missing .title {font-style:italic;}

.title {font-size:1.6em; font-weight:bold;}

.missing .subtitle {display:none;}
.subtitle {font-size:1.1em;}

.tiddler .button {padding:0.2em 0.4em;}

.tagging {margin:0.5em 0.5em 0.5em 0; float:left; display:none;}
.isTag .tagging {display:block;}
.tagged {margin:0.5em; float:right;}
.tagging, .tagged {font-size:0.9em; padding:0.25em;}
.tagging ul, .tagged ul {list-style:none; margin:0.25em; padding:0;}
.tagClear {clear:both;}

.footer {font-size:.9em;}
.footer li {display:inline;}

.annotation {padding:0.5em; margin:0.5em;}

* html .viewer pre {width:99%; padding:0 0 1em 0;}
.viewer {line-height:1.4em; padding-top:0.5em;}
.viewer .button {margin:0 0.25em; padding:0 0.25em;}
.viewer blockquote {line-height:1.5em; padding-left:0.8em;margin-left:2.5em;}
.viewer ul, .viewer ol {margin-left:0.5em; padding-left:1.5em;}

.viewer table, table.twtable {border-collapse:collapse; margin:0.8em 1.0em;}
.viewer th, .viewer td, .viewer tr,.viewer caption,.twtable th, .twtable td, .twtable tr,.twtable caption {padding:3px;}
table.listView {font-size:0.85em; margin:0.8em 1.0em;}
table.listView th, table.listView td, table.listView tr {padding:0px 3px 0px 3px;}

.viewer pre {padding:0.5em; margin-left:0.5em; font-size:1.2em; line-height:1.4em; overflow:auto;}
.viewer code {font-size:1.2em; line-height:1.4em;}

.editor {font-size:1.1em;}
.editor input, .editor textarea {display:block; width:100%; font:inherit;}
.editorFooter {padding:0.25em 0; font-size:.9em;}
.editorFooter .button {padding-top:0px; padding-bottom:0px;}

.fieldsetFix {border:0; padding:0; margin:1px 0px;}

.sparkline {line-height:1em;}
.sparktick {outline:0;}

.zoomer {font-size:1.1em; position:absolute; overflow:hidden;}
.zoomer div {padding:1em;}

* html #backstage {width:99%;}
* html #backstageArea {width:99%;}
#backstageArea {display:none; position:relative; overflow: hidden; z-index:150; padding:0.3em 0.5em;}
#backstageToolbar {position:relative;}
#backstageArea a {font-weight:bold; margin-left:0.5em; padding:0.3em 0.5em;}
#backstageButton {display:none; position:absolute; z-index:175; top:0; right:0;}
#backstageButton a {padding:0.1em 0.4em; margin:0.1em;}
#backstage {position:relative; width:100%; z-index:50;}
#backstagePanel {display:none; z-index:100; position:absolute; width:90%; margin-left:3em; padding:1em;}
.backstagePanelFooter {padding-top:0.2em; float:right;}
.backstagePanelFooter a {padding:0.2em 0.4em;}
#backstageCloak {display:none; z-index:20; position:absolute; width:100%; height:100px;}

.whenBackstage {display:none;}
.backstageVisible .whenBackstage {display:block;}
/*}}}*/
/***
StyleSheet for use when a translation requires any css style changes.
This StyleSheet can be used directly by languages such as Chinese, Japanese and Korean which need larger font sizes.
***/
/*{{{*/
body {font-size:0.8em;}
#sidebarOptions {font-size:1.05em;}
#sidebarOptions a {font-style:normal;}
#sidebarOptions .sliderPanel {font-size:0.95em;}
.subtitle {font-size:0.8em;}
.viewer table.listView {font-size:0.95em;}
/*}}}*/
/*{{{*/
@media print {
#mainMenu, #sidebar, #messageArea, .toolbar, #backstageButton, #backstageArea {display: none !important;}
#displayArea {margin: 1em 1em 0em;}
noscript {display:none;} /* Fixes a feature in Firefox 1.5.0.2 where print preview displays the noscript content */
}
/*}}}*/
<!--{{{-->
<div class='header' macro='gradient vert [[ColorPalette::PrimaryLight]] [[ColorPalette::PrimaryMid]]'>
<div class='headerShadow'>
<span class='siteTitle' refresh='content' tiddler='SiteTitle'></span>&nbsp;
<span class='siteSubtitle' refresh='content' tiddler='SiteSubtitle'></span>
</div>
<div class='headerForeground'>
<span class='siteTitle' refresh='content' tiddler='SiteTitle'></span>&nbsp;
<span class='siteSubtitle' refresh='content' tiddler='SiteSubtitle'></span>
</div>
</div>
<div id='mainMenu' refresh='content' tiddler='MainMenu'></div>
<div id='sidebar'>
<div id='sidebarOptions' refresh='content' tiddler='SideBarOptions'></div>
<div id='sidebarTabs' refresh='content' force='true' tiddler='SideBarTabs'></div>
</div>
<div id='displayArea'>
<div id='messageArea'></div>
<div id='tiddlerDisplay'></div>
</div>
<!--}}}-->
<!--{{{-->
<div class='toolbar' macro='toolbar [[ToolbarCommands::ViewToolbar]]'></div>
<div class='title' macro='view title'></div>
<div class='subtitle'><span macro='view modifier link'></span>, <span macro='view modified date'></span> (<span macro='message views.wikified.createdPrompt'></span> <span macro='view created date'></span>)</div>
<div class='tagging' macro='tagging'></div>
<div class='tagged' macro='tags'></div>
<div class='viewer' macro='view text wikified'></div>
<div class='tagClear'></div>
<!--}}}-->
<!--{{{-->
<div class='toolbar' macro='toolbar [[ToolbarCommands::EditToolbar]]'></div>
<div class='title' macro='view title'></div>
<div class='editor' macro='edit title'></div>
<div macro='annotations'></div>
<div class='editor' macro='edit text'></div>
<div class='editor' macro='edit tags'></div><div class='editorFooter'><span macro='message views.editor.tagPrompt'></span><span macro='tagChooser excludeLists'></span></div>
<!--}}}-->
To get started with this blank [[TiddlyWiki]], you'll need to modify the following tiddlers:
* [[SiteTitle]] & [[SiteSubtitle]]: The title and subtitle of the site, as shown above (after saving, they will also appear in the browser title bar)
* [[MainMenu]]: The menu (usually on the left)
* [[DefaultTiddlers]]: Contains the names of the tiddlers that you want to appear when the TiddlyWiki is opened
You'll also need to enter your username for signing your edits: <<option txtUserName>>
These [[InterfaceOptions]] for customising [[TiddlyWiki]] are saved in your browser

Your username for signing your edits. Write it as a [[WikiWord]] (eg [[JoeBloggs]])

<<option txtUserName>>
<<option chkSaveBackups>> [[SaveBackups]]
<<option chkAutoSave>> [[AutoSave]]
<<option chkRegExpSearch>> [[RegExpSearch]]
<<option chkCaseSensitiveSearch>> [[CaseSensitiveSearch]]
<<option chkAnimate>> [[EnableAnimations]]

----
Also see [[AdvancedOptions]]
<<importTiddlers>>
|Default|@@""" AlertCusAdj = 0 """@@|
คำสั่งเกี่ยวกับการฟ้อง error เมื่อโปรแกรมทำการลดหนี้ลูกหนี้ แล้วยอดหนี้คงเหลือติดลบทั้งจากการป้อนข้อมูลรายการลดหนี้ และการผ่านรายการลดหนี้จากการชำระบิลขาย
* ค่า default 0 คือ ให้โปรแกรมฟ้องเตือน รอให้ผู้ใช้ตัดสินใจว่าจะยอมให้ทำงานต่อ หรือหยุดทำงาน
* ถ้าใช้ค่า 1 โปรแกรมจะเก็บสถานะ error โดยไม่แจ้งให้ผู้ใช้ตัดสินใจ
* ถ้าใช้ค่าเป็น 2, 4, 6 โปรแกรมจะฟ้องเตือน และรอผู้ให้ผู้ใช้ตัดสินใจภายในเวลาที่กำหนด (2,4 หรือ 6 วินาที) ถ้าผู้ใช้ไม่กดภายในเวลา โปรแกรมจะเก็บสถานะเป็น error 
|Default|@@""" AlertStkAdj = 0 """@@|
คำสั่งเกี่ยวกับการฟ้อง error เมื่อโปรแกรมทำการตัดสต็อก แล้วยอดคงเหลือติดลบ
ทั้งจากการป้อนข้อมูลรายการสต็อกสินค้าออก และการผ่านรายการตัดสต็อกจากบิลขาย

* ค่า default 0 คือ ให้โปรแกรมฟ้องเตือน รอให้ผู้ใช้ตัดสินใจว่าจะยอมให้ทำงานต่อ หรือหยุดทำงาน
* ถ้าใช้ค่า 1 โปรแกรมจะเก็บสถานะ error โดยไม่แจ้งให้ผู้ใช้ตัดสินใจ
* ถ้าใช้ค่าเป็น 2, 4, 6 โปรแกรมจะฟ้องเตือน และรอผู้ให้ผู้ใช้ตัดสินใจภายในเวลาที่กำหนด (2,4 หรือ 6 วินาที) ถ้าผู้ใช้ไม่กดภายในเวลา โปรแกรมจะเก็บสถานะเป็น error 
|Default|@@"""  AlertVndAdj = 0 """@@|
คำสั่งเกี่ยวกับการฟ้อง error เมื่อโปรแกรมทำการลดหนี้เจ้าหนี้ แล้วยอดหนี้คงเหลือติดลบทั้งจากการป้อนข้อมูลรายการลดหนี้ และการผ่านรายการลดหนี้จากการชำระบิลซื้อ
* ค่า default 0 คือ ให้โปรแกรมฟ้องเตือน รอให้ผู้ใช้ตัดสินใจว่าจะยอมให้ทำงานต่อ หรือหยุดทำงาน
* ถ้าใช้ค่า 1 โปรแกรมจะเก็บสถานะ error โดยไม่แจ้งให้ผู้ใช้ตัดสินใจ
* ถ้าใช้ค่าเป็น 2, 4, 6 โปรแกรมจะฟ้องเตือน และรอผู้ให้ผู้ใช้ตัดสินใจภายในเวลาที่กำหนด (2,4 หรือ 6 วินาที) ถ้าผู้ใช้ไม่กดภายในเวลา โปรแกรมจะเก็บสถานะเป็น error 
text/plain
.txt .text .js .vbs .asp .cgi .pl
----
text/html
.htm .html .hta .htx .mht
----
text/comma-separated-values
.csv
----
text/javascript
.js
----
text/css
.css
----
text/xml
.xml .xsl .xslt
----
image/gif
.gif
----
image/jpeg
.jpg .jpe .jpeg
----
image/png
.png
----
image/bmp
.bmp
----
image/tiff
.tif .tiff
----
audio/basic
.au .snd
----
audio/wav
.wav
----
audio/x-pn-realaudio
.ra .rm .ram
----
audio/x-midi
.mid .midi
----
audio/mp3
.mp3
----
audio/m3u
.m3u
----
video/x-ms-asf
.asf
----
video/avi
.avi
----
video/mpeg
.mpg .mpeg
----
video/quicktime
.qt .mov .qtvr
----
application/pdf
.pdf
----
application/rtf
.rtf
----
application/postscript
.ai .eps .ps
----
application/wordperfect
.wpd
----
application/mswrite
.wri
----
application/msexcel
.xls .xls3 .xls4 .xls5 .xlw
----
application/msword
.doc
----
application/mspowerpoint
.ppt .pps
----
application/x-director
.swa
----
application/x-shockwave-flash
.swf
----
application/x-zip-compressed
.zip
----
application/x-gzip
.gz
----
application/x-rar-compressed
.rar
----
application/octet-stream
.com .exe .dll .ocx
----
application/java-archive
.jar
/***
|Name|AttachFilePlugin|
|Source|http://www.TiddlyTools.com/#AttachFilePlugin|
|Documentation|http://www.TiddlyTools.com/#AttachFilePluginInfo|
|Version|4.0.0|
|Author|Eric Shulman|
|License|http://www.TiddlyTools.com/#LegalStatements|
|~CoreVersion|2.1|
|Type|plugin|
|Requires|AttachFilePluginFormatters, AttachFileMIMETypes|
|Description|Store binary files as base64-encoded tiddlers with fallback links for separate local and/or remote file storage|
Store or link binary files (such as jpg, gif, pdf or even mp3) within your TiddlyWiki document and then use them as images or links from within your tiddler content.
> Important note: As of version 3.6.0, in order to //render// images and other binary attachments created with this plugin, you must also install [[AttachFilePluginFormatters]], which extends the behavior of the TiddlyWiki core formatters for embedded images ({{{[img[tooltip|image]]}}}), linked embedded images ({{{[img[tooltip|image][link]]}}}), and external/"pretty" links ({{{[[label|link]]}}}), so that these formatter will process references to attachment tiddlers as if a normal file reference had been provided. |
!!!!!Documentation
>see [[AttachFilePluginInfo]]
!!!!!Inline interface (live)
>see [[AttachFile]] (shadow tiddler)
><<tiddler AttachFile>>
!!!!!Revisions
<<<
2009.06.04 [4.0.0] changed attachment storage format to use //sections// instead of embedded substring markers.
|please see [[AttachFilePluginInfo]] for additional revision details|
2005.07.20 [1.0.0] Initial Release
<<<
!!!!!Code
***/
// // version
//{{{
version.extensions.AttachFilePlugin= {major: 4, minor: 0, revision: 0, date: new Date(2009,6,4)};

// shadow tiddler
config.shadowTiddlers.AttachFile="<<attach inline>>";

// add 'attach' backstage task (insert before built-in 'importTask')
if (config.tasks) { // for TW2.2b or above
	config.tasks.attachTask = {
		text: "attach",
		tooltip: "Attach a binary file as a tiddler",
		content: "<<attach inline>>"
	}
	config.backstageTasks.splice(config.backstageTasks.indexOf("importTask"),0,"attachTask");
}

config.macros.attach = {
// // lingo
//{{{
	label: "attach file",
	tooltip: "Attach a file to this document",
	linkTooltip: "Attachment: ",

	typeList: "AttachFileMIMETypes",

	titlePrompt: " enter tiddler title...",
	MIMEPrompt: "<option value=''>select MIME type...</option><option value='editlist'>[edit list...]</option>",
	localPrompt: " enter local path/filename...",
	URLPrompt: " enter remote URL...",

	tiddlerErr: "Please enter a tiddler title",
	sourceErr: "Please enter a source path/filename",
	storageErr: "Please select a storage method: embedded, local or remote",
	MIMEErr: "Unrecognized file format.  Please select a MIME type",
	localErr: "Please enter a local path/filename",
	URLErr: "Please enter a remote URL",
	fileErr: "Invalid path/file or file not found",

	tiddlerFormat: '!usage\n{{{%0}}}\n%0\n!notes\n%1\n!type\n%2\n!file\n%3\n!url\n%4\n!data\n%5\n',

//}}}
// // macro definition
//{{{
	handler:
	function(place,macroName,params) {
		if (params && !params[0])
			{ createTiddlyButton(place,this.label,this.tooltip,this.toggleAttachPanel); return; }
		var id=params.shift();
		this.createAttachPanel(place,id+"_attachPanel",params);
		document.getElementById(id+"_attachPanel").style.position="static";
		document.getElementById(id+"_attachPanel").style.display="block";
	},
//}}}
//{{{
	createAttachPanel:
	function(place,panel_id,params) {
		if (!panel_id || !panel_id.length) var panel_id="_attachPanel";
		// remove existing panel (if any)
		var panel=document.getElementById(panel_id); if (panel) panel.parentNode.removeChild(panel);
		// set styles for this panel
		setStylesheet(this.css,"attachPanel");
		// create new panel
		var title=""; if (params && params[0]) title=params.shift();
		var types=this.MIMEPrompt+this.formatListOptions(store.getTiddlerText(this.typeList)); // get MIME types
		panel=createTiddlyElement(place,"span",panel_id,"attachPanel",null);
		var html=this.html.replace(/%id%/g,panel_id);
		html=html.replace(/%title%/g,title);
		html=html.replace(/%disabled%/g,title.length?"disabled":"");
		html=html.replace(/%IEdisabled%/g,config.browser.isIE?"disabled":"");
		html=html.replace(/%types%/g,types);
		panel.innerHTML=html;
		if (config.browser.isGecko) { // FF3 FIXUP
			document.getElementById("attachSource").style.display="none";
			document.getElementById("attachFixPanel").style.display="block";
		}
		return panel;
	},
//}}}
//{{{
	toggleAttachPanel:
	function (e) {
		if (!e) var e = window.event;
		var parent=resolveTarget(e).parentNode;
		var panel = document.getElementById("_attachPanel");
		if (panel==undefined || panel.parentNode!=parent)
			panel=config.macros.attach.createAttachPanel(parent,"_attachPanel");
		var isOpen = panel.style.display=="block";
		if(config.options.chkAnimate)
			anim.startAnimating(new Slider(panel,!isOpen,e.shiftKey || e.altKey,"none"));
		else
			panel.style.display = isOpen ? "none" : "block" ;
		e.cancelBubble = true;
		if (e.stopPropagation) e.stopPropagation();
		return(false);
	},
//}}}
//{{{
	formatListOptions:
	function(text) {
		if (!text || !text.trim().length) return "";
		// get MIME list content from text
		var parts=text.split("\n----\n");
		var out="";
		for (var p=0; p<parts.length; p++) {
			var lines=parts[p].split("\n");
			var label=lines.shift(); // 1st line=display text
			var value=lines.shift(); // 2nd line=item value
			out +='<option value="%1">%0</option>'.format([label,value]);
		}
		return out;
	},
//}}}
// // interface definition
//{{{
	css:
	".attachPanel { display: none; position:absolute; z-index:10; width:35em; right:105%; top:0em;\
		background-color: #eee; color:#000; font-size: 8pt; line-height:110%;\
		border:1px solid black; border-bottom-width: 3px; border-right-width: 3px;\
		padding: 0.5em; margin:0em; -moz-border-radius:1em;-webkit-border-radius:1em; text-align:left }\
	.attachPanel form { display:inline;border:0;padding:0;margin:0; }\
	.attachPanel select { width:99%;margin:0px;font-size:8pt;line-height:110%;}\
	.attachPanel input  { width:98%;padding:0px;margin:0px;font-size:8pt;line-height:110%}\
	.attachPanel textarea { width:98%;margin:0px;height:2em;font-size:8pt;line-height:110%}\
	.attachPanel table { width:100%;border:0;margin:0;padding:0;color:inherit; }\
	.attachPanel tbody, .attachPanel tr, .attachPanel td { border:0;margin:0;padding:0;color:#000; }\
	.attachPanel .box { border:1px solid black; padding:.3em; margin:.3em 0px; background:#f8f8f8; \
		-moz-border-radius:5px;-webkit-border-radius:5px; }\
	.attachPanel .chk { width:auto;border:0; }\
	.attachPanel .btn { width:auto; }\
	.attachPanel .btn2 { width:49%; }\
	",
//}}}
//{{{
	html:
	'<form>\
		attach from source file\
		<input type="file" id="attachSource" name="source" size="56"\
			onChange="config.macros.attach.onChangeSource(this)">\
		<div id="attachFixPanel" style="display:none"><!-- FF3 FIXUP -->\
			<input type="text" id="attachFixSource" style="width:90%"\
				title="Enter a path/file to attach"\
				onChange="config.macros.attach.onChangeSource(this);">\
			<input type="button" style="width:7%" value="..."\
				title="Enter a path/file to attach"\
				onClick="config.macros.attach.askForFilename(document.getElementById(\'attachFixSource\'));">\
		</div><!--end FF3 FIXUP-->\
		<div class="box">\
		<table style="border:0"><tr style="border:0"><td style="border:0;text-align:right;width:1%;white-space:nowrap">\
			embed data <input type=checkbox class=chk name="useData" %IEdisabled% \
				onclick="if (!this.form.MIMEType.value.length)\
					this.form.MIMEType.selectedIndex=this.checked?1:0; ">&nbsp;\
		</td><td style="border:0">\
			<select size=1 name="MIMEType" \
				onchange="this.title=this.value; if (this.value==\'editlist\')\
					{ this.selectedIndex=this.form.useData.checked?1:0; story.displayTiddler(null,config.macros.attach.typeList,2); return; }">\
				<option value=""></option>\
				%types%\
			</select>\
		</td></tr><tr style="border:0"><td style="border:0;text-align:right;width:1%;white-space:nowrap">\
			local link <input type=checkbox class=chk name="useLocal"\
				onclick="this.form.local.value=this.form.local.defaultValue=this.checked?config.macros.attach.localPrompt:\'\';">&nbsp;\
		</td><td style="border:0">\
			<input type=text name="local" size=15 autocomplete=off value=""\
				onchange="this.form.useLocal.checked=this.value.length" \
				onkeyup="this.form.useLocal.checked=this.value.length" \
				onfocus="if (!this.value.length) this.value=config.macros.attach.localPrompt; this.select()">\
		</td></tr><tr style="border:0"><td style="border:0;text-align:right;width:1%;white-space:nowrap">\
			remote link <input type=checkbox class=chk name="useURL"\
				onclick="this.form.URL.value=this.form.URL.defaultValue=this.checked?config.macros.attach.URLPrompt:\'\';\">&nbsp;\
		</td><td style="border:0">\
			<input type=text name="URL" size=15 autocomplete=off value=""\
				onfocus="if (!this.value.length) this.value=config.macros.attach.URLPrompt; this.select()"\
				onchange="this.form.useURL.checked=this.value.length;"\
				onkeyup="this.form.useURL.checked=this.value.length;">\
		</td></tr></table>\
		</div>\
		<table style="border:0"><tr style="border:0"><td style="border:0;text-align:right;vertical-align:top;width:1%;white-space:nowrap">\
			notes&nbsp;\
		</td><td style="border:0" colspan=2>\
			<textarea name="notes" style="width:98%;height:3.5em;margin-bottom:2px"></textarea>\
		</td><tr style="border:0"><td style="border:0;text-align:right;width:1%;white-space:nowrap">\
			attach as&nbsp;\
		</td><td style="border:0" colspan=2>\
			<input type=text name="tiddlertitle" size=15 autocomplete=off value="%title%"\
				onkeyup="if (!this.value.length) { this.value=config.macros.attach.titlePrompt; this.select(); }"\
				onfocus="if (!this.value.length) this.value=config.macros.attach.titlePrompt; this.select()" %disabled%>\
		</td></tr></tr><tr style="border:0"><td style="border:0;text-align:right;width:1%;white-space:nowrap">\
			add tags&nbsp;\
		</td><td style="border:0">\
			<input type=text name="tags" size=15 autocomplete=off value="" onfocus="this.select()">\
		</td><td style="width:40%;text-align:right;border:0">\
			<input type=button class=btn2 value="attach"\
				onclick="config.macros.attach.onClickAttach(this)"><!--\
			--><input type=button class=btn2 value="close"\
				onclick="var panel=document.getElementById(\'%id%\'); if (panel) panel.parentNode.removeChild(panel);">\
		</td></tr></table>\
	</form>',
//}}}
// // control processing
//{{{
	onChangeSource:
	function(here) {
		var form=here.form;
		var list=form.MIMEType;
		var theFilename  = here.value;
		var theExtension = theFilename.substr(theFilename.lastIndexOf('.')).toLowerCase();
		// if theFilename is in current document folder, remove path prefix and use relative reference
		var h=document.location.href; folder=getLocalPath(decodeURIComponent(h.substr(0,h.lastIndexOf("/")+1)));
		if (theFilename.substr(0,folder.length)==folder) theFilename='./'+theFilename.substr(folder.length);
		else theFilename='file:///'+theFilename; // otherwise, use absolute reference
		theFilename=theFilename.replace(/\\/g,"/"); // fixup: change \ to /
		form.useLocal.checked = true;
		form.local.value = theFilename;
		form.useData.checked = !form.useData.disabled;
		list.selectedIndex=1;
		for (var i=0; i<list.options.length; i++) // find matching MIME type
			if (list.options[i].value.indexOf(theExtension)!=-1) { list.selectedIndex = i; break; }
		if (!form.tiddlertitle.disabled)
			form.tiddlertitle.value=theFilename.substr(theFilename.lastIndexOf('/')+1); // get tiddlername from filename
	},
//}}}
//{{{
	onClickAttach:
	function (here) {
		clearMessage();
		// get input values
		var form=here.form;
		var src=form.source; if (config.browser.isGecko) src=document.getElementById("attachFixSource");
		src=src.value!=src.defaultValue?src.value:"";
		var when=(new Date()).formatString(config.macros.timeline.dateFormat);
		var title=form.tiddlertitle.value;
		var local = form.local.value!=form.local.defaultValue?form.local.value:"";
		var url = form.URL.value!=form.URL.defaultValue?form.URL.value:"";
		var notes = form.notes.value;
		var tags = "attachment excludeMissing "+form.tags.value;
		var useData=form.useData.checked;
		var useLocal=form.useLocal.checked;
		var useURL=form.useURL.checked;
		var mimetype = form.MIMEType.value.length?form.MIMEType.options[form.MIMEType.selectedIndex].text:"";
		// validate checkboxes and get filename
		if (useData) {
			if (src.length) { if (!theLocation) var theLocation=src; }
			else { alert(this.sourceErr); src.focus(); return false; }
		}
		if (useLocal) {
			if (local.length) { if (!theLocation) var theLocation = local; }
			else { alert(this.localErr); form.local.focus(); return false; }
		}
		if (useURL) {
			if (url.length) { if (!theLocation) var theLocation = url; }
			else { alert(this.URLErr); form.URL.focus(); return false; }
		}
		if (!(useData||useLocal||useURL))
			{ form.useData.focus(); alert(this.storageErr); return false; }
		if (!theLocation)
			{ src.focus(); alert(this.sourceErr); return false; }
		if (!title || !title.trim().length || title==this.titlePrompt)
			{ form.tiddlertitle.focus(); alert(this.tiddlerErr); return false; }
		// if not already selected, determine MIME type based on filename extension (if any)
		if (useData && !mimetype.length && theLocation.lastIndexOf('.')!=-1) {
			var theExt = theLocation.substr(theLocation.lastIndexOf('.')).toLowerCase();
			var theList=form.MIMEType;
			for (var i=0; i<theList.options.length; i++)
				if (theList.options[i].value.indexOf(theExt)!=-1)
					{ var mimetype=theList.options[i].text; theList.selectedIndex=i; break; }
		}
		// attach the file
		return this.createAttachmentTiddler(src, when, notes, tags, title,
			useData, useLocal, useURL, local, url, mimetype);
	},
	getMIMEType:
	function(src,def) {
		var ext = src.substr(src.lastIndexOf('.')).toLowerCase();
		var list=store.getTiddlerText(this.typeList);
		if (!list || !list.trim().length) return def;
		// get MIME list content from tiddler
		var parts=list.split("\n----\n");
		for (var p=0; p<parts.length; p++) {
			var lines=parts[p].split("\n");
			var mime=lines.shift(); // 1st line=MIME type
			var match=lines.shift(); // 2nd line=matching extensions
			if (match.indexOf(ext)!=-1) return mime;
		}
		return def;
	},
	createAttachmentTiddler:
	function (src, when, notes, tags, title, useData, useLocal, useURL, local, url, mimetype, noshow) {
		if (useData) { // encode the data
			if (!mimetype.length) {
				alert(this.MIMEErr);
				form.MIMEType.selectedIndex=1; form.MIMEType.focus();
				return false;
			}
			var d = this.readFile(src); if (!d) { return false; }
			displayMessage('encoding '+src);
			var encoded = this.encodeBase64(d);
			displayMessage('file size='+d.length+' bytes, encoded size='+encoded.length+' bytes');
		}
		var usage=(mimetype.substr(0,5)=="image"?'[img[%0]]':'[[%0|%0]]').format([title]);
		var theText=this.tiddlerFormat.format([
			usage, notes.length?notes:'//none//', mimetype,
			useLocal?local.replace(/\\/g,'/'):'', useURL?url:'',
			useData?('data:'+mimetype+';base64,'+encoded):'' ]);
		store.saveTiddler(title,title,theText,config.options.txtUserName,new Date(),tags);
		var panel=document.getElementById("attachPanel"); if (panel) panel.style.display="none";
		if (!noshow) { story.displayTiddler(null,title); story.refreshTiddler(title,null,true); }
		displayMessage('attached "'+title+'"');
		return true;
	},
//}}}
// // base64 conversion
//{{{
	encodeBase64:
	function (d) {
		if (!d) return null;
		// encode as base64
		var keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
		var out="";
		var chr1,chr2,chr3="";
		var enc1,enc2,enc3,enc4="";
		for (var count=0,i=0; i<d.length; ) {
			chr1=d.charCodeAt(i++);
			chr2=d.charCodeAt(i++);
			chr3=d.charCodeAt(i++);
			enc1=chr1 >> 2;
			enc2=((chr1 & 3) << 4) | (chr2 >> 4);
			enc3=((chr2 & 15) << 2) | (chr3 >> 6);
			enc4=chr3 & 63;
			if (isNaN(chr2)) enc3=enc4=64;
			else if (isNaN(chr3)) enc4=64;
			out+=keyStr.charAt(enc1)+keyStr.charAt(enc2)+keyStr.charAt(enc3)+keyStr.charAt(enc4);
			chr1=chr2=chr3=enc1=enc2=enc3=enc4="";
		}
		return out;
	},
	decodeBase64: function(input) {
		var out="";
		var chr1,chr2,chr3;
		var enc1,enc2,enc3,enc4;
		var i = 0;
		// remove all characters that are not A-Z, a-z, 0-9, +, /, or =
		input=input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
		do {
			enc1=keyStr.indexOf(input.charAt(i++));
			enc2=keyStr.indexOf(input.charAt(i++));
			enc3=keyStr.indexOf(input.charAt(i++));
			enc4=keyStr.indexOf(input.charAt(i++));
			chr1=(enc1 << 2) | (enc2 >> 4);
			chr2=((enc2 & 15) << 4) | (enc3 >> 2);
			chr3=((enc3 & 3) << 6) | enc4;
			out=out+String.fromCharCode(chr1);
			if (enc3!=64) out=out+String.fromCharCode(chr2);
			if (enc4!=64) out=out+String.fromCharCode(chr3);
		} while (i<input.length);
		return out;
	},
//}}}
// // I/O functions
//{{{
	readFile: // read local BINARY file data
	function(filePath) {
		if(!window.Components) { return null; }
		try { netscape.security.PrivilegeManager.enablePrivilege("UniversalXPConnect"); }
		catch(e) { alert("access denied: "+filePath); return null; }
		var file = Components.classes["@mozilla.org/file/local;1"].createInstance(Components.interfaces.nsILocalFile);
		try { file.initWithPath(filePath); } catch(e) { alert("cannot read file - invalid path: "+filePath); return null; }
		if (!file.exists()) { alert("cannot read file - not found: "+filePath); return null; }
		var inputStream = Components.classes["@mozilla.org/network/file-input-stream;1"].createInstance(Components.interfaces.nsIFileInputStream);
		inputStream.init(file, 0x01, 00004, null);
		var bInputStream = Components.classes["@mozilla.org/binaryinputstream;1"].createInstance(Components.interfaces.nsIBinaryInputStream);
		bInputStream.setInputStream(inputStream);
		return(bInputStream.readBytes(inputStream.available()));
	},
//}}}
//{{{
	writeFile:
	function(filepath,data) {
		// TBD: decode base64 and write BINARY data to specified local path/filename
		return(false);
	},
//}}}
//{{{
	askForFilename: // for FF3 fixup
	function(target) {
		var msg=config.messages.selectFile;
		if (target && target.title) msg=target.title; // use target field tooltip (if any) as dialog prompt text
		// get local path for current document
		var path=getLocalPath(document.location.href);
		var p=path.lastIndexOf("/"); if (p==-1) p=path.lastIndexOf("\\"); // Unix or Windows
		if (p!=-1) path=path.substr(0,p+1); // remove filename, leave trailing slash
		var file=""
		var result=window.mozAskForFilename(msg,path,file,true); // FF3 FIXUP ONLY
		if (target && result.length) // set target field and trigger handling
			{ target.value=result; target.onchange(); }
		return result; 
	}
};
//}}}
//{{{
if (window.mozAskForFilename===undefined) { // also defined by CoreTweaks (for ticket #604)
	window.mozAskForFilename=function(msg,path,file,mustExist) {
		if(!window.Components) return false;
		try {
			netscape.security.PrivilegeManager.enablePrivilege('UniversalXPConnect');
			var nsIFilePicker = window.Components.interfaces.nsIFilePicker;
			var picker = Components.classes['@mozilla.org/filepicker;1'].createInstance(nsIFilePicker);
			picker.init(window, msg, mustExist?nsIFilePicker.modeOpen:nsIFilePicker.modeSave);
			var thispath = Components.classes['@mozilla.org/file/local;1'].createInstance(Components.interfaces.nsILocalFile);
			thispath.initWithPath(path);
			picker.displayDirectory=thispath;
			picker.defaultExtension='';
			picker.defaultString=file;
			picker.appendFilters(nsIFilePicker.filterAll|nsIFilePicker.filterText|nsIFilePicker.filterHTML);
			if (picker.show()!=nsIFilePicker.returnCancel)
				var result=picker.file.persistentDescriptor;
		}
		catch(ex) { displayMessage(ex.toString()); }
		return result;
	}
}
//}}}
/***
|Name|AttachFilePluginFormatters|
|Source|http://www.TiddlyTools.com/#AttachFilePluginFormatters|
|Version|4.0.1|
|Author|Eric Shulman|
|License|http://www.TiddlyTools.com/#LegalStatements|
|~CoreVersion|2.1.3|
|Type|plugin|
|Description|run-time library for displaying attachment tiddlers|
Runtime processing for //rendering// attachment tiddlers created by [[AttachFilePlugin]].   Attachment tiddlers are tagged with<<tag attachment>>and contain binary file content (e.g., jpg, gif, pdf, mp3, etc.) that has been stored directly as base64 text-encoded data or can be loaded from external files stored on a local filesystem or remote web server.  Note: after creating new attachment tiddlers, you can remove [[AttachFilePlugin]], as long as you retain //this// tiddler (so that images can be rendered later on).
!!!!!Formatters
<<<
This plugin extends the behavior of the following TiddlyWiki core "wikify()" formatters:
* embedded images: {{{[img[tooltip|image]]}}}
* linked embedded images: {{{[img[tooltip|image][link]]}}}
* external/"pretty" links: {{{[[label|link]]}}}
''Please refer to AttachFilePlugin (source: http://www.TiddlyTools.com/#AttachFilePlugin) for additional information.''
<<<
!!!!!Revisions
<<<
2009.10.10 [4.0.1] in fileExists(), check for IE to avoid hanging Chrome during startup
2009.06.04 [4.0.0] changed attachment storage format to use //sections// instead of embedded substring markers.
2008.01.08 [*.*.*] plugin size reduction: documentation moved to ...Info
2007.12.04 [*.*.*] update for TW2.3.0: replaced deprecated core functions, regexps, and macros
2007.10.29 [3.7.0] more code reduction: removed upload handling from AttachFilePlugin (saves ~7K!)
2007.10.28 [3.6.0] removed duplicate formatter code from AttachFilePlugin (saves ~10K!) and updated documentation accordingly.  This plugin ([[AttachFilePluginFormatters]]) is now //''required''// in order to display attached images/binary files within tiddler content.
2006.05.20 [3.4.0] through 2007.03.01 [3.5.3] sync with AttachFilePlugin
2006.05.13 [3.2.0] created from AttachFilePlugin v3.2.0
<<<
!!!!!Code
***/
// // version
//{{{
version.extensions.AttachFilePluginFormatters= {major: 4, minor: 0, revision: 1, date: new Date(2009,10,10)};
//}}}

//{{{
if (config.macros.attach==undefined) config.macros.attach= { };
//}}}
//{{{
if (config.macros.attach.isAttachment==undefined) config.macros.attach.isAttachment=function (title) {
	var tiddler = store.getTiddler(title);
	if (tiddler==undefined || tiddler.tags==undefined) return false;
	return (tiddler.tags.indexOf("attachment")!=-1);
}
//}}}

//{{{
// test for local file existence - returns true/false without visible error display
if (config.macros.attach.fileExists==undefined) config.macros.attach.fileExists=function(f) {
	if(window.Components) { // MOZ
		try { netscape.security.PrivilegeManager.enablePrivilege("UniversalXPConnect"); }
		catch(e) { return false; } // security access denied
		var file = Components.classes["@mozilla.org/file/local;1"].createInstance(Components.interfaces.nsILocalFile);
		try { file.initWithPath(f); }
		catch(e) { return false; } // invalid directory
		return file.exists();
	}
	else if (config.browser.isIE) { // IE
		var fso = new ActiveXObject("Scripting.FileSystemObject");
		return fso.FileExists(f);
	}
	else return true; // other browsers: assume file exists
}
//}}}

//{{{
if (config.macros.attach.getAttachment==undefined) config.macros.attach.getAttachment=function(title) {

	// extract embedded data, local and remote links (if any)
	var text=store.getTiddlerText(title,'');
	var embedded=store.getTiddlerText(title+'##data','').trim();
	var locallink=store.getTiddlerText(title+'##file','').trim();
	var remotelink=store.getTiddlerText(title+'##url','').trim();

	// backward-compatibility for older attachments (pre 4.0.0)
	var startmarker="---BEGIN_DATA---\n";
	var endmarker="\n---END_DATA---";
	var pos=0; var endpos=0;
	if ((pos=text.indexOf(startmarker))!=-1 && (endpos=text.indexOf(endmarker))!=-1)
		embedded="data:"+(text.substring(pos+startmarker.length,endpos)).replace(/\n/g,'');
	if ((pos=text.indexOf("/%LOCAL_LINK%/"))!=-1)
		locallink=text.substring(text.indexOf("|",pos)+1,text.indexOf("]]",pos));
	if ((pos=text.indexOf("/%REMOTE_LINK%/"))!=-1)
		remotelink=text.substring(text.indexOf("|",pos)+1,text.indexOf("]]",pos));

	// if there is a data: URI defined (not supported by IE)
	if (embedded.length && !config.browser.isIE) return embedded;

	// document is being served remotely... use remote URL (if any)  (avoids security alert)
	if (remotelink.length && document.location.protocol!="file:")
		return remotelink;  

	// local link only... return link without checking file existence (avoids security alert)
	if (locallink.length && !remotelink.length) 
		return locallink; 

	// local link, check for file exist... use local link if found
	if (locallink.length) { 
		locallink=locallink.replace(/^\.[\/\\]/,''); // strip leading './' or '.\' (if any)
		if (this.fileExists(getLocalPath(locallink))) return locallink;
		// maybe local link is relative... add path from current document and try again
		var pathPrefix=document.location.href;  // get current document path and trim off filename
		var slashpos=pathPrefix.lastIndexOf("/"); if (slashpos==-1) slashpos=pathPrefix.lastIndexOf("\\"); 
		if (slashpos!=-1 && slashpos!=pathPrefix.length-1) pathPrefix=pathPrefix.substr(0,slashpos+1);
		if (this.fileExists(getLocalPath(pathPrefix+locallink))) return locallink;
	}

	// no embedded data, no local (or not found), fallback to remote URL (if any)
	if (remotelink.length) return remotelink;

	// attachment URL doesn't resolve, just return input as is
	return title;
}
//}}}
//{{{
if (config.macros.attach.init_formatters==undefined) config.macros.attach.init_formatters=function() {
	if (this.initialized) return;

	// find the formatter for "image" and replace the handler
	for (var i=0; i<config.formatters.length && config.formatters[i].name!="image"; i++);
	if (i<config.formatters.length)	config.formatters[i].handler=function(w) {
		this.lookaheadRegExp.lastIndex = w.matchStart;
		var lookaheadMatch = this.lookaheadRegExp.exec(w.source)
		if(lookaheadMatch && lookaheadMatch.index == w.matchStart) // Simple bracketted link
			{
			var e = w.output;
			if(lookaheadMatch[5])
				{
				var link = lookaheadMatch[5];
				// ELS -------------
				var external=config.formatterHelpers.isExternalLink(link);
				if (external)
					{
					if (config.macros.attach.isAttachment(link))
						{
						e = createExternalLink(w.output,link);
						e.href=config.macros.attach.getAttachment(link);
						e.title = config.macros.attach.linkTooltip + link;
						}
					else
						e = createExternalLink(w.output,link);
					}
				else 
					e = createTiddlyLink(w.output,link,false,null,w.isStatic);
				// ELS -------------
				addClass(e,"imageLink");
				}
			var img = createTiddlyElement(e,"img");
			if(lookaheadMatch[1])
				img.align = "left";
			else if(lookaheadMatch[2])
				img.align = "right";
			if(lookaheadMatch[3])
				img.title = lookaheadMatch[3];
			img.src = lookaheadMatch[4];
			// ELS -------------
			if (config.macros.attach.isAttachment(lookaheadMatch[4]))
				img.src=config.macros.attach.getAttachment(lookaheadMatch[4]);
			// ELS -------------
			w.nextMatch = this.lookaheadRegExp.lastIndex;
		}
	}
//}}}
//{{{
	// find the formatter for "prettyLink" and replace the handler
	for (var i=0; i<config.formatters.length && config.formatters[i].name!="prettyLink"; i++);
	if (i<config.formatters.length)	{
		config.formatters[i].handler=function(w) {
			this.lookaheadRegExp.lastIndex = w.matchStart;
			var lookaheadMatch = this.lookaheadRegExp.exec(w.source);
			if(lookaheadMatch && lookaheadMatch.index == w.matchStart) {
				var e;
				var text = lookaheadMatch[1];
				if(lookaheadMatch[3]) {
					// Pretty bracketted link
					var link = lookaheadMatch[3];
					if (config.macros.attach.isAttachment(link)) {
						e = createExternalLink(w.output,link);
						e.href=config.macros.attach.getAttachment(link);
						e.title=config.macros.attach.linkTooltip+link;
					}
					else e = (!lookaheadMatch[2] && config.formatterHelpers.isExternalLink(link))
						? createExternalLink(w.output,link)
						: createTiddlyLink(w.output,link,false,null,w.isStatic);
				} else {
					e = createTiddlyLink(w.output,text,false,null,w.isStatic);
				}
				createTiddlyText(e,text);
				w.nextMatch = this.lookaheadRegExp.lastIndex;
			}
		}
	} // if "prettyLink" formatter found
	this.initialized=true;
}
//}}}
//{{{
config.macros.attach.init_formatters(); // load time init
//}}}
//{{{
if (TiddlyWiki.prototype.coreGetRecursiveTiddlerText==undefined) {
	TiddlyWiki.prototype.coreGetRecursiveTiddlerText = TiddlyWiki.prototype.getRecursiveTiddlerText;
	TiddlyWiki.prototype.getRecursiveTiddlerText = function(title,defaultText,depth) {
		return config.macros.attach.isAttachment(title)?
			config.macros.attach.getAttachment(title):this.coreGetRecursiveTiddlerText.apply(this,arguments);
	}
}
//}}}
/***
|''Name:''|BackupOptionsPlugin|
|''Version:''|1.0.1 (2007-09-29)|
|''Source:''|None|
|''Author:''|Tyler Akins|
|''Licence:''|Public domain|
|''TiddlyWiki:''|2.0+|
|''Browser:''|Firefox 1.0.4+; InternetExplorer 6.0|

!Description
Tired of having thousands of backups made due to saving every minor edit?  Do you only want one backup per hour, day, or just one backup ever?  Not a problem.

This plugin lets you define a file format that you want to use when saving backups.  Because backups will overwrite each other if they have the same name, you can now control how often new backup files are created.  If you want one created every day, just include the year, month, and day in your format and avoid using the hours, minutes, seconds, and milliseconds.  If you want only one backup, set a static name and it will just keep overwriting the old file.

!Configuration
Select what attributes you want to include in the backup filename in the order you like.  Dates are all in UTC format.  If the format field is left blank, it defaults to what the backups would normally be named:  {{{%N.%Y%M%D.%h%m%s%n.html}}}

{{wideInput{<<option txtBackupOptionsFormat 40>>}}}

|!Code|!Description|!Example|
| ''%D'' |Day of month, two digits| 18 |
| ''%h'' |Hour, two digits, 24 hour format| 21 |
| ''%M'' |Month number, two digits| 11 |
| ''%m'' |Minute, two digits| 59 |
| ''%N'' |Base name of the wiki| TiddlyWiki |
| ''%n'' |Millisecond, four digits| 8441 |
| ''%s'' |Seconds, two digits| 06 |
| ''%Y'' |Year, four digits| 2006 |
| ''%y'' |Year, two digits| 06 |
| ''%%'' |A percent symbol| % |

!Examples
Based on a base filename of "TiddlyWiki.html", and a date of 2006-11-18 21:59:06.8441, here are a few format options:

|!''Format''|!Description|
|!//Sample//|~|
| ''%N.bak'' |Saves only one backup, ever.  Always overwrites the .bak file with a new backup, keeping just one file around.|
| //TiddlyWiki.bak// |~|
| ''%N.%Y%M%D.%h%m%s%n.html'' |This is the default format that TiddlyWiki uses when making a new backup.|
| //TiddlyWiki.20061118.2159068441.html// |~|
| ''%N-%Y-%M-%D.html'' |Keep around only one backup per day.  When a new backup is made, it will overwrite any other backups made on that day.|
| //TiddlyWiki-2006-11-18.html// |~|
| ''Backups\%Y%M\%N-%D-%h%m.bak'' |Save all backups in a set of directories, with one directory that contains all, then another subdirectory that holds a year and month, and then the backup file.|
| //Backups\200611\TiddlyWiki-18-2159.bak// |~|

!Revision history
* v1.0.0 (2007-09-29)
** Initial release

!Code
***/
//{{{
//============================================================================
//                           BackupOptionsPlugin

// Ensure that the BackupOptionsPlugin is only installed once.
//

if (!version.extensions.BackupOptionsPlugin) {

setStylesheet(".wideInput input { width:30em; }","BackupOptionsStylesheet");

version.extensions.BackupOptionsPlugin = {
    major: 1, minor: 0, revision: 0,
    date: new Date(2007, 9, 11), 
    type: 'plugin',
    source: "http://rumkin.com/tools/tiddlywiki/#BackupOptionsPlugin"
};


if (!config.options.txtBackupOptionsFormat)
	config.options.txtBackupOptionsFormat = "%N.%Y%M%D.%h%m%s%n.html"; // Same as default format
if (config.optionsDesc)
	config.optionsDesc.txtBackupOptionsFormat = "Filename format for backups."

if (version.major < 2) alertAndThrow("BackupOptionsPlugin requires TiddlyWiki 2.0 or newer.");

//============================================================================
// Overwrite the built-in functions

getBackupPath = function(localPath) {
	var formatString = config.options['txtBackupOptionsFormat'];
	if (formatString == undefined || ! formatString || formatString == '')
		formatString = '%N.%Y%M%D.%h%m%s%n.html';
	var backSlash = true;
	var dirPathPos = localPath.lastIndexOf("\\");
	if (dirPathPos == -1)
	{
		dirPathPos = localPath.lastIndexOf("/");
		backSlash = false;
	}
	var backupFolder = config.options.txtBackupFolder;
	if (! backupFolder || backupFolder == '')
		backupFolder = '.';
	backupFolder += (backSlash ? "\\" : '/');
	var backupPath = localPath.substr(0, dirPathPos) + (backSlash ? "\\" : '/') + backupFolder;
	var backupBase = localPath.substr(dirPathPos)
	backupBase = backupBase.substr(0, backupBase.lastIndexOf('.'));
	var d = new Date()
	while (formatString.length > 0)
	{
		var formatHandled = 0;
		if (formatString.length > 1 && formatString.charAt(0) == '%')
		{
			formatHandled = 1;
			switch (formatString.charAt(1))
			{
				case 'D':
					backupPath += String.zeroPad(d.getUTCDate(), 2);
					break;
				case 'h':
					backupPath += String.zeroPad(d.getUTCHours(), 2);
					break;
				case 'M':
					backupPath += String.zeroPad(d.getUTCMonth(), 2);
					break;
				case 'm':
					backupPath += String.zeroPad(d.getUTCMinutes(), 2);
					break;
				case 'N':
					backupPath += backupBase;
					break;
				case 'n':
					backupPath += String.zeroPad(d.getUTCMilliseconds(), 4);
					break;
				case 's':
					backupPath += String.zeroPad(d.getUTCSeconds(), 4);
					break;
				case 'Y':
					backupPath += String.zeroPad(d.getUTCFullYear(), 4);
					break;
				case 'y':
					backupPath += String.zeroPad(d.getUTCFullYear() % 100, 4);
					break;
				case '%':
					backupPath += '%';
					break;
				default:
					formatHandled = 0;
			}
			if (formatHandled)
				formatString = formatString.substr(2);
		}
		if (! formatHandled)
		{
			backupPath += formatString.charAt(0);
			formatString = formatString.substr(1);
		}
	}

	return backupPath;
}


} // of "install only once"
//}}}
/***

!Licence and Copyright
You are free to use this however you like.  I place this code into the public domain.
***/
!usage
{{{[img[Bracket2SubFld01.jpg]]}}}
[img[Bracket2SubFld01.jpg]]
!notes
//none//
!type
image/jpeg
!file
./img/Bracket2SubFld01.jpg
!url

!data

!usage
{{{[img[Bracket2SubFld02.jpg]]}}}
[img[Bracket2SubFld02.jpg]]
!notes
//none//
!type
image/jpeg
!file
./img/Bracket2SubFld02.jpg
!url

!data

!usage
{{{[img[Bracket2SubFld03.jpg]]}}}
[img[Bracket2SubFld03.jpg]]
!notes
//none//
!type
image/jpeg
!file
./img/Bracket2SubFld03.jpg
!url

!data

/***
|''Name:''|ChangeModePlugin|
|''Description:''|Change template and styleSheet|
|''Credits:''|SaqImtiaz for is PresentationPlugin|
|''Version:''|0.0.4|
|''Date:''|Jun 5, 2007|
|''Source:''|http://tiddlywiki.bidix.info/#changeModePlugin|
|''Usage:''|{{{<<changeMode [newMode]>>}}}<br>{{{newMode: if omitted the default mode is applied}}}|
|''Author:''|BidiX (BidiX (at) bidix (dot) info)|
|''[[License]]:''|[[BSD open source license|http://tiddlywiki.bidix.info/#%5B%5BBSD%20open%20source%20license%5D%5D ]]|
|''CoreVersion:''|2.2.0|
***/
//{{{
version.extensions.ChangeModePlugin = 
{
	major: 0, minor: 0, revision: 4, 
	date: new Date("Jun 9, 2007"),
	source: 'http://tiddlywiki.bidix.info/#ChangeModePlugin',
	author: 'BidiX (BidiX (at) bidix (dot) info',
	coreVersion: '2.2.0'
};

// From SaqImtiaz's PresentationPlugin
//---------------------------------------------------
TiddlyWiki.prototype.isTiddler= function (title) 
{
	return store.tiddlerExists(title) || store.isShadowTiddler(title);
};

TiddlyWiki.prototype.removeNotification = function(title,fn) 
{
	for (var i=0; i<this.namedNotifications.length; i++)
	if((this.namedNotifications[i].name == title) && (this.namedNotifications[i].notify == fn))
 		this.namedNotifications.splice(i,1);
};

Story.prototype.chooseTemplateForTiddler_core = Story.prototype.chooseTemplateForTiddler;

Story.prototype.chooseTemplateForTiddler = function(title,template)
{
	if (!template)
 		template = DEFAULT_VIEW_TEMPLATE;
 	var mode = config.macros.changeMode.currentMode;
 	if (template == DEFAULT_VIEW_TEMPLATE) {
 		if (store.isTiddler(mode+"ViewTemplate"))
 			return mode+"ViewTemplate";
 	} else if (template == DEFAULT_EDIT_TEMPLATE) {
 		if (store.isTiddler(mode+"EditTemplate"))
 			return mode+"EditTemplate";
 	}
 	return this.chooseTemplateForTiddler_core(title,template);
}


Story.prototype.lewcidrefreshAllTiddlers = function() 
{
	var place = document.getElementById(this.container);
 	var e = place.firstChild;
 	if(!e) return;
 	this.refreshTiddler(e.getAttribute("tiddler"),null,true);
 	while((e = e.nextSibling) != null)
 		this.refreshTiddler(e.getAttribute("tiddler"),null,true);
}
//---------------------------------------------------
// manage different modes
// 
// config.macros.changeMode.initMode: the name of the initial Mode
// config.macros.changeMode.readOnlyModes: array containing readOnly mode (no Backstage and readOnly)
//

config.macros.changeMode = 
{
	currentMode: '',	// defaultMode
	initMode: 'Reader',
	readOnlyModes : ['Reader'],
	noBackstage : ['Reader'],  // ['Reader', 'Author'],
	singlePageMode: [], // ['Reader', 'Author'],
	lingo: {
		label: "%0 mode",
		prompt: "Change the current mode to '%0'",
		modeName: {
			Author: 'Author',
			Reader: 'Reader',
			'': '(default)'
		}
	},
	handler: function(place,macroName,params) {
		var newMode = (params[0] ? params[0]: ''); // default to ''
		if (newMode == '') {
			if (this.currentMode=='Reader') newMode = 'Author';
			if (this.currentMode=='Author') newMode = 'Reader';
		}
		var newModeName = (this.lingo.modeName[newMode] ? this.lingo.modeName[newMode]: newMode); // default to ''
		var label = this.lingo.label.format([newModeName]);
		var prompt = this.lingo.prompt.format([newModeName]);
		createTiddlyButton(place, label, prompt, function() {config.macros.changeMode.action(newMode);}, null, null, null);		
	},	
	action: function(template) {
		config.macros.changeMode.applyMode(template);
	},
	defaults: [
		{name: "StyleSheet", notify: refreshStyles},
		{name: "PageTemplate", notify: refreshPageTemplate}
	],	
	applyMode: function (newMode) {
		var oldMode = this.currentMode;
		var oldStyleElement = document.getElementById(oldMode+"StyleSheet");
		if (oldStyleElement) {
			oldStyleElement.parentNode.removeChild(oldStyleElement);
		}
		// change Palette
		if (store.isTiddler(newMode + 'ColorPalette')) {
			var tiddler = new Tiddler('ColorPalette');
			//tiddler.tags.push('admin');
			if (!newMode) {
				if (store.isTiddler('defaultColorPalette'))
					tiddler.text = store.getTiddlerText('defaultColorPalette');
				else
					tiddler.text = config.shadowTiddlers['ColorPalette'];
				} else {
				tiddler.text = store.getTiddlerText(newMode + 'ColorPalette');
			}
			store.addTiddler(tiddler);
		}
		for (var i=0; i< this.defaults.length; i++)
		{
			var name = this.defaults[i]["name"];
			var newElement = store.isTiddler(newMode + name) ? newMode + name : name;
			store.removeNotification(oldMode + name, this.defaults[i]["notify"]);
			store.addNotification(newElement,this.defaults[i]["notify"]);
			store.notify(newElement); //just one do blanket notify instead?
		}
		if (backstage && !backstage.button)
			backstage.init();
		// change readOnly
		if (this.readOnlyModes.indexOf(newMode) == -1) {
			readOnly = false;
		}
		else {
			readOnly = true;
		}	
		// change backstage display
		if (backstage && backstage.button) {
			if (this.noBackstage.indexOf(newMode) == -1) {
				backstage.button.style.display = "block";
			}
			else {
				backstage.hide();
				backstage.button.style.display = "none";
			}
		// change singlePageMode
		if (this.singlePageMode.indexOf(newMode) == -1) {
			config.options.chkSinglePageMode = false;
		}
		else {
			config.options.chkSinglePageMode= true;
		}	

		}
			
		
		this.currentMode = newMode;
		story.lewcidrefreshAllTiddlers ();
		// store.refreshPalette();
		store.notifyAll();
		},
	init: function() {
		if (!store.isTiddler('defaultColorPalette'))
			config.shadowTiddlers['defaultColorPalette'] = config.shadowTiddlers['ColorPalette'];
		config.macros.changeMode.applyMode(this.initMode);
	}
}

config.paramifiers.mode = {
	onconfig: function(mode) {
		if (mode == 'false')
			config.macros.changeMode.initMode = null;
		else
			config.macros.changeMode.initMode = mode;	
	}
};


//}}}
|Default|@@""" ChkTrnClosed = 8 """@@|
กำหนดระดับ user ขั้นต่ำ ที่สามารถแก้ไขข้อมูล transaction ก่อนวันที่สั่งปิดไปแล้วได้ 

ประกอบด้วย transaction ของ สต็อก (เข้า-ออก), ลูกหนี้ (เพิ่ม-ลดหนี้), เจ้าหนี้ (เพิ่ม-ลดหนี้), บัญชีธนาคาร (นำเข้า-จ่ายออก) 

| 0 |user1 (อนุญาต user ทุกคน) |
| 1 |user2 |
| 2 |user3 |
| 3 |user4 |
| 4 |user5 |
| 5 |user6 |
| 6 |user7 |
| 7 |บัญชี |
| 8 |ผู้จัดการ |
| 9 |supervisor (ไม่อนุญาต user ทุกคน)|
Background: #fff
Foreground: #000
PrimaryPale: #8cf
PrimaryLight: #18f
PrimaryMid: #04b
PrimaryDark: #014
SecondaryPale: #ffc
SecondaryLight: #fe8
SecondaryMid: #db4
SecondaryDark: #841
TertiaryPale: #eee
TertiaryLight: #ccc
TertiaryMid: #999
TertiaryDark: #666
Error: #f88
//{{{
readOnly = true;
//}}}
|Default|@@""" CostDigit = 3 """@@|
จำนวนหลักหลังจุดทศนิยม ของ field __มูลค่าต้นทุนรวม__ของสินค้า 

ดูเพิ่มเติม: CostError มูลค่าต้นทุนต่อหน่วย
|Default|@@""" CostError = 6 """@@|
จำนวนหลักหลังจุดทศนิยม ของ field __มูลค่าต้นทุนต่อหน่วย__ของสินค้า

จำนวนทศนิยมของ CostError จะมีผลในการคำนวณต้นทุนเฉลี่ย ฯลฯ และสัมพันธ์กับ จำนวนของสินค้าที่เคลื่อนไหว เช่น สินค้าที่มีจำนวนเคลื่อนไหวในระดับพันชิ้น ถ้าจำนวนทศนิยมของ CostError น้อยเกินไป จะทำให้การปัดเศษต้นทุนเพี้ยนได้ง่าย 

ดูเพิ่มเติม: CostDigit มูลค่าต้นทุนรวม
|Default|@@""" CostShow = 0 """@@|
กำหนดระดับ user ต่ำสุดที่สามารถเห็นต้นทุนสินค้า 

| 0 |user1 (อนุญาต user ทุกคน) |
| 1 |user2 |
| 2 |user3 |
| 3 |user4 |
| 4 |user5 |
| 5 |user6 |
| 6 |user7 |
| 7 |บัญชี |
| 8 |ผู้จัดการ |
| 9 |supervisor (ไม่อนุญาต user ทุกคน)|
|Default|@@""" CusUtilOpenShare = OFF """@@|
ปกติ ในการทำงานอรรถประโยชน์ ของระบบลูกค้า เช่น เปลี่ยนชื่อ หรือ คำนวณยอดสุดท้ายใหม่ โปรแกรมจะเปิดไฟล์แบบ exclusive คือ lock ไม่ให้ผู้อื่นเข้าไปเพิ่ม, ลบ, แก้ไข ไฟล์ข้อมูลลูกค้า เพื่อป้องกันไม่ให้ประมวลผลผิดพลาด 

ดังนั้น ในระหว่างที่มีการประมวลผลอรรถประโยชน์อยู่ ผู้อื่นไม่สามารถทำงานใดๆ ที่เกี่ยวกับลูกค้าจะต้องรอจนกว่า คำสั่งอรรถประโยชน์นั้น ทำงานเสร็จก่อน 

เปลี่ยนค่านี้เป็น ON จะเปิดไฟล์แบบ share แทน มีผลทำให้ ผู้อื่นสามารถทำงานเกี่ยวกับลูกค้าได้ตามปกติ ในระหว่างที่ คำสั่งอรรถประโยชน์ทำงาน 

กรณีนี้ ผู้ใช้จะต้องรับความเสี่ยงเอง โอกาสที่การประมวลผลผิดพลาด เช่น ระหว่างที่คำนวณยอดหนี้ล่าสุดอยู่ มีการเพิ่มหนี้ของลูกค้ารายนั้นจากผู้อื่นพอดี 
Home
//{{{
window.coreWikify = wikify;
window.wikify = function(source,output,highlightRegExp,tiddler)
{
	if (source) arguments[0]=source.replace(/<br>\\\n/mg,'<br>')
	else if (source) arguments[0]=source.replace(/\\\\\n/mg,' ');
	coreWikify.apply(this,arguments);
}
//}}}
{{{
ผมไม่แน่ใจว่ามีคนถามเรื่องนี้แล้วหรือยัง คือลูกค้ามีความต้องการที่จะใช้ความสามารถ  F1 ที่ช่องหมายเหตุ 
วัตถุประสงค์เพื่อเก็บ รายละเอียดไว้เป็นกลุ่ม ๆ เหมือนกัน เมื่อมีข้อความเหมือนกัน ก็จะดึงข้อมูลรายการที่เก็บไว้มาใช้งานอีก

อีกข้อที่ต้องการ คือต้องการให้รายงานบิลซื้อ บิลขาย มีความสามารถที่จะรายงาน เลือกหัวข้อรายงาน ใช้ไวล์การ์ด (*) ในการกรุ๊ปรายงานได้
สองข้อนี้ สามารถทำได้หรือไม่ หรือต้องการโมดิฟาย จะทำได้หรือไม่ เท่าไร

ขอบคุณครับ
}}}

{{{
update 30-9-51

จัดให้
วิธีที่ 1 โปรแกรมเดิม F1 ในช่องหมายเหตุไม่ได้
อาจไม่สะดวกนัก ใช้ Copy & Paste จาก Field ข้อมูลเดิมมาใช้แทนก่อน
http://admthai.homeip.net/forum/index.php/topic,482.0.html
ข้อเสียคือ ทุกครั้งที่ออกจากโปรแกรม Copy & Paste จะหายไปด้วย

แต่รายงาน R_Po สามารถกำหนดเงื่อนไขใน .FRM ได้ดังนี้
cnd.Remark=ON,"wildcard...."
เพื่อให้โปรแกรมกรองข้อมูลส่วน หมายเหตุด้วย

วิธีที่ 2 update โปรแกรมเพิ่มเติมให้แล้วครับ
สามารถกำหนดให้ช่อง หมายเหตุ สามารถกด F1 เลือกรายการจาก list ข้อมูลที่อื่นมาใช้ ดังนี้
กำหนดใน __FSYS.INI
_PoRemarkList=ชื่อข้อมูล

สำหรับชื่อข้อมูลได้แก่
CusName ชื่อลูกค้า
VndName ชื่อเจ้าหนี้
PName ชื่อสินค้า
WkrName  ชื่อพนักงาน (ระบบเงินเดือน/ภาษี ณ ที่จ่าย)
Who ชื่อผู้เกี่ยวข้อง (แยกประเภท)
InvBrief ชื่อสรุปบิลขาย
PoBrief ชื่อสรุปบิลซื้อ
BDetail ชื่อรายการบัญชี (ธนาคาร)

ตัวอย่างเช่น
_PoRemarkList=PoBrief

หรือถ้ากำหนดให้ใช้เฉพาะบิลหมวดที่ต้องการ
_PoRemarkList.XX=PoBrief

XX คือ หมวดบิล

นอกจากนี้ยัง update จอเงื่อนไขรายงาน บิลซื้อ(ใหม่) เพิ่มช่องให้ระบุเงื่อนไข หมายเหตุ ได้

ไฟล์ที่ update ได้แก่ _Po1_.EXE, R_Po.EXE

 
}}}
To get started with this blank [[TiddlyWiki]], you'll need to modify the following tiddlers:
* [[SiteTitle]] & [[SiteSubtitle]]: The title and subtitle of the site, as shown above (after saving, they will also appear in the browser title bar)
* [[MainMenu]]: The menu (usually on the left)
* [[DefaultTiddlers]]: Contains the names of the tiddlers that you want to appear when the TiddlyWiki is opened
You'll also need to enter your username for signing your edits: <<option txtUserName>>

<<changeMode>>
* help
** [[Tags|http://tiddlywiki.org/wiki/Tags]]
** [[Markup|http://tiddlywiki.org/wiki/TiddlyWiki_Markup]]
** [[Macro|http://tiddlywiki.org/wiki/Core_Macros]]
* plugin
** [[Alias|http://tiddlywiki.abego-software.de/Beta.html#TiddlerAliasPlugin]]
** [[Micro Browser|http://www.tiddlytools.com/#MicroBrowser]]
** [[Launch Application|http://remotely-helpful.com/TiddlyWiki/LaunchApplication.html]]
** [[External Tiddler|http://www.TiddlyTools.com/#ExternalTiddlersPlugin]]
** [[Story Viewer|http://www.tiddlytools.com/#StoryViewerPlugin]]
<<tiddler "UpdateTalks">>

UpdateLogs
|<<tiddler "UpdateLogs">>|

UpdateLinks
|<<tiddler "UpdateLinks">>|
/***
|Name|ImageSizePlugin|
|Source|http://www.TiddlyTools.com/#ImageSizePlugin|
|Version|1.2.1|
|Author|Eric Shulman|
|License|http://www.TiddlyTools.com/#LegalStatements|
|~CoreVersion|2.1|
|Type|plugin|
|Description|adds support for resizing images|
This plugin adds optional syntax to scale an image to a specified width and height and/or interactively resize the image with the mouse.
!!!!!Usage
<<<
The extended image syntax is:
{{{
[img(w+,h+)[...][...]]
}}}
where ''(w,h)'' indicates the desired width and height (in CSS units, e.g., px, em, cm, in, or %). Use ''auto'' (or a blank value) for either dimension to scale that dimension proportionally (i.e., maintain the aspect ratio). You can also calculate a CSS value 'on-the-fly' by using a //javascript expression// enclosed between """{{""" and """}}""". Appending a plus sign (+) to a dimension enables interactive resizing in that dimension (by dragging the mouse inside the image). Use ~SHIFT-click to show the full-sized (un-scaled) image. Use ~CTRL-click to restore the starting size (either scaled or full-sized).
<<<
!!!!!Examples
<<<
{{{
[img(100px+,75px+)[images/meow2.jpg]]
}}}
[img(100px+,75px+)[images/meow2.jpg]]
{{{
[<img(34%+,+)[images/meow.gif]]
[<img(21% ,+)[images/meow.gif]]
[<img(13%+, )[images/meow.gif]]
[<img( 8%+, )[images/meow.gif]]
[<img( 5% , )[images/meow.gif]]
[<img( 3% , )[images/meow.gif]]
[<img( 2% , )[images/meow.gif]]
[img(  1%+,+)[images/meow.gif]]
}}}
[<img(34%+,+)[images/meow.gif]]
[<img(21% ,+)[images/meow.gif]]
[<img(13%+, )[images/meow.gif]]
[<img( 8%+, )[images/meow.gif]]
[<img( 5% , )[images/meow.gif]]
[<img( 3% , )[images/meow.gif]]
[<img( 2% , )[images/meow.gif]]
[img(  1%+,+)[images/meow.gif]]
{{tagClear{
}}}
<<<
!!!!!Revisions
<<<
2009.02.24 [1.2.1] cleanup width/height regexp, use '+' suffix for resizing
2009.02.22 [1.2.0] added stretchable images
2008.01.19 [1.1.0] added evaluated width/height values
2008.01.18 [1.0.1] regexp for "(width,height)" now passes all CSS values to browser for validation
2008.01.17 [1.0.0] initial release
<<<
!!!!!Code
***/
//{{{
version.extensions.ImageSizePlugin= {major: 1, minor: 2, revision: 1, date: new Date(2009,2,24)};
//}}}
//{{{
var f=config.formatters[config.formatters.findByField("name","image")];
f.match="\\[[<>]?[Ii][Mm][Gg](?:\\([^,]*,[^\\)]*\\))?\\[";
f.lookaheadRegExp=/\[([<]?)(>?)[Ii][Mm][Gg](?:\(([^,]*),([^\)]*)\))?\[(?:([^\|\]]+)\|)?([^\[\]\|]+)\](?:\[([^\]]*)\])?\]/mg;
f.handler=function(w) {
	this.lookaheadRegExp.lastIndex = w.matchStart;
	var lookaheadMatch = this.lookaheadRegExp.exec(w.source)
	if(lookaheadMatch && lookaheadMatch.index == w.matchStart) {
		var floatLeft=lookaheadMatch[1];
		var floatRight=lookaheadMatch[2];
		var width=lookaheadMatch[3];
		var height=lookaheadMatch[4];
		var tooltip=lookaheadMatch[5];
		var src=lookaheadMatch[6];
		var link=lookaheadMatch[7];

		// Simple bracketted link
		var e = w.output;
		if(link) { // LINKED IMAGE
			if (config.formatterHelpers.isExternalLink(link)) {
				if (config.macros.attach && config.macros.attach.isAttachment(link)) {
					// see [[AttachFilePluginFormatters]]
					e = createExternalLink(w.output,link);
					e.href=config.macros.attach.getAttachment(link);
					e.title = config.macros.attach.linkTooltip + link;
				} else
					e = createExternalLink(w.output,link);
			} else 
				e = createTiddlyLink(w.output,link,false,null,w.isStatic);
			addClass(e,"imageLink");
		}

		var img = createTiddlyElement(e,"img");
		if(floatLeft) img.align="left"; else if(floatRight) img.align="right";
		// modify by Sathit, March, 2010
		if (!(width||height)){width="80%+";height="auto"}
		if(width||height) {
			var x=width.trim(); var y=height.trim();
			var stretchW=(x.substr(x.length-1,1)=='+'); if (stretchW) x=x.substr(0,x.length-1);
			var stretchH=(y.substr(y.length-1,1)=='+'); if (stretchH) y=y.substr(0,y.length-1);
			if (x.substr(0,2)=="{{")
				{ try{x=eval(x.substr(2,x.length-4))} catch(e){displayMessage(e.description||e.toString())} }
			if (y.substr(0,2)=="{{")
				{ try{y=eval(y.substr(2,y.length-4))} catch(e){displayMessage(e.description||e.toString())} }
			img.style.width=x.trim(); img.style.height=y.trim();
			config.formatterHelpers.addStretchHandlers(img,stretchW,stretchH);
		}
		if(tooltip) img.title = tooltip;

		// GET IMAGE SOURCE
		if (config.macros.attach && config.macros.attach.isAttachment(src))
			src=config.macros.attach.getAttachment(src); // see [[AttachFilePluginFormatters]]
		else if (config.formatterHelpers.resolvePath) { // see [[ImagePathPlugin]]
			if (config.browser.isIE || config.browser.isSafari) {
				img.onerror=(function(){
					this.src=config.formatterHelpers.resolvePath(this.src,false);
					return false;
				});
			} else
				src=config.formatterHelpers.resolvePath(src,true);
		}
		img.src=src;
		w.nextMatch = this.lookaheadRegExp.lastIndex;
	}
}

config.formatterHelpers.addStretchHandlers=function(e,stretchW,stretchH) {
	e.title=((stretchW||stretchH)?'DRAG=stretch/shrink, ':'')
		+'SHIFT-CLICK=show full size, CTRL-CLICK=restore initial size';
	e.statusMsg='width=%0, height=%1';
	e.style.cursor='move';
	e.originalW=e.style.width;
	e.originalH=e.style.height;
	e.minW=Math.max(e.offsetWidth/20,10);
	e.minH=Math.max(e.offsetHeight/20,10);
	e.stretchW=stretchW;
	e.stretchH=stretchH;
	e.onmousedown=function(ev) { var ev=ev||window.event;
		this.sizing=true;
		this.startX=!config.browser.isIE?ev.pageX:(ev.clientX+findScrollX());
		this.startY=!config.browser.isIE?ev.pageY:(ev.clientY+findScrollY());
		this.startW=this.offsetWidth;
		this.startH=this.offsetHeight;
		return false;
	};
	e.onmousemove=function(ev) { var ev=ev||window.event;
		if (this.sizing) {
			var s=this.style;
			var currX=!config.browser.isIE?ev.pageX:(ev.clientX+findScrollX());
			var currY=!config.browser.isIE?ev.pageY:(ev.clientY+findScrollY());
			var newW=(currX-this.offsetLeft)/(this.startX-this.offsetLeft)*this.startW;
			var newH=(currY-this.offsetTop )/(this.startY-this.offsetTop )*this.startH;
			if (this.stretchW) s.width =Math.floor(Math.max(newW,this.minW))+'px';
			if (this.stretchH) s.height=Math.floor(Math.max(newH,this.minH))+'px';
			clearMessage(); displayMessage(this.statusMsg.format([s.width,s.height]));
		}
		return false;
	};
	e.onmouseup=function(ev) { var ev=ev||window.event;
		if (ev.shiftKey) { this.style.width=this.style.height=''; }
		if (ev.ctrlKey)  { this.style.width=this.originalW; this.style.height=this.originalH; }
		this.sizing=false;
		clearMessage();
		return false;
	};
	e.onmouseout=function(ev) { var ev=ev||window.event;
		this.sizing=false;
		clearMessage();
		return false;
	};
}
//}}}
รอเรียบเรียง
MOD-CDS-2007-09
MOD-CDS-2007-10
MOD-CDS-2007-11
MOD-CDS-2007-12
<<list filter "[tag[__FSys.INI]][sort[-modified]]">>
/%<<list filter "[tag[__FSys.INI]]">>%/
ดูเพิ่มเติม <<matchTags popup "label:Draft" " __FSys.INI AND Draft>>
<<matchTags inline "|[[%0]]<br>%6|%5|" "\n" __FSys.INI AND NOT Draft>>
<<list filter "[tag[MOD]][sort[-modified]]">>
/%<<list filter "[tag[MOD]]">>%/
ดูเพิ่มเติม <<matchTags popup "label:Draft" " MOD AND Draft>>
<<matchTags inline "|[[%0]]<br>%6|" "\n" MOD AND NOT Draft>>
<<timeline "modified" "20">>
ใช้กำหนดชื่อชุดข้อมูล ของชื่อสินค้า ที่เป็นฐานข้อมูลหลัก เป็นส่วนหนึ่งของ ~MOD-SPC มีผลเฉพาะกับไฟล์ ~StkUtilX ส่วนอรรถประโยชน์เปลี่ยนชื่อสินค้า 

กรณีที่ชื่อสินค้าในชุดข้อมูลหลักถูกเปลี่ยนไปแล้ว แต่ชื่อสินค้าในชุดข้อมูลสาขา ยังไม่ได้เปลี่ยนชื่อ ช่วยให้สามารถกดหาชื่อสินค้าที่เปลี่ยนไปแล้ว จากชุดข้อมูลหลักได้

@@"""
MDir.Stock=ชื่อชุดข้อมูลของ HO
"""@@

ในช่อง เปลี่ยนชื่อเป็น สามารถกด CTRL+F1 เพื่อเลือกชื่อสินค้า จากชุดข้อมูล ของ Head Office
{{{
28/9/2550

note: case พิเศษ
1. รวมใบจ่ายหลายใบ จ่ายด้วยเช็คใบเดียว
2. ชื่อบนเช็ค ไม่ตรงกับชื่อหักภาษี
3. เงื่อนไข ตามความสามารถพิเศษของ TFB
- Pickup Location
- Pickup Document
- Advice Mode
- Fax No
}}}
{{{
เงื่อนไขที่สำคัญ ก่อนใช้ CHEQUE DIRECT SYSTEM (CDS)

ใบจ่าย กรณีอ้างมาจากบิลซื้อ
ในหมวดใบจ่าย จะต้องมี ~V
เพื่อให้แยกบรรทัด VAT ออกจากค่าสินค้า

กรณี บันทึกค่าใช้จ่าย โดยไม่มีบิลซื้อ (เช่น เงินสดย่อย)
จะต้องแยกยอด VAT ออกมาเป็นบรรทัดต่างหาก 
โดยระบุข้อความ บรรทัดว่า

VAT หรือ ภาษีมูลค่าเพิ่ม

สามารถใช้ความสามารถของ inline calculate เช่น

**VAT [+7%]


ขณะเดียวกัน ถ้าใช้ Genesis จะต้องตรวจสอบด้วยว่าต้องปรับ ALI
ให้รองรับข้อความบรรทัด VAT หรือยัง
}}}
{{{
4/10/50

:::: คำสั่งพิเศษ

จะถูกตรวจหา และจัดเก็บในระหว่างการเพิ่มข้อมูล CHQ DIRECT

1. อยู่ในบรรทัดรายการ ของใบจ่าย (หรือ ได้มาจากชื่อสรุปบิลซื้อ)

2. อยู่ในข้อมูลหมวดใบจ่าย (บรรทัดชื่อ, บรรทัดหัว, บรรทัดท้าย)

3. อยู่ในข้อมูลชื่อแทนเจ้าหนี้ (ต้อง MOD เพิ่มเติม)

4. อยู่ใน __FSYS.INI

:::: คำสั่งเกี่ยวกับ วิธีการจ่ายเช็ค
{//BANK} จ่ายที่ counter
{//MAIL} ส่งทางไปรษณีย์
{//RET} คืนให้บริษัท ใช้กรณี ทำเช็ค จ่ายสรรพากร, จ่ายค่าโทรศัพท์ ฯลฯ

กรณีจ่ายที่ counter จะต้องมีการกำหนด คำสั่งต่อไปนี้ด้วย 
(อย่างน้อยควรกำหนดไว้ใน __FSYS.INI เพื่อเป็นค่า default)

{//ATnn} รหัสสาขา 

{//+RTI} กำหนดให้เก็บหลักฐาน เป็น combination R=ใบเสร็จ T=ใบกำกับภาษี I=บัตรประชาชน เช่น 
{//+} แปลว่า ไม่ต้องเก็บเอกสารใดๆ 
{//+R} เก็บเฉพาะใบเสร็จ
{//+RI} เก็บใบเสร็จ และบัตรประชาชน (กรณีจ่ายเช็คให้ บุคคลธรรมดา)
{//+RT} เก็บใบเสร็จ และใบกำกับภาษี (กรณีเจ้าหนี้ ส่ง Fax มาวางบิล หรือ ชำระค่าบริการ)

{//FAX} หรือ {//FAXnnnnnnnnn} ส่งแฟกซ์  แจ้งเตือนให้รับเช็ค ถ้าไม่มีหมายเลขโปรแกรมจะเอาเลข FAX ที่อยู่ในข้อมูลเจ้าหนี้ 

ได้คำสั่งพิเศษมาแล้ว โปรแกรมต้องเอามาตีความเพื่อเป็น code ของ KBANK อีกทีหนึ่ง
   > delivery: "CC" จ่ายเช็คที่เคาน์เตอร ์ ไม่เก็บหลักฐาน , "CR" จ่ายเช็คที่เคาน์เตอร์ เก็บหลักฐาน , "MR" ส่งไปรษณีย์ , "RC" ส่งคืนบริษัท
   > pickuploc: "RET" ใช้คู่กับ RC , "M" ใช้คู่กับ MR , or "01" - "07" สำหรับ CC หรือ CR
   > pickupdoc: "NONE" ไม่มี หรือ "B" ใบรับวางบิล , "DISR" ใบเสร็จ , "DIST" ใบกำกับภาษี , "R" ใบเสร็จ , "T" ใบกำกับภาษี , "INV" ใบแจ้งหนี้ , "ID" สำเนาบัตรประชาชน
}}}

{{{
13/10/50

1.วันที่สั่งจ่ายบนเช็ค
ใน Batch ของ CDS บังคับให้ต้องมีวันที่บนเช็คเดียวกัน เท่านั้น
ดังนั้น จึงควรระวัง กรณีที่มีเช็คจ่ายล่วงหน้า 
จะต้องแยก Batch เพื่อ ส่งไปทำเช็ค ต่างหาก

กรณีใช้ อรรถประโยชน์ คัดข้อมูลจากใบจ่าย

> ถ้าใบจ่ายนั้น มีระบุ วันทีนัด/ชำระ 
โปรแกรมจะใช้วันที่นัด/ชำระ เป็นวันที่สั่งจ่ายบนเช็ค

> ถ้าใบจ่ายนั้น ไม่มีระบุ วันที่นัด/ชำระ
โปรแกรมจะ ใช้วันที่ส่ง CDS เป็นวันที่สั่งจ่ายบนเช็คด้วย

2. ชื่อสั่งจ่ายบนเช็ค

> สามารถระบุชื่อสั่งจ่าย แตกต่างจาก ชื่อเจ้าหนี้ ในรายการ CDS ได้

> ถ้าไม่ระบุ โปรแกรมจะใช้ชื่อเจ้าหนี้ หรือ ชื่อแทนของเจ้าหนี้ (ถ้ามีระบุชื่อแทน)

> กรณีใช้ อรรถประโยชน์ คัดข้อมูลจากใบจ่าย
โปรแกรมจะใช้ชื่อสั่งจ่าย ตามข้อมูลรายการ CDS ที่เคยจ่ายไปแล้ว ที่วันที่ล่าสุด

3. การรวมยอดเพื่อจ่ายเช็คใบเดียวอัตโนมัติ
กรณี มีรายการ CDS ที่สั่งจ่ายเจ้าหนี้รายเดียวกันมากกว่า 1 รายการ ใน Batch เดียวกัน
ถ้าในรายการ CDS นั้น ไม่ได้ระบุให้แยกเช็ค
และในรายการ CDS นั้น ไม่มีรายการ หักภาษี ณ ที่จ่าย ที่จะต้องพิมพ์แนบคู่ไปกับเช็ค
โปรแกรมจะรวมยอดกับรายการ CDS อื่น ที่มีชื่อเจ้าหนี้ และชื่อสั่งจ่ายเดียวกัน
เพื่อให้ทำเช็คจ่ายเพียงใบเดียว
}}}

{{{
21/10/50

สรุปคำสั่ง __FSys.INI

CDS.AccNo=nnnnnnnn (เลขบัญชี 10 ตัว)

CDS.WTaxBranch=nn (เลขสาขาที่ออกใบหักภาษี ของบริษัท)

CDS.Charge=Y หรือ N (Y=เก็บค่าธรรมเนียมจากผู้รับเงิน N=เก็บค่าธรรมเนียมจากผู้สั่งจ่าย)

CDS.XCmd=คำสั่งพิเศษ 

CDS.UseWTax=ON หรือ OFF (เลือกว่าจะให้พิมพ์ใบหักภาษี ควบกับเช็คหรือไม่)


สรุปเกี่ยวกับคำสั่งพิเศษ

1. คำสั่งกำหนดวิธีการจ่ายเช็ค
{//RET} รับเช็คกลับมาที่บริษัท
{//MAIL} ส่ง mail ไปให้เจ้าหนี้
{//BANK} ให้เจ้าหนี้ไปรับเช็คที่ธนาคาร

2. คำสั่งกำหนดสาขาที่ให้เจ้าหนี้ไปรับเช็ค เฉพาะสำหรับ {//BANK}
{//ATxxx} ใช้เลขสาขา หรือรหัสสาขา ตามที่กำหนดจาก BANK
มี 01 - 07 และ EXP1 เช่น {//AT02}

3. คำสั่งกำหนดเอกสารที่ทาง BANK ต้องรวบรวมให้ เฉพาะสำหรับ {//BANK}
{//+xxx yyy} ให้รหัสเอกสาร ตามที่กำหนดจาก BANK
เช่น {//+DISR T}

4. คำสั่งกำหนดให้ส่ง FAX แจ้งเตือนเจ้าหนี้
{//FAX} หรือ {//FAX@} หรือ {//FAXnnnnnnnnnnn}
กรณี {//FAX} แปลว่าไม่ส่งแฟกซ์
กรณี {//FAX@} โปรแกรมจะใช้เลข Fax จากฐานข้อมูลเจ้าหนี้
}}}

{{{
คำสั่งพิเศษ ใส่ไว้ที่ไหนได้บ้าง

1. ไว้ในบรรทัดใดก็ได้ ของใบจ่าย
โปรแกรมจะ scan หาคำสั่งพิเศษ (มีทั้งหมด 4 กลุ่มคำสั่ง)
ไล่จากบรรทัดที่ 1 ลงไปเรื่อยๆ
ถ้าเจอในบรรทัดไหนก่อน ก็จะใช้คำสั่งนั้น เช่น
ถ้า ในบรรทัดที่ 1 มีคำสั่ง {//RET}
และในบรรทัดที่ 2 มีคำสั่ง {//BANK}
โปรแกรมจะเอาคำสั่ง {//RET} ไปใช้

2. ไว้ในข้อมูลหมวดใบจ่าย
ในชื่อใบจ่าย, หัวบิล1-3, ท้ายบิล 1-3
คำสั่งกลุ่มไหน ที่หลุดรอดไม่มีกำหนดไว้ตามข้างบน
โปรแกรมก็จะหาต่อใน ข้อมูลหมวดใบจ่าย

คำสั่งที่ ตรวจจับมาได้ตามข้อ 1 และ 2
จะนำมาแสดงไว้ใน field "คำสั่งCDS" ของข้อมูล CDS Transaction
ดังนั้น ผู้ใช้ อาจเปลี่ยนแปลง หรือเพิ่มเติมคำสั่งก่อน save ข้อมูล CDS Transaction อีกทีหนึ่งได้

ส่วนขั้นตอนการสร้าง TEXT File จากหัวข้อ อรรถประโยชน์
โปรแกรมจะตรวจสอบคำสั่งพิเศษเพิ่มเติมดังหนี้

1. ดูจาก Field "คำสั่งCDS"  (ซึ่งได้มาจากการประมวลผล ข้อมูลใบจ่ายและหมวดใบจ่ายตามข้างต้น)
ใช้ประโยชน์ กรณีสั่งทำเช็คเร่งด่วน (EXPRESS)

2. ดูจาก Field ชื่อสั่งจ่าย ในข้อมูล CDS Transaction
ใช้ประโยชน์กรณี เจ้าหนี้ต้องการเงื่อนไขแตกต่างจากทั่วไป เช่น ให้ส่งแฟกซ์แจ้ง หรือรับเช็คที่สาขาอื่น หรือต้องการให้ส่ง MAIL

3. ดูจาก Field คำสั่ง CDS ในข้อมูล เจ้าหนี้เพิ่มเติม
ใช้ประโยชน์ คล้ายกรณีข้อ 2

4. ดูจาก __FSys.INI
ใช้ประโยชน์ กำหนดเป็นค่า default ของระบบ
สำหรับกรณีของ CINNAMON อาจต้องตั้งเป็น
{//BANK}{//AT01}{//+R}

5. สุดท้าย ใช้ค่าที่ตั้งไว้ในโปรแกรม ได้แก่
{//RET} กำหนดให้ บริษัทรับเช็คกลับมา

*** กรณี หมายเลข Fax โดยเฉพาะเมื่อใช้ จากฐานข้อมูลเจ้าหนี้
โปรแกรม มีระบบ ตัดตัวสัญญลักษณ์ ที่ไม่เกี่ยวข้องออก อัตโนมัติ
เช่น ( ) - ดังนั้น ถึงแม้ใน ฐานข้อมูล มีการป้อนไว้เป็น (02)789-0000
โปรแกรมก็สามารถแปลงเป็น 027890000 ให้
}}}

{{{
22/10/50

เพิ่มคำสั่ง __FSYS.INI

CDS.UseMD5 = ON หรือ OFF (default ON)

คำสั่งให้คำนวณ MD5 ใส่ต่อท้าย Text File
}}}
{{{
อ้างจาก: "adm"
อ้างจาก: "jojosati"
เงื่อนไขที่สำคัญ ก่อนใช้ CHEQUE DIRECT SYSTEM (CDS)

ใบจ่าย กรณีอ้างมาจากบิลซื้อ
ในหมวดใบจ่าย จะต้องมี ~V
เพื่อให้แยกบรรทัด VAT ออกจากค่าสินค้า




ไม่เข้าใจ  ทดลองกำหนดหมวดใบจ่ายมี ~V แล้วดึงบิลซื้อมา
ไม่แยก Vat ในจอใบจ่าย ยังคงมายอดเต็ม ๆ


ขออภัยครับ ต้องใช้ ~FCV
}}}

{{{
19/11/50

แก้ไขข้อผิดพลาดของ Text File ที่จะ upload ไปให้ bank

1. BatchRef ใช้ format CHyymmddnn ตามที่ bank แนะนำ (วันที่ใช้วันที่ส่งข้อมูลให้ทำเช็ค)
2. วันที่ Effective กับ Pick Date ใช้วันที่เดียวกันคือ วันที่บนเช็ค
3. แก้ไขข้อผิดพลาดของ field ตัวเลข ทั้งหมด 
4. แก้ไข MD5 ที่ปิดท้ายไฟล์ แปลงเลขฐาน 16 ให้ใช้ตัวภาษาอังกฤษตัวเล็ก
5. คำสั่ง __FSYS.INI CDS.WTaxBranch เดิมตีความเป็นตัวเลข 
เช่น ถ้าระบุว่า CDS.WTaxBranch=1 
เมื่อแปลงใส่ text file จะเติมเลข 0 ข้างหน้าเป็น 00001 (ถ้าไม่ได้กำหนดจะเป็น 00000) 
แก้ไขเป็น รับค่า string ถ้าไม่ระบุ จะเป็น blank 
ถ้าต้องการให้มี 0 ข้างหน้าก็ต้องระบุเอง เช่น CDS.WTaxBranch=0001
}}}
{{{
5/12/2550 (วันพ่อ)
แก้ไขเพิ่มเติม 8/12/2550

เพิ่ม สำหรับ Direct Credit / Media Direct

เรื่องกำหนดเลขบัญชี
{//ACCรหัสบัญชี10ตัว} ใช้กับ direct credit โอนเงินธนาคารเดียวกัน
{//ACCรหัสบัญชี10ตัว/รหัสธนาคาร} ใช้กับ Media Direct ใช้ 3 ตัวหน้าของรหัสบัญชีเป็น เลขสาขา หรือ Direct Credit กรณีรหัสธนาคารคือ 004 = กสิกร
{//ACCรหัสบัญชีอื่นๆ ที่ไม่มาตรฐาน/รหัสธนาคาร/รหัสสาขา} ใช้กับ Media Direct ดูอ้างอิงจากไฟล์ bankcode ข้างบน

ซึ่งข้อมูล ACC นี้ ควรแนบอยู่ในข้อมูลชื่อสั่งจ่ายของเจ้าหนี้
เช่น บริษัท กขค จำกัด {//ACC011-2-12345-0/002}

เพิ่มคำสั่งเกี่ยวกับวิธีจ่ายเงิน
{//TR} จ่ายโดยการโอนเงิน ผ่าน MD หรือ DC
{//CQ} จ่ายโดยการโอนเงิน ผ่าน Cheque (CD)

และใน __FSYS.INI เพิ่มคำสั่ง
CDS.CompanyID = รหัสบริษัท (ใช้กับ DC) ทาง KBANK ต้องให้มา
CDS.CompanyName=ชื่อบัญชีของบริษัท (ใช้กับ DC)
}}}

{{{
8/12/2550

คำแนะนำ
__FSYS.INI ต้องกำหนดอย่างน้อย เพื่อเป็นค่า default ดังนี้

CDS.AccNo=เลขบัญชีของบริษัท
CDS.CompanyID=รหัสบริษัท (KBank ออกให้)
CDS.CompanyName=ชื่อบัญชีของบริษัท
CDS.XCmd={//TR}

ส่วนเลขบัญชี ควรใส่ไว้ใน ชื่อสั่งจ่าย ของ CDSTrn
{//ACCxxxxxxxxxxxxx}

ข้อจำกัด
Direct Credit ไม่สามารถระบุชื่อหักภาษี แตกต่างจากชื่อบัญชีปลายทาง
}}}

{{{
11/12/2550

แก้ไข
BatchRef ใช้ format 
TRyymmddnn สำหรับโอน KBank ด้วยกัน (Direct Credit)
TXyymmddnn สำหรับโอนต่าง ธนาคาร (Media Clearing)

note เพิ่มเติม
เกี่ยวกับ MD5 ที่ปะท้าย text file
ไม่แน่ใจว่า ทาง Direct Credit กับ Media Clearing จะ support เหมือน cheque direct รึเปล่านะครับ
ลองตรวจสอบด้วย
}}}

{{{
19/12/50

-Import files สำเร็จแล้ว

สอบถามแบงค์เพิ่มเติมได้ความว่า
-ชื่อผู้รับเงิน ไม่สนใจ ใส่แค่พอรู้เรื่องพอ, เน้นเลขที่บัญชีให้ครับ
-Upload ถ้าสำเร็จต้องเป็นตัวดำ, ถ้าสีแดงน่าจะเลขที่บัญชีผิด
-Bank ตรวจเลขบัญชีให้เฉพาะที่เป็น ธนาคารกสิกรเท่านั้น, ถ้าต่าง Bank ไม่ตรวจ
รู้ผลเมื่อโอน
}}}

{{{
19/12/50

แก้ BUG เกี่ยวกับ การดึงชื่อสั่งจ่าย อัตโนมัติ

ในระหว่างป้อนข้อมูล CDSTrn

ในการสั่งดึงข้อมูลจากใบจ่าย U_CDS2

21/12/50

แก้ไข wording จาก CHEQUE DIRECT เป็น KCash
}}}
[[Home]]
----
<<slider chkINICmdSlider TabINICmd "INI Commands">>/%<<tag "__FSys.INI" "INI Commands">>%/
<<slider chkMODSlider TabMOD "MOD Jobs">>/%<<tag "MOD" "MOD Jobs">>%/
<<slider chkModulesSlider TabModules "Modules">>
<<tag "Draft">>
----
<<slider chkListsSlider TabLists "Lists..">>
<<slider chkHelpsSlider TabHelps "Author..">>
::Default
/***
|Name|MatchTagsPlugin|
|Source|http://www.TiddlyTools.com/#MatchTagsPlugin|
|Documentation|http://www.TiddlyTools.com/#MatchTagsPluginInfo|
|Version|2.0.3|
|Author|Eric Shulman|
|License|http://www.TiddlyTools.com/#LegalStatements|
|~CoreVersion|2.1|
|Type|plugin|
|Description|'tag matching' with full boolean expressions (AND, OR, NOT, and nested parentheses)|
!!!!!Documentation
> see [[MatchTagsPluginInfo]]
!!!!!Revisions
<<<
2010.03.02 2.0.3 added %6 format (tags)
| please see [[MatchTagsPluginInfo]] for additional revision details |
2008.02.28 1.0.0 initial release
<<<
!!!!!Code
***/
//{{{
version.extensions.MatchTagsPlugin= {major: 2, minor: 0, revision: 3, date: new Date(2010,3,2)};

// store.getMatchingTiddlers() processes boolean expressions for tag matching
//    sortfield (optional) sets sort order for tiddlers - default=title
//    tiddlers (optional) use alternative set of tiddlers (instead of current store)
TiddlyWiki.prototype.getMatchingTiddlers = function(tagexpr,sortfield,tiddlers) {

	var debug=config.options.chkDebug; // abbreviation
	var cmm=config.macros.matchTags; // abbreviation
	var r=[]; // results are an array of tiddlers
	var tids=tiddlers||store.getTiddlers(sortfield||"title");
	if (tiddlers && sortfield) store.sortTiddlers(tids,sortfield);
	if (debug) displayMessage(cmm.msg1.format([tids.length]));

	// try simple lookup to quickly find single tags or tags that
	// contain boolean operators as literals, e.g. "foo and bar"
	for (var t=0; t<tids.length; t++)
		if (tids[t].isTagged(tagexpr)) r.pushUnique(tids[t]);
	if (r.length) {
		if (debug) displayMessage(cmm.msg4.format([r.length,tagexpr]));
		return r;
	}
	
	// convert expression into javascript code with regexp tests,
	// so that "tag1 AND ( tag2 OR NOT tag3 )" becomes
	// "/\~tag1\~/.test(...) && ( /\~tag2\~/.test(...) || ! /\~tag3\~/.test(...) )"

	// normalize whitespace, tokenize operators, delimit with "~"
	var c=tagexpr.trim(); // remove leading/trailing spaces
	c = c.replace(/\s+/ig," "); // reduce multiple spaces to single spaces
	c = c.replace(/\(\s?/ig,"~(~"); // open parens
	c = c.replace(/\s?\)/ig,"~)~"); // close parens
	c = c.replace(/(\s|~)?&&(\s|~)?/ig,"~&&~"); // &&
	c = c.replace(/(\s|~)AND(\s|~)/ig,"~&&~"); // AND
	c = c.replace(/(\s|~)?\|\|(\s|~)?/ig,"~||~"); // ||
	c = c.replace(/(\s|~)OR(\s|~)/ig,"~||~"); // OR
	c = c.replace(/(\s|~)?!(\s|~)?/ig,"~!~"); // !
	c = c.replace(/(^|~|\s)NOT(\s|~)/ig,"~!~"); // NOT
	c = c.replace(/(^|~|\s)NOT~\(/ig,"~!~("); // NOT(
	// change tag terms to regexp tests
	var terms=c.split("~"); for (var i=0; i<terms.length; i++) { var t=terms[i];
		if (/(&&)|(\|\|)|[!\(\)]/.test(t) || t=="") continue; // skip operators/parens/spaces
		if (t==config.macros.matchTags.untaggedKeyword)
			terms[i]="tiddlertags=='~~'"; // 'untagged' tiddlers
		else
			terms[i]="/\\~"+t+"\\~/.test(tiddlertags)";
	}
	c=terms.join(" ");
	if (debug) { displayMessage(cmm.msg2.format([tagexpr])); displayMessage(cmm.msg3.format([c])); }

	// scan tiddlers for matches
	for (var t=0; t<tids.length; t++) {
	 	// assemble tags from tiddler into string "~tag1~tag2~tag3~"
		var tiddlertags = "~"+tids[t].tags.join("~")+"~";
		try { if(eval(c)) r.push(tids[t]); } // test tags
		catch(e) { // error in test
			displayMessage(cmm.msg2.format([tagexpr]));
			displayMessage(cmm.msg3.format([c]));
			displayMessage(e.toString());
			break; // skip remaining tiddlers
		}
	}
	if (debug) displayMessage(cmm.msg4.format([r.length,tagexpr]));
	return r;
}
//}}}
//{{{
config.macros.matchTags = {
	msg1: "scanning %0 input tiddlers",
	msg2: "looking for '%0'",
	msg3: "using expression: '%0'",
	msg4: "found %0 tiddlers matching '%1'",
	noMatch: "no matching tiddlers",
	untaggedKeyword: "-",
	untaggedLabel: "no tags",
	untaggedPrompt: "show tiddlers with no tags",
	defTiddler: "MatchingTiddlers",
	defTags: "",
	defFormat: "[[%0]]",
	defSeparator: "\n",
	reportHeading: "Found %0 tiddlers tagged with: '{{{%1}}}'\n----\n",
	handler: function(place,macroName,params,wikifier,paramString,tiddler) {
		var mode=params[0]?params[0].toLowerCase():'';
		if (mode=="inline")
			params.shift();
		if (mode=="report" || mode=="panel") {
			params.shift();
			var target=params.shift()||this.defTiddler;
		}
		if (mode=="popup") {
			params.shift();
			if (params[0]&&params[0].substr(0,6)=="label:") var label=params.shift().substr(6);
			if (params[0]&&params[0].substr(0,7)=="prompt:") var prompt=params.shift().substr(7);
		} else {
			var fmt=(params.shift()||this.defFormat).unescapeLineBreaks();
			var sep=(params.shift()||this.defSeparator).unescapeLineBreaks();
		}
		var sortBy="+title";
		if (params[0]&&params[0].substr(0,5)=="sort:") sortBy=params.shift().substr(5);
		var expr = params.join(" ");
		if (mode!="panel" && (!expr||!expr.trim().length)) return;
		if (expr==this.untaggedKeyword)
			{ var label=this.untaggedLabel; var prompt=this.untaggedPrompt };
		switch (mode) {
			case "popup": this.createPopup(place,label,expr,prompt,sortBy); break;
			case "panel": this.createPanel(place,expr,fmt,sep,sortBy,target); break;
			case "report": this.createReport(target,this.defTags,expr,fmt,sep,sortBy); break;
			case "inline": default: this.createInline(place,expr,fmt,sep,sortBy); break;
		}
	},
	formatList: function(tids,fmt,sep) {
		var out=[];
		for (var i=0; i<tids.length; i++) { var t=tids[i];
			var title=t.title;
			var who=t.modifier;
			var when=t.modified.toLocaleString();
			var text=t.text;
			var first=t.text.split("\n")[0];
			var desc=store.getTiddlerSlice(t.title,"description");
			desc=desc||store.getTiddlerSlice(t.title,"Description");
			desc=desc||store.getTiddlerText(t.title+"##description");
			desc=desc||store.getTiddlerText(t.title+"##Description");
			// modify by Sathit, March, 2010
			if (!desc) desc = store.getTiddlerText("MatchTagsDescription") ;
                        if (desc && (desc.substr(0,2)=="::" || desc.substr(0,2)=="##")) desc = store.getTiddlerText(t.title+desc);
			//var tags=t.tags.length?'[['+t.tags.join(']] [[')+']]':'';
			var tags=t.tags.length?'<<tag "'+t.tags.join('">><<tag "')+'">>':'';
			out.push(fmt.format([title,who,when,text,first,desc,tags]));
		}
		return out.join(sep);
	},
	createInline: function(place,expr,fmt,sep,sortBy) {
		wikify(this.formatList(store.sortTiddlers(store.getMatchingTiddlers(expr),sortBy),fmt,sep),place);
	},
	createPopup: function(place,label,expr,prompt,sortBy) {
		var btn=createTiddlyButton(place,
			(label||expr).format([expr]),
			(prompt||config.views.wikified.tag.tooltip).format([expr]),
			function(ev){ return config.macros.matchTags.showPopup(this,ev||window.event); });
		btn.setAttribute("sortBy",sortBy);
		btn.setAttribute("expr",expr);
	},
	showPopup: function(here,ev) {
		var p=Popup.create(here); if (!p) return false;
		var tids=store.getMatchingTiddlers(here.getAttribute("expr"));
		store.sortTiddlers(tids,here.getAttribute("sortBy"));
		var list=[]; for (var t=0; t<tids.length; t++) list.push(tids[t].title);
		if (!list.length) createTiddlyText(p,this.noMatch);
		else {
			var b=createTiddlyButton(createTiddlyElement(p,"li"),
				config.views.wikified.tag.openAllText,
				config.views.wikified.tag.openAllTooltip,
				function() {
					var list=this.getAttribute("list").readBracketedList();
					story.displayTiddlers(null,tids);
				});
			b.setAttribute("list","[["+list.join("]] [[")+"]]");
			createTiddlyElement(p,"hr");
		}
		var out=this.formatList(tids," &nbsp;[[%0]]&nbsp; ","\n"); wikify(out,p);
		Popup.show();
		ev.cancelBubble=true;
		if(ev.stopPropagation) ev.stopPropagation();
		return false;
	},
	createReport: function(target,tags,expr,fmt,sep,sortBy) {
		var tids=store.sortTiddlers(store.getMatchingTiddlers(expr),sortBy);
		if (!tids.length) { displayMessage('no matches for: '+expr); return false; }
		var msg=config.messages.overwriteWarning.format([target]);
		if (store.tiddlerExists(target) && !confirm(msg)) return false;
		var out=this.reportHeading.format([tids.length,expr])
		out+=this.formatList(tids,fmt,sep);
		store.saveTiddler(target,target,out,config.options.txtUserName,new Date(),tags,{});
		story.closeTiddler(target); story.displayTiddler(null,target);
	},
	createPanel: function(place,expr,fmt,sep,sortBy,tid) {
		var s=createTiddlyElement(place,"span"); s.innerHTML=store.getTiddlerText("MatchTagsPlugin##html");
		var f=s.getElementsByTagName("form")[0];
		f.expr.value=expr; f.fmt.value=fmt; f.sep.value=sep.escapeLineBreaks();
		f.tid.value=tid; f.tags.value=this.defTags;
	}
};
//}}}
/***
//{{{
!html
<form style='display:inline;white-space:nowrap'>
<input type='text'    name='expr' style='width:50%' title='tag expression'><!--
--><input type='text'    name='fmt'  style='width:10%' title='list item format'><!--
--><input type='text'    name='sep'  style='width:5%'  title='list item separator'><!--
--><input type='text'    name='tid'  style='width:12%' title='target tiddler title'><!--
--><input type='text'    name='tags' style='width:10%' title='target tiddler tags'><!--
--><input type='button'  name='go'   style='width:8%'  value='go' onclick="
	var expr=this.form.expr.value;
	if (!expr.length) { alert('Enter a boolean tag expression'); return false; }
	var fmt=this.form.fmt.value;
	if (!fmt.length) { alert('Enter the list item output format'); return false; }
	var sep=this.form.sep.value.unescapeLineBreaks();
	var tid=this.form.tid.value;
	if (!tid.length) { alert('Enter a target tiddler title'); return false; }
	var tags=this.form.tags.value;
	config.macros.matchTags.createReport(tid,tags,expr,fmt,sep,'title');
	return false;">
</form>
!end
//}}}
***/
//{{{
// SHADOW TIDDLER for displaying default panel input form
config.shadowTiddlers.MatchTags="<<matchTags panel>>";
//}}}
//{{{
// TWEAK core filterTiddlers() for enhanced boolean matching in [tag[...]] syntax:
// use getMatchingTiddlers instead getTaggedTiddlers
var fn=TiddlyWiki.prototype.filterTiddlers;
fn=fn.toString().replace(/getTaggedTiddlers/g,"getMatchingTiddlers");
eval("TiddlyWiki.prototype.filterTiddlers="+fn);
//}}}
//{{{
// REDEFINE core handler for enhanced boolean matching in tag:"..." paramifier
// use filterTiddlers() instead of getTaggedTiddlers() to get list of tiddlers.
config.paramifiers.tag = {
	onstart: function(v) {
		var tagged = store.filterTiddlers("[tag["+v+"]]");
		story.displayTiddlers(null,tagged,null,false,null);
	}
};
//}}}
|Default|@@""" PriceDigit = 2 """@@|
จำนวนหลักหลังจุดทศนิยม ของ field __ราคาต่อหน่วย__ของสินค้า
!usage
{{{[img[QInf-Stock.jpg]]}}}
[img[QInf-Stock.jpg]]
!notes
//none//
!type
image/jpeg
!file
./img/QInf-Stock.jpg
!url

!data

|Default|@@""" QtyDigit = 3 """@@|
จำนวนหลักมากสุด หลังจุดทศนิยม ของ field จำนวนสินค้า 

ดูเพิ่มเติม QtyFixDigit จำนวนหลักขั้นต่ำ หลังทศนิยม
|Default|@@""" QtyFixDigit = 0 """@@|
จำนวนหลักขั้นต่ำ หลังจุดทศนิยม ของ field จำนวนสินค้า ที่ต้องแสดงเป็นตัวเลข 

ตัวอย่าง 
||| จำนวน |>|>|>|h
| QtyFixDigit | QtyDigit | ''10'' | ''20.2'' | ''30.45'' | ''12.2753'' |
| 0 | 3 | 10 | 20.2 | 30.45 | 12.275 |
| 1 | 3 | 10.0 | 20.2 | 30.45 | 12.275 |
| 2 | 3 | 10.00 | 20.20 | 30.45 | 12.275 |
| 3 | 3 | 10.000 | 20.200 | 30.450 | 12.275 |
 
|Default|@@""" QtyShow = 0 """@@|
กำหนดระดับ user ต่ำสุด ที่สามารถเห็นยอดคงเหลือสินค้า 

| 0 |user1 (อนุญาต user ทุกคน) |
| 1 |user2 |
| 2 |user3 |
| 3 |user4 |
| 4 |user5 |
| 5 |user6 |
| 6 |user7 |
| 7 |บัญชี |
| 8 |ผู้จัดการ |
| 9 |supervisor (ไม่อนุญาต user ทุกคน)|
(มี.ค.53) เนื่องจากจอ Browse ไม่สามารถแสดงข้อความให้เห็นครบทุกตัวอักษร เช่น ชื่อสินค้า หรือชื่อสินค้า ทำให้้มีความยากลำบากในการอ่านชื่อ กรณีชื่อนั้นมีข้อความด้านหน้าเหมือนกัน ส่วนข้อความท้ายๆ ที่ต่างกัน จอ QuickInfo จึงเป็นจอที่ใช้สำหรับแสดงรายละเอียดของข้อมูลเพื่อให้ผู้ใช้สามารถอ่านได้ง่ายขึ้น โดยเปิดเป็นจอพิเศษ ที่ด้านล่างของจอ Browse ของข้อมูลสต็อก ผู้ใช้สามารถกด ''Alt + I'' เพื่อสลับระหว่าง ซ่อน/แสดงจอ QuickInfo
[img[QInf-Stock.jpg]]
''คำสั่งที่เกี่ยวข้อง''
* [[_StockShowQInf]]
* [[_CustomerShowQInf]]
* [[_VendorShowQInf]]
| 0 หรือ OFF |ปิด ไม่แสดง QuickInfo |
| +1 หรือ ON |เปิดใช้ QuickInfo ให้แสดงชื่อแทน |
| +2 |เปิดใช้ QuickInfo ให้แสดงหมวด(สินค้า/ลูกค้า/เจ้าหนี้) |
| +4 |เปิดใช้ QuickInfo ให้แสดง @@{min/max/reorder}@@ (สินค้า) |
|~|เปิดใช้ QuickInfo ให้แสดงวงเงินคงเหลือ  @@{999.99}@@ (ลูกค้า/เจ้าหนี้)  ถ้าไม่ได้กำหนดวงเงินในข้อมูลลูกค้าเพิ่มเติม จะไม่แสดง |
| +8 |เปิดใช้ QuickInfo ให้แสดงยอดค้างชำระ @@[999.99]@@ (ลูกค้า/เจ้าหนี้) ถ้าใช้ควบกับคำสั่ง แสดงวงเงินคงเหลือ จะแสดงยอดค้างชำระเมื่อไม่ได้กำหนดวงเงิน |
| + 16 |เปิดใช้ QuickInfo ให้แสดงดัชนีราคาของตารางราคา @@(#99)@@ สำหรับลูกค้า |
| +128 |เปิดใช้ QuickInfo และแสดงห้ามปิด ไม่สามารถใช้ ~Alt-I เพื่อสั่งซ่อน (เม.ย.53 เปลี่ยนจากเดิม +100) |
| - (ลบ) |กรณีกลับค่าเป็นลบ เปิดใช้ QuickInfo แต่ไม่แสดงตอนเริ่มต้น ต้องกด "Alt + I" ก่อน |
<<search>><<closeAll>><<permaview>><<newTiddler>><<newJournal "DD MMM YYYY" "journal">><<saveChanges>><<slider chkSliderOptionsPanel OptionsPanel "options »" "Change TiddlyWiki advanced options">>
/***
|''Name''|SimpleSearchPlugin|
|''Description''|displays search results as a simple list of matching tiddlers|
|''Authors''|FND|
|''Version''|0.4.1|
|''Status''|stable|
|''Source''|http://devpad.tiddlyspot.com/#SimpleSearchPlugin|
|''CodeRepository''|http://svn.tiddlywiki.org/Trunk/contributors/FND/plugins/SimpleSearchPlugin.js|
|''License''|[[Creative Commons Attribution-ShareAlike 3.0 License|http://creativecommons.org/licenses/by-sa/3.0/]]|
|''Keywords''|search|
!Revision History
!!v0.2.0 (2008-08-18)
* initial release
!!v0.3.0 (2008-08-19)
* added Open All button (renders Classic Search option obsolete)
* sorting by relevance (title matches before content matches)
!!v0.4.0 (2008-08-26)
* added tag matching
!To Do
* tag matching optional
* animations for container creation and removal
* when clicking on search results, do not scroll to the respective tiddler (optional)
* use template for search results
!Code
***/
//{{{
if(!version.extensions.SimpleSearchPlugin) { //# ensure that the plugin is only installed once
version.extensions.SimpleSearchPlugin = { installed: true };

if(!config.extensions) { config.extensions = {}; }

config.extensions.SimpleSearchPlugin = {
	heading: "Search Results",
	containerId: "searchResults",
	btnCloseLabel: "close",
	btnCloseTooltip: "dismiss search results",
	btnCloseId: "search_close",
	btnOpenLabel: "Open all",
	btnOpenTooltip: "open all search results",
	btnOpenId: "search_open",

	displayResults: function(matches, query) {
		story.refreshAllTiddlers(true); // update highlighting within story tiddlers
		var el = document.getElementById(this.containerId);
		query = '"""' + query + '"""'; // prevent WikiLinks
		if(el) {
			removeChildren(el);
		} else { //# fallback: use displayArea as parent
			var container = document.getElementById("displayArea");
			el = document.createElement("div");
			el.id = this.containerId;
			el = container.insertBefore(el, container.firstChild);
		}
		var msg = "!" + this.heading + "\n";
		if(matches.length > 0) {
			msg += "''" + config.macros.search.successMsg.format([matches.length.toString(), query]) + ":''\n";
			this.results = [];
			for(var i = 0 ; i < matches.length; i++) {
				this.results.push(matches[i].title);
				msg += "* [[" + matches[i].title + "]]\n";
			}
		} else {
			msg += "''" + config.macros.search.failureMsg.format([query]) + "''"; // XXX: do not use bold here!?
		}
		createTiddlyButton(el, this.btnCloseLabel, this.btnCloseTooltip, config.extensions.SimpleSearchPlugin.closeResults, "button", this.btnCloseId);
		wikify(msg, el);
		if(matches.length > 0) { // XXX: redundant!?
			createTiddlyButton(el, this.btnOpenLabel, this.btnOpenTooltip, config.extensions.SimpleSearchPlugin.openAll, "button", this.btnOpenId);
		}
	},

	closeResults: function() {
		var el = document.getElementById(config.extensions.SimpleSearchPlugin.containerId);
		removeNode(el);
		config.extensions.SimpleSearchPlugin.results = null;
		highlightHack = null;
	},

	openAll: function(ev) {
		story.displayTiddlers(null, config.extensions.SimpleSearchPlugin.results);
		return false;
	}
};

config.shadowTiddlers.StyleSheetSimpleSearch = "/*{{{*/\n" +
	"#" + config.extensions.SimpleSearchPlugin.containerId + " {\n" +
	"\toverflow: auto;\n" +
	"\tpadding: 5px 1em 10px;\n" +
	"\tbackground-color: [[ColorPalette::TertiaryPale]];\n" +
	"}\n\n" +
	"#" + config.extensions.SimpleSearchPlugin.containerId + " h1 {\n" +
	"\tmargin-top: 0;\n" +
	"\tborder: none;\n" +
	"}\n\n" +
	"#" + config.extensions.SimpleSearchPlugin.containerId + " ul {\n" +
	"\tmargin: 0.5em;\n" +
	"\tpadding-left: 1.5em;\n" +
	"}\n\n" +
	"#" + config.extensions.SimpleSearchPlugin.containerId + " .button {\n" +
	"\tdisplay: block;\n" +
	"\tborder-color: [[ColorPalette::TertiaryDark]];\n" +
	"\tpadding: 5px;\n" +
	"\tbackground-color: [[ColorPalette::TertiaryLight]];\n" +
	"}\n\n" +
	"#" + config.extensions.SimpleSearchPlugin.containerId + " .button:hover {\n" +
	"\tborder-color: [[ColorPalette::SecondaryMid]];\n" +
	"\tbackground-color: [[ColorPalette::SecondaryLight]];\n" +
	"}\n\n" +
	"#" + config.extensions.SimpleSearchPlugin.btnCloseId + " {\n" +
	"\tfloat: right;\n" +
	"\tmargin: -5px -1em 5px 5px;\n" +
	"}\n\n" +
	"#" + config.extensions.SimpleSearchPlugin.btnOpenId + " {\n" +
	"\tfloat: left;\n" +
	"\tmargin-top: 5px;\n" +
	"}\n" +
	"/*}}}*/";
store.addNotification("StyleSheetSimpleSearch", refreshStyles);

// override Story.search()
Story.prototype.search = function(text, useCaseSensitive, useRegExp) {
	highlightHack = new RegExp(useRegExp ? text : text.escapeRegExp(), useCaseSensitive ? "mg" : "img");
	var matches = store.search(highlightHack, null, "excludeSearch");
	var q = useRegExp ? "/" : "'";
	config.extensions.SimpleSearchPlugin.displayResults(matches, q + text + q);
};

// override TiddlyWiki.search() to sort by relevance
TiddlyWiki.prototype.search = function(searchRegExp, sortField, excludeTag, match) {
	var candidates = this.reverseLookup("tags", excludeTag, !!match);
	var primary = [];
	var secondary = [];
	var tertiary = [];
	for(var t = 0; t < candidates.length; t++) {
		if(candidates[t].title.search(searchRegExp) != -1) {
			primary.push(candidates[t]);
		} else if(candidates[t].tags.join(" ").search(searchRegExp) != -1) {
			secondary.push(candidates[t]);
		} else if(candidates[t].text.search(searchRegExp) != -1) {
			tertiary.push(candidates[t]);
		}
	}
	var results = primary.concat(secondary).concat(tertiary);
	if(sortField) {
		results.sort(function(a, b) {
			return a[sortField] < b[sortField] ? -1 : (a[sortField] == b[sortField] ? 0 : +1);
		});
	}
	return results;
};

} //# end of "install only once"
//}}}
for customized Scraft.
[[__FSys.INI]]
|Default|ยกเลิกไม่ใช้|
เป็น mod โปรแกรมรุ่นเก่า สำหร้บป้อนข้อมูลชื่อสินค้าแบบ fix format เช่น เลนส์แว่นตา
ปัจจุบัน สามารถใช้ subfield แทนได้ 
|Default|@@""" StkKeepQty=ON """@@|
ใช้ควบคุมเกี่ยวกับการคำนวณและเก็บค่ายอดคงเหลือล่าสุดของสต็อกสินค้า 

| StkKeepQty | คำอธิบาย |h
| OFF หรือ 0 |ไม่เก็บยอดคงเหลือล่าสุดใน master และ ไม่แสดงยอดคงเหลือล่าสุดในจอ browse list ถ้าต้องการดูยอดคงเหลือล่าสุด จะต้องกด space ที่ชื่อสินค้านั้น เพื่อดู stock info โปรแกรมจะคำนวณยอดคงเหลือล่าสุดจาก รายการเข้า-ออก |
| ON หรือ 1 |เก็บยอดคงเหลือล่าสุดใน master ข้อมูลสินค้า |
| 2 ถึง 9 |มีผลเหมือน 1 ใช้ lock กรณี user ระดับต่ำกว่าค่าที่กำหนด จะไม่สามารถเข้าไปแก้ยอดคงเหลือใน master ข้อมูลสินค้า ตรงๆ เช่น กำหนด 7 มีผลคือ user1 - user6 ไม่สามารถแก้ยอดคงเหลือได้ (ส่วนค่าติดลบ หรือ 0 มีผล lock อัตโนมัติอยู่แล้ว เพราะยอดคงเหลือ ไม่ได้เก็บใน master) |
| -1 |เหมือนกับ 0 แต่แสดงยอดคงเหลือในจอ browse list โดยการคำนวณ และไม่ตรวจยอดคงเหลือติดลบระหว่างตัดสต็อก (ไม่ฟ้องถ้าตัดสต็อกแล้วติดลบ) |
| -10 |เหมือนกับ -1 แต่ฟ้องถ้าตัดสต็อกแล้วติดลบ (ต้องเสียเวลาคำนวณเพื่อตรวจยอด เมื่อตัดสต็อก) |
| -11 |เหมือนกับ -1 สำหรับสต็อกที่ไม่มีโกดัง ไม่คำนวณฟ้องติดลบ เฉพาะสต็อกที่มีโกดัง ฟ้องถ้าตัดสต็อกแล้วติดลบ |

** bias +10 ใช้ควบกับ 1 ถึง 9 เพื่อให้โปรแกรมตรวจสอบยอดคงเหลือติดลบ ในระดับโกดังด้วย เมื่อมีการตัดสต็อก (ต้องเสียเวลาคำนวณ) 
** bias +100 ใช้ควบกับค่าข้างต้น optimize ความเร็วเมื่อกด space เพื่อดู stock info? ใช้กับระบบสต็อกที่ไม่ได้ใช้แยกโกดัง (StkQtyLoc) โปรแกรมจะไม่วิ่งคำนวณเพื่อหายอดแยกตามโกดัง แต่ใช้ยอดคงเหลือใน master มาแสดงเลย 
** bias -100 ใช้ควบกับค่า -1, -10 หรือ -11 optimize ความเร็วเมื่อกด space เพื่อดู stock info ใช้กับระบบสต็อกที่ไม่ได้ใช้แยกโกดัง (StkQtyLoc) โปรแกรมจะไม่วิ่งคำนวณเพื่อหายอดแยกตามโกดังซ้ำ แต่ใช้ยอดที่คำนวณได้และแสดงในจอ browse list 
|Default|@@""" StkQtyAltShow = 0 """@@|
ในจอ browse list ของข้อมูลสินค้า เราสามารถกด space เพื่อดู stock info
ถ้ากำหนดค่านี้เป็น 1 รายละเอียดของ stock info จะแสดงยอด Min/Max/Reorder ต่อท้ายยอดรวมด้วย 

เพิ่มเติม เม.ย. 2553
| 0 | แสดง Stock Info ปกติ (ยอดคงเหลือ) |
| +1 |แสดงยอด {Min/Max/ReOrder} ต่อท้ายยอดคงเหลือ |
| +10 |แสดงยอดค้างรับ (ถ้าไม่มียอดค้าง ไม่แสดง) |
| +20 |แสดงยอดค้างส่ง (ถ้าไม่มียอดค้าง ไม่แสดง) |
||@@""" StkQtyBDir = ชุดข้อมูล1,ชุดข้อมูล2,.. """@@|
ในจอแสดงรายการข้อมูลสต็อกสินค้า (browse list) เมื่อกด space ที่ชื่อสินค้าที่ต้องการ โปรแกรมจะแสดงจอ stock info ขึ้นมาให้ 

คำสั่งนี้ เป็นการกำหนดเพิ่มเติม ให้โปรแกรมไปหายอดคงเหลือ ของสินค้าชื่อเดียวกันนี้ จากชุดข้อมูลอื่นมาแสดงด้วย ช่วยให้สามารถดูยอดสต็อกคงเหลือ เปรียบเทียบระหว่างสาขาต่างๆ ได้สะดวกรวดเร็ว 

> note เราต้องกำหนดรายชื่อชุดข้อมูลเป็นชื่อเต็ม โดยใช้ , หรือ | คั่น (ไม่สามารถใช้ wildcard) 
||@@""" StkQtyLoc=โกดัง1*,โกดัง2*,.. """@@|
กำหนดให้โปรแกรมแสดงยอดคงเหลือสินค้า เฉพาะจากโกดัง ที่กำหนดโดย wildcard string

นอกจากนี้ คำสั่งนี้ มีผลให้ โปรแกรม lock ฟิลด์ ยอดคงเหลือ ใน master สต็อกสินค้า ไม่ให้ผู้ใช้เข้าไปแก้ยอดโดยตรง

คำสั่งนี้ ยังมีผลเมื่อกด space ที่ชื่อสินค้าใน browse list เพื่อดู stock info โปรแกรมจะแสดงยอดแยกตามโกดัง เฉพาะที่ตรงตามเงื่อนไข wildcard ของคำสั่งนี้ 
||@@""" StkRenameBDir = ชุดข้อมูล1,ชุดข้อมูล2,..  """@@|
ใช้กับอรรถประโยชน์ เปลี่ยนชื่อสินค้า สามารถกำหนดให้เปลี่ยนชื่อสินค้า หลายชุดข้อมูลพร้อมกันได้

ใช้ประโยชน์ กรณีกิจการมีหลายสาขา แยกข้อมูลเก็บไว้หลายชุดข้อมูล แต่ข้อมูลชื่อสินค้า จะต้องใช้ชื่อเหมือนกัน 

> note ไม่จำเป็นต้องกำหนดชื่อชุดข้อมูล ของตนเอง โปรแกรมจะเปลี่ยนชื่อในชุดข้อมูลของตนเองอัตโนมัติอยู่แล้ว 

> ปรับปรุง กรณีเปลี่ยนชื่อสินค้า 
> (25-9-2550)  โปรแกรมจะเปลี่ยนชื่อสินค้าใน Transaction แต่ละชุดข้อมูลก่อน(~StkIn, ~StkOut)  เสร็จแล้วถ้าไม่มี error จึงเปลี่ยนชื่อสินค้าใน Master แต่ละชุดข้อมูลอีกครั้ง(~StkItm, ~AltStk, Stock) 
@@
StkRenameLog = ชื่อแฟ้ม 
@@
ใช้กำหนด ชื่อไฟล์ ที่จะเก็บ log การสั่งเปลี่ยนชื่อ เพื่อใช้ตรวจสอบภายหลัง 
||@@""" StkRenameLog = ชื่อแฟ้ม """@@|
ใช้กำหนด ชื่อไฟล์ ที่จะเก็บ log การสั่งเปลี่ยนชื่อ เพื่อใช้ตรวจสอบภายหลัง 
ดูรายละเอียดใน StkRenameBDir
|Default|@@""" StkResourceDirs = dir1,dir2,.. """<br>""" StkResourceExts=.JPG;.GIF;.TIF """<br>""" StkResourceMode=0 """@@|
ใช้กรณีข้อมูลสินค้า จำเป็นต้องมีไฟล์ภายนอก มาช่วยเสริมข้อมูล เช่น โบรชัวร์, รูปภาพสินค้า ฯลฯ
หลักการทำงานคล้ายลักษณะของ hyperlink ขณะที่อยู่ในจอแสดงรายการสินค้า ถ้ากด ''ALT+S'' โปรแกรมจะตรวจหาคำสั่ง hyperlink ที่แทรกอยู่ในเนื้อข้อมูลสินค้า ถ้าพบก็จะเรียกเปิดไฟล์นั้นอัตโนมัติ เหมือนกับการ double click ที่ไฟล์นั้นใน windows 

''การใช้งาน'' 
คำสั่งสำหรับกำหนด link ชื่อไฟล์ภายนอก ที่ใช้แทรกในเนื้อข้อมูลสต็อค (ในชื่อสินค้า หรือชื่อแทน) :คือ ''{^ชื่อไฟล์}'' หรือ หลายชื่อในคราวเดียว ''{^ไฟล์1;ไฟล์2}'' สามารถกำหนดไว้สะเปะสะปะ หลายๆ ที่ โปรแกรมจะรวบรวมไฟล์ทั้งหมดขึ้นมาให้เลือก เช่น ไว้ในชื่อสินค้าก็มี {^xxx} แถมไว้ในชื่อแทนก็มี {^yyy} อีกด้วย 
* ชื่อไฟล์ใช้ wildcard ไม่ได้ 
* ชื่อไฟล์สามารถระบุนามสกุลก็ได้ 
* โปรแกรมตรวจสอบชื่อไฟล์ซ้ำอัตโนมัติ 
* ต้องกำหนด StkResourceDirs (ถ้าไม่กำหนด จะไม่ทำงาน) 
** กำหนดแบบทั่วไป เช่น StkResourceDirs=d:\pics\paper;d:\pics\other 
** กำหนดแบบให้ค้นใน subfolder ด้วยอัตโนมัติ เช่น StkResourceDirs=d:\pics\*;e:\pics\* 
** ดังนั้น จะจัดหมวดหมู่ของรูปภาพแบ่งเป็น folder ยังไงก็ได้ ตั้งชื่อ subfolder ยาวยังไงก็ได้  โปรแกรมจะความหาใน subfolder เองอัตโนมัติ 
** สามารถกำหนดได้หลาย ๆ dir โดยใช้เครื่องหมาย ; คั่น  โปรแกรมจะความหาจาก dir1 ก่อน ตามด้วย dir2 ตามด้วย subfolder ใน dir1 แล้ว subfolder ใน dir2 แล้วจึง subfolder ของ subfolder ใน dir1 , dir2 ตามลำดับ 
** ข้อควรระวัง ชื่อโฟลเดอร์จะต้องระบุเป็นแบบ 8.3 เท่านั้น เช่น c:\mydocu~1\mypict~1 
* กำหนด StkResourceExts สำหรับ ชื่อไฟล์ ที่ไม่ระบุนามสกุล 
** ค่า default คือ .JPG;.GIF;.TIF : หรือกำหนดเองใหม่ เช่น StkResourceExts=.JPG;.PDF;.LNK 
** จะต้องมี . นำหน้า และใช้ ; คั่น 
** รายการนามสกุลเป็นแบบ priority คือ โปรแกรมจะหาตามนามสกุลแรกก่อน ถ้าเจอก็จะใช้นามสกุลนั้น ไม่หาต่อไป 
** ยกเว้นมีการใช้ StkResourceMode +4 จึงหาไฟล์ชื่อเดียวกัน ทุกนามสกุลขึ้นมาให้เลือก 
* ใช้ StkResourceMode เพื่อปรับรูปแบบการใช้งาน 
** ค่า default 0 ถ้าพบ link เดียว จะเปิดไฟล์ทันที ถ้าพบมากกว่า 1 link จะขึ้นรายชื่อมาให้เลือกก่อน 
** +1 = เอา รหัสสินค้า มาใช้เป็น link (ชื่อไฟล์) ด้วย 
** +2 = แสดงชื่อไฟล์ให้ก่อนทุกครั้ง ถึงแม้ว่าจะพบแค่ 1 link 
** +4 = หาไฟล์ตามนามสกุลที่กำหนด จนครบทุกนามสกุล 
ดูรายละเอียดใน StkResourceDirs
ดูรายละเอียดใน StkResourceDirs
|Default|@@""" StkUtilOpenShare = OFF """@@|
ปกติ ในการทำงานอรรถประโยชน์ ของระบบสต็อก เช่น เปลี่ยนชื่อ หรือ คำนวณยอดสุดท้ายใหม่ โปรแกรมจะเปิดไฟล์แบบ exclusive คือ lock ไม่ให้ผู้อื่นเข้าไปเพิ่ม, ลบ, แก้ไข ไฟล์ข้อมูลสต็อก เพื่อป้องกันไม่ให้ประมวลผลผิดพลาด 

ดังนั้น ในระหว่างที่มีการประมวลผลอรรถประโยชน์อยู่ ผู้อื่นไม่สามารถทำงานใดๆ ที่เกี่ยวกับสต็อก จะต้องรอจนกว่า คำสั่งอรรถประโยชน์นั้น ทำงานเสร็จก่อน 

เปลี่ยนค่านี้เป็น ON จะเปิดไฟล์แบบ share แทน มีผลทำให้ ผู้อื่นสามารถทำงานเกี่ยวกับสต็อกได้ตามปกติ ในระหว่างที่ คำสั่งอรรถประโยชน์ทำงาน กรณีนี้ ผู้ใช้จะต้องรับความเสี่ยงเอง ถ้าเกิดประมวลผลผิดพลาด เช่น ระหว่างที่คำนวณยอดสต็อกสุดท้ายอยู่ มีการตัดสต็อกสินค้าเดียวกันจากผู้อื่นพอดี 
|Default|@@""" StkZoomTrn = max[,mode] """@@|
จากจอแสดงรายชื่อสินค้า เมื่อกด space ปกติโปรแกรมจะแสดงจอ stock info เมื่อใช้คำสั่งนี้ โปรแกรมจะแสดงจอ รายละเอียดเข้า-ออกล่าสุด ขึ้นมาก่อน

* ''max'' กำหนดจำนวนรายการที่แสดง 
* ''mode'' เงื่อนไขการแสดง ถ้าไม่กำหนด default คือ 7 
** 1 เลือกเฉพาะรายการเข้า 
** 2 เลือกเฉพาะรายการออก 
** 3 แสดงทั้งรายการเข้าและออก 
** +4 แสดงจำนวน 
** +8 แสดงราคาซื้อ(เข้า)/ราคาขาย(ออก) 
** +16 แสดงชื่อผู้ขาย(เข้า)/ผู้ซื้อ(ออก) 

ตัวอย่าง  
StkZoomTrn=5 (แสดงรายการเข้า-ออกล่าสุด 5 รายการ แสดงเฉพาะจำนวน) 
StkZoomTrn=5,23 (แสดงรายการเข้า-ออกล่าสุด 5 รายการ แสดงจำนวนและชื่อผู้ซื้อ/ผู้ขาย) 

ใช้ประโยชน์เพื่ออำนวยความสะดวก ให้ผู้ใช้สามารถตรวจสอบคร่าวๆ ว่าสินค้านี้ ล่าสุด เคยซื้อ หรือขายให้ใครไปบ้าง 

นอกจากนี้ ในหน้าจอแสดงรายการสินค้า (browse list) เมื่อมีการใช้คำสั่งนี้ โปรแกรมจะเอาหมวดบิลล่าสุด มาแสดงหน้าชื่อสินค้าให้ด้วย เป็นบิลซื้อ หรือบิลขาย ขึ้นอยู่กับเงื่อนไข mode 1,2,3 ที่กำหนดเลือกรายการ เช่น ถ้าใช้ mode=1 จะแสดงหมวดบิลซื้อล่าสุด mode=2 จะแสดงหมวดบิลขายล่าสุด

(4 ม๊.ค. 2553) 
''mode'' เพิ่มเติม +64, +128 เกี่ยวกับการกำหนดแสดง/ไม่แสดงหมวดบิลล่าสุด __ยกเลิก__การแสดงหมวดบิลล่าสุด ที่ขึ้นอยู่กับ mode 1,2,3
* ''mode'' เงื่อนไขแสดงหมวดบิลล่าสุด
** +64 แสดงหมวดบิลซื้อล่าสุด
** +128 แสดงหมวดบิลขายล่าสุด
** +192 แสดงหมวดบิลซื้อ/บิลขายล่าสุด (ถ้าล่าสุดวันเดียวกัน แสดงหมวดบิลซื้อ)
{{{
24/1/2552

MOD - StockZoomTrn

วิธีใช้
StkZoomTrn=maxshow,mode

mode: เอาค่าต่อไปนี้มาบวกกัน
1 = เอา stkin
2 = เอา stkout
4 = แสดง จำนวน 
8 = แสดง ต้นทุน/ราคา
16 = แสดง ชื่อลูกค้า

default mode คือ 7

ของคุณนพดล น่าจะใช้ mode เป็น 1+2+8+16 = 27

ดังนั้นถ้าต้องการให้แสดง trn ล่าสุด 3 รายการจะได้เป็น 
StkZoomTrn=3,27

ไฟล์ที่ update คือ _Stk1_.EXE และ H_Stk1_.EXE (ใช้สำหรับ F1)


การทดสอบ
เมื่อกำหนด StkZoomTrn
จอสต็อก จะขึ้นหมวดบิลล่าสุด หน้าชื่อสินค้า อัตโนมัติ

และถ้ากด SPACE จะขึ้น Info ของ Transaction 
คล้ายกับกรณี หลายโกดัง

}}}
{{{
28/8/50

**เดิม**
เมื่อกด SPACE ที่ข้อมูลสต็อกสินค้า 
โปรแกรมจะแสดง จอเหมือนตอนป้อนข้อมูลสต็อก

ถ้ามีข้อมูลแยก โกดัง(ที่เก็บ) โปรแกรมจะขึ้นจอพิเศษ
แสดงยอดย่อยแต่ละที่เก็บมาให้ดู

**เดิม**
สามารถกำหนด StkKeepQty เพื่อกำหนดวิธีใช้ยอดคงเหลือ
StkKeepQty = 0 ไม่เก็บยอดคงเหลือไว้ที่ข้อมูล Master Stock 
และไม่แสดงยอดคงเหลือ ในจอ browse สต๊อก (ต้องกด SPACE จึงทราบยอดคงเหลือ)
StkKeepQty = -1 ไม่เก็บยอดคงเหลือไว้ที่ข้อมูล Master Stock แต่จะคำนวณอัตโนมัติเพื่อแสดงในจอ browse ไม่จำเป็นต้องกด SPACE แต่จะทำงานช้ากว่า

**ใหม่**
สามารถกำหนด StkQtyBDir = ชื่อชุดข้อมูล1|ชื่อชุดข้อมูล2
เมื่อกด SPACE ครั้งแรก โปรแกรมจะแสดงข้อมูลของสต็อกเหมือนเดิม
แต่ถ้ากด SPACE อีกครั้ง โปรแกรมจะแสดงยอดสต็อกคงเหลือในชุดข้อมูลอื่นนั้น


**ใหม่**
เพื่อ optimize ความเร็วในการคำนวณยอดคงเหลือ ไม่ให้ซ้ำซ้อน
เวลาแสดงข้อมูลเมื่อกด SPACE
เราสามารถกำหนด StkKeepQty = 2 
เป็นการกำหนดให้โปรแกรมเอายอดคงเหลือจาก Master Stock มาใช้
โดยไม่ต้องคำนวณจาก transaction เข้า-ออก
(มีผลข้างเคียงคือ ไม่สามารถแสดงยอดแบบแยกโกดัง)
หรือเราอาจกำหนด StkKeepQty = -2
เพื่อให้แสดงยอดคงเหลือ จากยอดที่คำนวณได้ในจอ Browse ไม่ต้องคำนวณซ้ำ
}}}

{{{
กรณี SPC1 ซึ่ง Master Stock เก็บยอดคงเหลือด้วย

จึงควรกำหนด StkKeepQty=2


กรณี SPC2 - SPC4 ซึ่ง share ใช้ Master Stock จาก SPC1
จะต้องคำนวณจาก Transaction

ดังนั้นจะต้องกำหนดเป็น
StkKeepQty = 0 (ปิดไม่คำนวณยอดคงเหลือ ระหว่าง browse) หรือ StkKeepQty= -2 แล้วแต่ความสะดวก

ตอนนี้ เจอปัญหา เกี่ยวกับ StkKeepQty

ที่ขัดแย้งกับ definition เก่า ที่ให้ตั้งเป็น 0 หรือ -10

กำลังสรุป เงื่อนไขการตั้งค่าใหม่อยู่ครับ...


31/8/50
แก้ไขโปรแกรมใหม่

เปลี่ยนเป็นใช้ +100 สำหรับ StkKeepQty

สรุปคือ SPC1 ควรใช้ StkKeepQty = 101

ส่วน SPC อื่น ควรใช้ StkKeepQty = -110 หรือ -100

เพิ่ม bias +100 หรือ -100 เข้าไปมีผลคือ
โปรแกรมจะไม่วิ่งคำนวณ ยอดคงเหลือจาก transaction ซ้ำซ้อน เมื่อเคาะ space
แต่ใช้ค่ายอดคงเหลือที่คำนวณและแสดงในหน้าต่างข้อมูลสต็อก บนจออยู่แล้ว 
}}}

{{{
6/9/51

- ผมมีข้อสงสัยครับ ผม Set StkKeepQty=101 แล้วพบว่าเวลาเคาะ space bar ไล่สินค้า
ในข้อมูลสต๊อค ทีละตัว พบว่า บางครั้งก็ไม่มีจอ วิ่งคำนวณ, สินค้าบางตัวคำนวณ Stockin, StockOut
วิ่ง ด้านล่างให้เห็น (ซึ่งบางตัววิ่งนาน) พอมากดซ้ำอีกครั้ง วิ่งคำนวณ เหมือนกันแต่เร็วขึ้น ..
ไม่ทราบว่า โปรแกรมวิ่งคำนวณอะไร แล้วทำยังไงจะไม่คำนวณ (เหมือนโปรแกรมรุ่นเก่า) เพราะบางตัววิ่งนาน..


ปล. 
1. ในชุดข้อมูลมีการใช้ ที่เก็บด้วย
2. สินค้าตัวที่วิ่งคำนวณ หลายตัวไม่มีการระบุที่เก็บ (เคาะเสร็จวิ่งคำนวณ แต่ไม่มีอะไรโผล่ขึั้นมา ไม่รู้ทำอะไร)
3. สินค้าบางตัววิ่ง(เหมือน)คำนวณ นานจนน่ารำคาญ ครับ
4. ทดสอบ ลบไฟล์ __Fsys.ini ไม่ให้ Set อะไรเลย ก็ยังวิ่งคำนวณอยู่ครับ

การกำหนดเป็น 1 และมีการใช้โกดัง โปรแกรมจะต้องวิ่งคำนวณเก็บยอดแยกตามโกดัง ตาม transaction ตั้งแต่ต้นทุกครั้ง (หรือวิ่งคำนวณไปจนจบแล้วพบว่า สินค้านั้นไม่มีโกดังเลย)

}}}

{{{
7/9/2550

1. แก้ Bug กรณี StkKeepQty=OFF กด SPACE แล้วไม่ขึ้น

2. แก้ไขเรื่องเว้นบรรทัดห่าง

3. เพิ่มคำสั่ง
StkQtyAltShow=1
ให้แสดงยอด Min/Max/ReOrder ต่อท้ายยอดรวมด้วย

4. แก้ Bug กรณี __FSYS.INI ใช้ #FCFG
}}}
[[Markup Help|http://tiddlywiki.org/wiki/TiddlyWiki_Markup]]
[[Tags Help|http://tiddlywiki.org/wiki/Tags]]
[[Macro Help|http://tiddlywiki.org/wiki/Core_Macros]]
----
<<matchTags popup "label:...more (Draft)" __FSys.INI AND Draft>>
[[...Open List|ListCommandsByName]]
----
<<matchTags inline "[[%0]]" "\n" __FSys.INI AND NOT Draft>>
[[Recent Changed|ListRecentChanged]]
[[Cmds by Name|ListCommandsByName]]
[[Cmds by Date|ListCommandsByModified]]
[[Jobs by Name|ListJobsByName]]
[[Jobs by Date|ListJobsByModified]]
----
<<matchTags popup "label:...Draft" MOD AND Draft>>
[[...Open List|ListJobsByName]]
----
<<matchTags inline "[[%0]]" "\n" MOD AND NOT Draft>>
<<tag "ภาษีมูลค่าเพิ่ม">>
<<tag "สต็อก">>
<<tag "ลูกค้า-ลูกหนี้">>
<<tag "เจ้าหนี้">>
<<tag "บิลขาย">>
<<tag "บิลซื้อ">>
<<tag "ใบเสร็จ">>
<<tag "ใบจ่าย">>
<<tag "บัญชีธนาคาร">>
<<tag "บัญชีแยกประเภท">>
<<tag "report">>
<<tag "security">>
<<tag "export">>
<<timeline>>
|Default|@@""" TmpAutoCancel = 0 """@@|
ในระหว่างป้อนข้อมูลหรือแก้ไขข้อมูล แล้วทิ้งจอไว้ ถ้าไม่มีการกดคีย์ใดๆ นานเกินเวลาที่กำหนด (วินาที) โปรแกรมจะทำการยกเลิกการป้อนข้อมูลนั้นอัตโนมัติ (ถ้าใช้ค่าเป็นลบ จะออกจากจอโดยไม่มีการเตือนให้ตอบยกเลิก) 
|Default|@@""" TmpRefresh = 60 """@@|
ในจอ Browse ข้อมูลแบบ list โปรแกรมจะทำการ อ่านข้อมูลจาก database เพื่อ refresh list ของข้อมูล ภายในเวลาที่กำหนด (default คือ 60 วินาที) 
ไฟล์นี้จะ update ใหม่เรื่อยๆ สามารถ download ได้ที่ 
| share folder | html | zip (html + img) |
| [[4Shared server|http://www.4shared.com/dir/32763227/b6404fe2/html.html]] |[[download|http://www.4shared.com/file/233996079/7c9dd7ee/__FSysINI.html]] | [[download|http://www.4shared.com/file/242548179/be60cc8d/__FSysINI.html]] |
| [[SkyDrive server|http://cid-a5abc44714ee2e14.skydrive.live.com/browse.aspx/Free%20Scraft/html]] |[[download|http://cid-a5abc44714ee2e14.skydrive.live.com/self.aspx/Free%20Scraft/html/^_^_FSysINI.html]] | [[download|http://cid-a5abc44714ee2e14.skydrive.live.com/self.aspx/Free%20Scraft/html/^_^_FSysINI.zip]] |
| [[ZumoDrive server|http://www.zumodrive.com/share/4d43OWE3Mj]] | [[download|http://www.zumodrive.com/share/4aZhYTE2MT]] | [[download|http://www.zumodrive.com/share/4d42ODYxZj]] |
| Dropbox server | [[download|http://dl.dropbox.com/u/5135572/__FSysINI.html]] | [[download|http://dl.dropbox.com/u/5135572/__FSysINI.zip]] |
หมายเหตุ
# ไฟล์เดี่ยว html สามารถใช้กับ Opera 7.2+, Firefox, Safari, Netscape, Mozilla ได้ทันที
# ไฟล์รวม zip (html + img) สำหรับ Internet Explorer ซึ่งไม่ support [[inline images|http://www.websiteoptimization.com/speed/tweak/inline-images/]] ในรูปแบบของ [[Data URI|http://en.wikipedia.org/wiki/Data_URI_scheme]] จะต้อง download zip ไฟล์ ซึ่งมีไฟล์ html และ ไฟล์รูปภาพประกอบ สำหรับใช้ร่วมกัน
9 ก.ค. 2553
*ปรับปรุงคำสั่ง [[Work Flow]]
5 ก.ค. 2553
*ปรับปรุงคำสั่ง [[Work Flow]] [[_InvAutoReferNo]] [[_InvLockHiDate]]
25 เม.ย. 2553
*ปรับปรุงคำสั่ง QuickInfo เลือกเปิด/ปิด ข้อมูลที่ต้องการให้แสดงได้
8 เม.ย. 2553
*จัด list ของ เนื้อหาใหม่ แสดงส่วนของค่า default ของคำสั่ง เสร็จแล้ว
*ปรับ Main Menu ด้านซ้ายใหม่
*update MatchTagsPlugin และปรับแก้ code ให้ support options MatchTagsDescription
*ปรับแก้ code ImageSizePlugin ให้ default เป็น resize width 75%
4 เม.ย. 2553
* ปรับปรุงคำสั่ง StkQtyAltShow
17 มี.ค. 2553
*กำลังจัดโครงสร้างของ เนื้อหาใหม่ เพื่อให้สามารถ list ดูง่ายกว่าเดิม
โปรแกรม Scraft มีประวัติการใช้งานที่ยาวนานกว่า 20 ปี มีความสามารถปรับเปลี่ยน คุณสมบัติโปรแกรม โดยผ่านการกำหนดค่า ด้วยคำสั่งในไฟล์ """__FSys.INI"""

ปัญหาก็คือ ไม่มีใครเข้าถึงความสามารถทั้งหมดของโปรแกรม แม้กระทั่งตัวโปรแกรมเมอร์เอง

ดังนั้น ก่อนที่จะหลงลืมไป ผมจึงมีความคิดว่าจะรวบรวม คำสั่งทั้งหมด ที่โปรแกรมนี้เปิดให้ ปรับเปลี่ยน พร้อมทั้งอธิบายความหมายของมัน 

คำอธิบาย พยายามทำเป็น 2 รูปแบบ คือ ลักษณะของ <<tag "__FSys.INI" "คำสั่ง">> กับ อธิบายลักษณะของ <<tag "MOD" "งาน">>

ที่ทำไปแล้วคือ
1. แปลง คำอธิบาย ที่เคยทำไว้ใน wiki เดิม มาใส่ไว้ที่นี่
2. รวบรวมคำสั่ง ที่เคยเขียนอธิบายอยู่ใน Forum [[NetScraft Talk|http://admthai.homeip.net/forum]] บางส่วนก็ยังเป็น <<tag "Draft" "ร่างสำเนา">> ไม่ได้เรียบเรียง ใช้วิธีลอกเอามาเก็บไว้เป็นหลักฐานก่อน

สำหรับงานขั้นต่อไป คือ รวบรวมคำสั่ง ที่หลงเหลือ หลุดรอดไป จาก source code โดยตรง พูดง่ายๆ คือ แกะจาก source code และพยายามตีความคำสั่ง จากการอ่าน source code นั้น
|Default|@@""" VATRate = 7 """@@|
กำหนดอัตรา ภาษีมูลค่าเพิ่ม 
|Default|@@""" VndUtilOpenShare = OFF """@@|
ปกติ ในการทำงานอรรถประโยชน์ ของระบบเจ้าหนี้ เช่น เปลี่ยนชื่อ หรือ คำนวณยอดสุดท้ายใหม่ โปรแกรมจะเปิดไฟล์แบบ exclusive คือ lock ไม่ให้ผู้อื่นเข้าไปเพิ่ม, ลบ, แก้ไข ไฟล์ข้อมูลเจ้าหนี้ เพื่อป้องกันไม่ให้ประมวลผลผิดพลาด 

ดังนั้น ในระหว่างที่มีการประมวลผลอรรถประโยชน์อยู่ ผู้อื่นไม่สามารถทำงานใดๆ ที่เกี่ยวกับเจ้าหนี้ จะต้องรอจนกว่า คำสั่งอรรถประโยชน์นั้น ทำงานเสร็จก่อน 

เปลี่ยนค่านี้เป็น ON จะเปิดไฟล์แบบ share แทน มีผลทำให้ ผู้อื่นสามารถทำงานเกี่ยวกับเจ้าหนี้ได้ตามปกติ ในระหว่างที่ คำสั่งอรรถประโยชน์ทำงาน กรณีนี้ ผู้ใช้จะต้องรับความเสี่ยงเอง โอกาสที่การประมวลผลผิดพลาด เช่น ระหว่างที่คำนวณยอดหนี้ล่าสุดอยู่ มีการเพิ่มหนี้ของเจ้าหนี้รายนั้นจากผู้อื่นพอดี 
{{{
คืนนี้ได้ข้อสรุปแล้ว...
หลังจากที่โทรคุยกันกับ adm อยู่นาน

เปลี่ยนไปใช้ระบบ Work Flow MOD

ร่างแนวคิดคร่าวๆ ดังนี้

Move - ย้ายบิล ไปหมวดบิลอื่น (ใช้คำสั่ง update database)
        สามารถกำหนด ให้คืนรายการบิลต้นฉบับ อัตโนมัติ
        สามารถกำหนด ใช้วันที่ปัจจุบัน หรือใช้วันที่เดิม
        สามารถกำหนด run เลขที่ใหม่ หรือใช้เลขที่เดิม
        สามารถกำหนด ให้ทำงานตามคำสั่งตัวหนอน ของบิลหมวดใหม่ เหมือนตอนกด F10 save

Copy - คัดลอก ไปหมวดบิลอื่น (ใช้คำสั่ง insert database)
        สามารถกำหนด ให้คืนรายการบิลต้นฉบับ อัตโนมัติได้
        สามารถกำหนด ใช้วันที่ปัจจุบัน หรือใช้วันที่เดิม
        สามารถกำหนด run เลขที่ใหม่ หรือใช้เลขที่เดิม
        สามารถกำหนด ให้รอ user confirm เพื่อ save ก่อนได้ (edit)
        สามารถกำหนด ให้ทำงานตามคำสั่งตัวหนอน ของบิลหมวดใหม่ เหมือนตอนกด F10 save
      สามารถกำหนด ให้ลบข้อมูลบิลต้นฉบับ หลังจาก copy สำเร็จได้

อื่นๆ จะต้องรองรับ auto-wf ทำงานเมื่อกด F10 save ด้วย
}}}

{{{
10/8/52

เอ๊า... เร่เข้ามา...

1. custom Work Flow Menu (ไฟล์ .INI หรือ .MNU)
กำหนดใน __FSYS.INI
_InvWFMenu=ชื่อไฟล์ [.INI หรือ .MNU]
หรือ
_InvWFMenu.หมวดบิล=ชื่อไฟล์ [.INI หรือ .MNU]
ภายในเมนูไฟล์ ใช้ format เหมือนกับเมนูรายงาน ฯลฯ เช่น
ไฟล์คำสั่ง.WFC=ข้อความที่แสดงในเมนู 

2. Work Flow Command File (ไฟล์ .WFC)
สำหรับเขียนคำสั่ง Work Flow มีคำสั่งสำหรับใช้ดังนี้

WFCmd=n [0=copy (defaut), 1=change, 2=copy auto, 3=change auto, 4=send to other bdir]
สำหรับคำสั่ง 0,1 นั้น โปรแกรมจะหยุดรอที่หน้าจอ edit ให้ user แก้ไขเนื้อหาในบิล แล้ว กด F10 confirm หรือ Esc ยกเลิก

WFNewType=หมวดบิล ถ้าเว้นว่าง แปลว่าไม่เปลี่ยนหมวด

WFNewBDir=ชื่อชุดข้อมูล ใช้กับ cmd 4 send to other bdir เท่านั้น ถ้าเว้นว่าง แปลว่าไม่เปลี่ยนชุดข้อมูล

WFChgVal=n กำหนดให้เปลี่ยนค่าบางฟิลด์ +1=เลขที่บิล (run ใหม่), +2=วันที่ (ใส่วันที่ปัจจุบัน), +16=ไม่ให้ใส่เลขที่ต้นฉบับ ในช่องอ้างถึง (ปกติจะใส่ให้อัตโนมัติ)

WFRefTxt=n กำหนดให้สร้างข้อความ Refer {REF#เลขใบต้นฉบับ วันที่ต้นฉบับ}{BREF#เลขอ้างถึงเดิมของต้นฉบับ} ขนาดของ ref ยาวเกินช่องหมายเหตุ เปลี่ยนเป็น {#เลขใบต้นฉบับ วันที่ต้นฉบับ}{~#เลขอ้างถึงเดิมของต้นฉบับ} 1=ใส่ในหมายเหตุ (ถ้าช่องหมายเหตุว่าง) 2=บังคับใส่ในช่องหมายเหตุ ไม่สนใจว่าว่างหรือไม่ 3=ใส่ในช่องหมายเหตุ ถ้าว่าง และ ไม่ต้องเอาไปใส่ในบรรทัดรายการอีก (ต้องใช้ควบกับ 4) +4=ใส่เพิ่มในบรรทัดรายการ (สามารถใช้ควบกับ 1 หรือ 2 หรือ 3) เช่น 6 (2+4) คือ ใส่ทั้งช่องหมายเหตุ และบรรทัดรายการ

WFBefore=n กำหนดให้ทำงาน/ตรวจสอบสถานะก่อนคัดโอนบิล  
1=ผ่านรายการ, 2=ยกเลิก, 3=คืนรายการ  *note: คำสั่ง 1-3 โปรแกรมจะตรวจสถานะบิลด้วยอัตโนมัติ
4=เฉพาะบิลที่ไม่ผ่านรายการ, 5=เฉพาะบิลที่ผ่านรายการ, 6=เฉพาะบิลที่ยกเลิก, 7=เฉพาะบิลที่ไม่ยกเลิก *note: คำสั่ง 4-7 โปรแกรมจะตรวจสถานะบิลเท่านั้น

WFAfter=n กำหนดให้ทำงานหลังจากคัดโอนบิลสำเร็จ ค่า 1-3 เหมือนกับ WFBefore 
มีเพิ่มเติม
 +8 สั่งลบบิลต้นฉบับ  สามารถใช้ควบกับคำสั่งคืนรายการ เช่น 11 (3+8)
 +16 กระโดดไปที่ จอบิลที่คัดโอนไปใหม่ (หมวดใหม่)
}}}

{{{
ข้อแตกต่าง ระหว่างคำสั่ง copy + delete after กับคำสั่ง change คือ

คำสั่ง change จะทำงานโดย ใช้ข้อมูล record เดิมใน database
เปลี่ยน หมวดบิล (และ/หรือ เลขที่บิล, วันที่) แล้วทำงานคำสั่ง UPDATE 
(ถ้าอยู่ในจอ edit จะอยู่ใน โหมด แก้ไข โดยกด F4)
ดังนั้น change จะทำงานได้กับ บิลที่ไม่ได้ถูก lock หรือบิลที่ยังไม่ได้ผ่านรายการเท่านั้น

ขณะที่ copy จะทำงานโดย คัดลอกข้อมูล เพื่อไป INSERT record ใหม่ใน database
เสร็จแล้ว จึงกลับมา DELETE record ต้นฉบับ
คล้ายกับการกด F5 เพิ่ม แล้วกด F5 อีกครั้ง copy ข้อมูลเดิมมาใส่
หลังจาก save แล้ว กลับไปสั่งลบ โดยกด F6 ที่ข้อมูลเดิม
ดังนั้น copy สามารถทำงานไปก่อน โดยที่บิลนั้น lock อยู่
หลังจากสำเร็จ จึงค่อยสั่งคืนรายการ เพื่อลบบิลต้นฉบับ

ตัวอย่างคำสั่ง WFC


โค๊ด:
#คัดลอกบิลไปหมวด XX
WFNewType=XX
WFRefTxt=3

โค๊ด:
#ย้ายบิลไปหมวด YY
WFNewType=YY
WFCmd=1
WFRefTxt=1
}}}

{{{
ตอนนี้ ยังมีการปรับปรุง เกี่ยวกับคำสั่งย่อยๆ อีกเล็กน้อย 
เพื่อให้ทำงานได้ครอบคลุมหลากหลายยิ่งขึ้น

อธิบายต่อ เกี่ยวกับ WFCmd=4 send to other bdir
ทำงานคล้ายกับ copy แต่ไม่มีเรื่องหยุดรอให้ edit ก่อน save
มีความพิเศษคือ สามารถใช้ส่งข้ามชุดข้อมูลได้ และไม่มีการตรวจเช็ค ว่าหมวดบิลนั้น มีกำหนดใน database หรือไม่ (ขณะที่คำสั่งอื่น ถ้าหมวดบิลนั้นไม่มี จะไม่สามารถ save ได้)
และไม่ทำงานใดๆ ตามคำสั่งตัวหนอน เมื่อ save ด้วย

นอกจากนี้ยังมีข้อจำกัดคือ ไม่สามารถเปลี่ยน เลขที่/วันที่
ถ้าไม่ระบุชุดข้อมูล แปลว่าใช้ชุดข้อมูลเดียวกัน
ถ้าไม่ระบุหมวดบิลใหม่ แปลว่าใช้หมวดบิลเดิม
ถ้าไม่ระบุทั้งคู่ ก็ error ครับ ข้อมูลซ้ำ เพราะ INSERT ไปที่บิลเลขเดิม

แต่คำสั่ง WFBefore และ WFAfter ยังสามารถใช้ได้ สามารถสั่งผ่านรายการ/คืนรายการ หรือลบบิลต้นฉบับ หลังจากส่งบิลไปเสร็จแล้ว

ดังนั้น เราจึงสามารถใช้ประโยชน์ ในแง่การใช้ทำสำเนา (carbon copy) 
โดยส่งไปที่บิลหมวดอื่น หรือส่งไปที่ชุดข้อมูลอื่น

ไฟล์ที่ update คือ _Inv1_.EXE (_Inv2_.EXE, _Inv3_.EXE)

เกือบลืม คีย์สำหรับเรียก Work Flow Menu จากหน้าบิลขายคือ Ctrl+F8

13/8/52
ยกเลิก ความสามารถนี้ใน _Inv*.EXE เนื่องจากทำให้ไฟล์มีขนาดโตมา มีปัญหาเรื่อง Memory Free 
เอาไปใส่ไว้ใน S_Inv1_.EXE (ดูด้านล่าง) 
}}}

{{{
ลืมแจ้ง update อีกเรื่องหนึ่ง
ที่ทำเพิ่มไปแล้ว พร้อมๆ กับ _InvCustomListType
คือ คำสั่งต่อเนื่องจาก Work Flow
_InvWFAutoCmd=ชื่อไฟล์.WFC

สำหรับกำหนดให้ ทำงานคำสั่ง Work Flow อัตโนมัติ
เมื่อมีการกด F10 เพื่อ save หลังจาก เพิ่ม (F4) หรือ แก้ไข (F5)

เพิ่มเติม อีกนิด
สำหรับโปรแกรมปกติที่ใช้ _Inv1_.EXE จะไม่มีความสามารถของ WF นะครับ
จะต้องใช้ S_Inv1_.EXE + M_Inv1_.EXE เท่านั้น

อีกเรื่อง ได้เพิ่ม Hot Key สำหรับ WF นอกจาก Ctrl+F8 แล้ว (ซึ่งต้องใช้ 2 มือช่วยกันกด และผมไม่ถนัดเลย เวลา test บ่อยๆ) ยังสามารถกด Alt+W ได้อีกด้วย

สำหรับด้าน บิลซื้อ ตอนนี้ก็เพิ่ม ปรับ _PoCustomListType กับ WF 
ให้ทำงานได้แบบเดียวกับบิลขายแล้ว (แต่ยังไม่ได้ test จ้า)

}}}

{{{
6-10-52

ตามที่ adm เสนอ ให้โปรแกรมหา maxrunno ระหว่างหมวดบิลใน workflow
เพื่อแก้ปัญหา กรณี move บิลจาก หมวด A ไป หมวด B
จะเกิดปัญหา runno ของ หมวด A กลับไปใช้เลขเดิม
เนื่องจาก เลขบิลล่าสุด ถูก move หายไปแล้ว
ดังนั้นถ้าให้โปรแกรมใช้ runno ที่มากที่สุด ของทั้งหมวด A และ B
ก็จะทำให้สามารถ หมวด A run ได้ถูกต้อง

เพิ่มคำสั่ง
_InvCrossRunNo.หมวดบิล=หมวดบิล,หมวดบิล,...
_PoCrossRunNo.หมวดบิล=หมวดบิล,หมวดบิล,...
การทำงาน
โปรแกรมจะหา runno ของหมวดบิลที่ใช้อยู่ก่อน เหมือนปกติ
หลังจากนั้น จะตรวจสอบคำสั่ง CrossRunNo
ถ้ามี ก็จะหา runno ของหมวดบิล ตามที่ระบุ
และเลือกใช้ runno ที่เป็นเลขมากที่สุด

ตัวอย่าง 
_InvCrossRunNo.I0=I1,I2
}}}

{{{
28/6/2553
ปรับปรุงคำสั่ง WFCmd
เพิ่ม WFCmd=5 update to refer
หลักการคือ WFNewType=หมวดบิล มีหมวดบิลตรงกับที่หมวดบิลที่ระบุอ้างถึงในบิลใบนั้น
โปรแกรมจะทำการ update(หรือเพิ่มถ้ายังไม่มี)บิลตามหมวดและเลขที่ ตามที่ระบุในอ้างถึง
ใช้ประโยชน์กรณี อพาร์ตเมนต์ ที่ต้องการ update ต้นแบบของใบแจ้งหนี้ล่าสุด เพื่อใช้ในอ้างถึงในการเปิดบิลครั้งต่อไป (ได้ update เลขที่มิเตอร์น้ำ/ไฟฟ้าล่าสุดติดมาด้วย)

- ถ้าบิลที่จะ update นั้นผ่านรายการแล้ว โปรแกรมจะฟ้อง error
- ทำงานคล้ายคำสั่ง 4 send to other bdir คือ สามารถใช้ส่งข้ามชุดข้อมูลได้ และไม่มีการตรวจเช็ค ว่าหมวดบิลนั้น มีกำหนดใน database หรือไม่ (ขณะที่คำสั่งอื่น ถ้าหมวดบิลนั้นไม่มี จะไม่สามารถ save ได้) และไม่ทำงานใดๆ ตามคำสั่งตัวหนอน เมื่อ save ด้วย
}}}

{{{
9/7/2553
WFChgVal=n กำหนดให้เปลี่ยนค่าบางฟิลด์ +1=เลขที่บิล (run ใหม่), +2=วันที่ (ใส่วันที่ปัจจุบัน), +16=ไม่ให้ใส่เลขที่ต้นฉบับ ในช่องอ้างถึง (ปกติจะใส่ให้อัตโนมัติ)
เพิ่มเติม เพื่อให้รองรับคำสั่ง _InvSaleAsLogin
+32=reset field พนักงาน (sale) 
+64=reset field หมายเหตุ
+128=reset field ชื่อสรุปบิล
}}}
|Default|@@""" XCostLevel = 0 """@@|
เป็นคำสั่งใช้กับงาน <<tag MOD "Modify Job">> เรื่อง เกี่ยวกับ [[ต้นทุนแฝง]]
[[ต้นทุนแฝง]] แบ่งเป็น 3 ระดับ คือ ~XCost, Expense และ Deposit 

ถ้าต้องการให้ผนวก ~XCost level ต่างๆ เข้ากับต้นทุนปกติ ทำให้รายงานเดิมทั้งหมด เช่น รายงานวิเคราะห์ขาย รายงานสต็อกสินค้า ฯลฯ สามารถแสดงต้นทุนแบบผนวกแทนอัตโนมัติ โดยไม่แก้ไขรายงานเดิม 

| 1 |~XCost |
| 2 |Expense |
| 4 |Deposit |
| 128 |กรณีที่ไม่ต้องการให้ผนวก แต่เปิดการทำงาน[[ต้นทุนแฝง]]เราจะใช้ ค่าที่นอกเหนือจาก 3 level ดังกล่าว |
| +64 |Direct input ป้อน[[ต้นทุนแฝง]] โดยตรงที่จอ สินค้าเข้าสต็อค ใช้วิธี เคาะ space แล้วกด * (ดอกจัน)|

ตัวอย่าง การผสมค่า ~XCost
| 3 (1+2) |~XCost + Expense |
| 5 (1+4) |~XCost + Deposit |
| 6 (2+4) |Expense + Deposit |
| 7 (1+2+4) |~XCost + Expense + Deposit |

> รายการ[[ต้นทุนแฝง]] ที่เกิดจาก Direct input จะมีข้อสังเกตคือ เลขที่อ้างถึงเป็น ''**''

ดูเพิ่มเติม: [[ต้นทุนแฝง]]
|Default|@@"""_BatchDelay""" = [[_DialogDelay]]/20 @@|
ตั้งเวลา หน่วยเป็น millisec (1/1000 วินาที) ใช้หน่วงเวลากรณีแสดงหน้าต่างข้อความ ภายใน loop การทำงานแบบ Batch ที่ขึ้นหน้าต่างติดต่อกันจำนวนมาก เช่น การสั่งคำนวณยอดสต็อกสุดท้ายใหม่ 

ถ้าไม่มีการกำหนดค่านี้ โปรแกรมจะใช้ค่าเท่ากับ [[_DialogDelay]]/20 แทน
|Default|@@""" _BrowsePage = OFF """@@|
ถ้ากำหนดเป็น ON จะทำให้การแสดงข้อมูลของแต่ละ module เป็นแบบ Single Page แทนที่จะเป็นแบบ list รายการข้อมูลตามปกติ มีประโยชน์คือ สำหรับกรณีที่ connection database นั้นช้ามากๆ ถ้าโปรแกรมแสดง list ตามปกติ จะต้องเสียเวลาอ่านข้อมูลจาก database เป็นจำนวนมาก ทำให้เสียเวลานานกว่าการแสดงข้อมูลแบบ Single Page 
|Default|@@""" _BuyTaxChain = OFF """@@|
กรณีที่ใช้ [[_BuyTaxChain]]=ON จะมีผลเปลี่ยน sequence การป้อนข้อมูลภาษีซื้อ

ในการเพิ่มข้อมูลภาษีซื้อ เมื่อกด F10 เพื่อ save ข้อมูลภาษีซื้อ โปรแกรมจะส่ง วันที่/เลขที่ใบกำกับภาษี ของรายการภาษีซื้อนั้น ข้ามไปป้อนข้อมูล เพิ่มหนี้เจ้าหนี้ และ สต็อกสินค้าเข้า ก่อนที่จะกลับมาเพิ่มข้อมูลภาษีซื้อรายการต่อไป

นอกจากนี้ ยังมีผลกับ หน้าจอแสดงรายการภาษีซื้อ (browse list) เราสามารถใช้วิธีกด Space แล้วกด F1 เพื่อเข้าไปป้อนข้อมูล เพิ่มหนี้เจ้าหนี้ หรือสต็อกสินค้าเข้า ที่มี วันที่/เลขที่เอกสารเดียวกัน 

คำสั่ง [[_BuyTaxChain]]=2 ทำงานเหมือน [[_BuyTaxChain]]=ON และเพิ่มกรณีสั่งลบรายการภาษีซื้อ โปรแกรมจะส่งคำสั่งไปลบรายการ เพิ่มหนี้เจ้าหนี้ และสต็อกสินค้าเข้า อัตโนมัติ 
|Default|@@""" _CSVCode = 1 """@@|
กำหนดรหัสภาษาไทย ของ export ไฟล์ .CSV 
* 0 = รหัส เกษตร 
* 1 = รหัส สมอ. (windows-tis) 
|Default|@@""" _CSVDate = 0 """@@|
กำหนด format วันที่ สำหรับ export ไฟล์ .CSV
* 0 = yyyy/mm/dd 
* 1 = dd/mm/yyyy 
* 2 = mm/dd/yyyy 
|Default|@@""" _CSVStr = 1 """@@|
กำหนดเงื่อนไขการใส่ quote ครอบข้อความ สำหรับ export ไฟล์ .CSV กรณีที่ภายในข้อความมี ตัวอักษร comma (,) หรือ pipe (|) จะใส่เครื่องหมาย double quote ครอบข้อความอัตโนมัติ แต่ถ้าไม่มีตัวอักษรดังกล่าว จะพิจารณาจากค่าที่กำหนดโดยคำสั่งนี้ 
* 0 = ไม่ต้องใส่ quote 
* 1 = ใส่ single quote (') นำหน้าข้อความ 
* 2 = ใส่ double quote (") ครอบข้อความ ทุกกรณี 
>note: ถ้าภายในข้อความมีตัว double quote โปรแกรมจะแปลงเป็น single quote 2 ถ้าภายในข้อความมีตัว return (ขึ้นบรรทัดใหม่) โปรแกรมจะแปลงเป็น tab 
|Default|@@""" _CheckBreakInterval = 2 """@@|
สำหรับการทำงานที่ต้องวน loop จำนวนมากๆ ใช้กำหนดจำนวน loop ที่ให้โปรแกรมเช็คว่า มีการกดคีย์ Esc หรือไม่ ปกติคือ เมื่อวนครบ 2 รอบ ก็จะเช็คคีย์ 1 ครั้ง ถ้ากำหนดค่านี้น้อยลงเช่น 1 จะทำให้โปรแกรมทำงานช้าลง เพราะต้องเสียเวลาเช็คคีย์บ่อย แต่ถ้ากำหนดค่านี้มากไป โปรแกรมทำงานเร็วขึ้น แต่จะทำให้การตอบสนองต่อการกดคีย์ไม่ดี 
|Default|@@""" _ChkBtr = ON """@@|
ให้โปรแกรมตรวจสอบก่อนเข้าโปรแกรมว่ามี Btrieve database manager ทำงานอยู่หรือไม่ 
ให้โปรแกรมซ่อนชื่อลูกค้า โดยกำหนดเงื่อนไขจากข้อความในช่องผู้ติดต่อ สามารถใช้ wildcard และ , (comma) เช่น @@"""_CustomerHideContact"""=*ซ่อน*,*{H}*@@
ให้โปรแกรมซ่อนชื่อลูกค้า โดยกำหนดเงื่อนไขจากข้อความในช่องหมวดลูกค้า สามารถใช้ wildcard และ , (comma) เช่น @@"""_CustomerHideCusGroup""" = ส่วนตัว,*เลิก@@
|Default|@@""" _CustomerShowQInf=-1 """@@|
(มี.ค.53) ใช้ควบคุมการแสดงข้อมูลเพิ่มเติม (QuickInfo) ที่ด้านล่างของจอ Browse 

ดูรายละเอียดที่ [[_StockShowQInf]]
|Default|@@""" _DelayFactor = 1.0 """@@|
ปกติจะมีค่าเป็น 1 ใช้เป็นตัวคูณสำหรับ ค่า delay ในการ idle โปรแกรม เพื่อหน่วงให้โปรแกรมทำงานช้าลง และแบ่งเวลา CPU ให้ process อื่นได้ทำงานมากขึ้น 

เช่น กรณี [[_DialogDelay]] กำหนดเป็น 100 millisec ถ้าเปลี่ยนค่า [[_DelayFactor]] เป็น 1.5 ก็จะกลายเป็น delay จริง 150 millisec 
|Default|@@""" _DialogDelay = 100 """@@|
ค่า default การหน่วงเวลา กรณีแสดง หน้าต่างข้อความต่างๆ ของโปรแกรม (แบบที่ไม่ต้องรอให้ผู้ใช้กดคีย์) หน่วยเป็น millisec (1/1000 วินาที) (คูณด้วย [[_DelayFactor]]) 
ใช้สำหรับการแสดงผลตัวเลข (ในรายงาน) โดยการแปลงตัวเลข (0123456789 .-+) ให้เป็นตัวอักษรตามที่กำหนด เปรียบเสมือนรหัสลับ ป้องกันไม่ให้ผู้อื่นมาอ่านค่าตัวเลขในรายงาน เช่น ใช้ปกปิดต้นทุนในรายงาน จะต้องใช้คู่กับการกำหนด [[Field Picture]] ให้เป็น #E 

ตัวอย่าง 
| [[_EncryptNumPhase]] | ตัวเลข | แสดงผล |h
| abcdefghijklmn | 325.00 | dcflaa |
|Default|@@""" _ExportLevel = 7 """@@|
กำหนดระดับผู้ใช้ ที่มีสิทธิ์เลือกให้พิมพ์ออกไฟล์ (.PRN, .TXT, .CSV) 

ค่า default 7 คือ ผู้ใช้ที่มีระดับตั้งแต่ user7 จึงสามารถสั่ง export ได้ 

| 1 | user1 |
| 2 | user2 |
| 3 | user3 |
| 4 | user4 |
| 5 | user5 |
| 6 | user6 |
| 7 | user7 |
| 8 | บัญชี |
| 9 | ผู้จัดการ |
| 10 | supervisor |
ชุดคำสั่งที่เกี่ยวข้องมีดังนี้
| [[_ExportPath1]] | [[_ExportPath2]] | [[_ExportPath3]] |
| [[_ExportPath1.PRN]] | [[_ExportPath2.PRN]] | [[_ExportPath3.PRN]] |
| [[_ExportPath1.TXT]] | [[_ExportPath2.TXT]] | [[_ExportPath3.TXT]] |
| [[_ExportPath1.CSV]] | [[_ExportPath2.CSV]] | [[_ExportPath3.CSV]] |

สำหรับกำหนด directory สำหรับเก็บไฟล์ที่ export 
[[_ExportPath1]] ใช้กรณีสั่ง export โดยตั้ง พิมพ์ออก เป็น lpt1 
[[_ExportPath2]] ใช้กรณีสั่ง export โดยตั้ง พิมพ์ออก เป็น lpt2 
[[_ExportPath3]] ใช้กรณีสั่ง export โดยตั้ง พิมพ์ออก เป็น lpt3 (server) 

''ลำดับการเลือกใช้ [[_ExportPath*]]''
# ใช้ตามคำสั่ง ที่กำหนดแยกตามชนิดของไฟล์ที่ export (@@_~ExportPath#.xxx@@) 
# ถ้าไม่มี ก็จะใช้ค่าจากคำสั่งที่ไม่ระบุชนิดไฟล์ (@@_~ExportPath#@@) 
# ถ้าไม่มี ก็จะใช้ directory ของ ชุดข้อมูล นั้นๆ เป็นที่เก็บ export ไฟล์ ยกเว้นมีคำสั่ง [[_PrintExportToTemp]] = ON ก็จะเก็บ export ไฟล์ ไว้ใน temp directory 

''ลำดับการเลือกใช้ temp directory''
# ถ้ากำหนด [[_UseLocalTmp]] = ON จะใช้ directory ที่กำหนดโดย คำสั่ง TMP หรือ TEMP ตามลำดับ ใน environment DOS ของคอมพิวเตอร์เครื่องนั้นๆ เช่น C:\WINDOWS\TEMP 
# ใช้ directory ชื่อ (btr) ที่อยู่ใน directory ของโปรแกรม scraft 
ดูรายละเอียดใน [[_ExportPath*]]
<<tiddler "_ExportPath1"">> 
<<tiddler "_ExportPath1">> 
<<tiddler "_ExportPath1">> 
<<tiddler "_ExportPath1">> 
<<tiddler "_ExportPath2">> 
<<tiddler "_ExportPath2">> 
<<tiddler "_ExportPath2">> 
<<tiddler "_ExportPath1">> 
<<tiddler "_ExportPath3">> 
<<tiddler "_ExportPath3"##"_ExportPath3">> 
<<tiddler "_ExportPath3">> 
|Default|@@""" _FCYBegin = 1/1 """@@|
กำหนด วันที่/เดือน สำหรับเริ่มรอบปีบัญชี Fiscal Year 
|Default|@@""" _GVLimitVType = ON """@@|
ใช้สำหรับกำหนดให้โปรแกรม ต้องแสดงรายการหมวดใบสำคัญ มาให้เลือก ก่อนเข้าไปทำใบสำคัญหรือไม่
ดูคำอธิบายจาก [[_InvLimitInvType]]
ใช้สำหรับซ่อนเล่มใบสำคัญ ไม่ให้เลือกใช้ตอนเข้าไปเปิดใบสำคัญลงรายการ
ดูคำอธิบายจาก [[_InvCfgHideList]]
|Default|@@""" _GuestLogin = ON """@@|
สำหรับชุดข้อมูลที่ ยังไม่มีรายชื่อ/รหัสผ่านของผู้ใช้ 
เดิม โปรแกรมรุ่นเก่า จะยินยอมให้เข้าไปใช้ชุดข้อมูลนั้นอัตโนมัติโดย ไม่มีชื่อผู้ใช้

ส่วน โปรแกรมรุ่นใหม่ กรณีเข้าไปใช้ชุดข้อมูลนั้นอัตโนมัติ จะใช้ชื่อผู้ใช้เป็น ''@@ * @@'' (ดอกจัน)
แต่เราสามารถกำหนดให้กลับไปทำงานแบบโปรแกรมรุ่นเดิม โดยใช้ค่า OFF

> ข้อแตกต่างระหว่างมีชื่อผู้ใช้ กับไม่มีชื่อผู้ใช้ อยู่ที่ความสามารถในการกำหนด """__FSys.INI""" เพื่อ customize แบบมีเงื่อนไขแตกต่างกันตามชื่อ user ได้
|Default|@@""" _InvAllowChangeDsc = ON """@@|
กำหนดระดับ user ต่ำสุด ที่ยินยอมให้แก้ไขช่อง__ส่วนลด__ต่อรายการสินค้าในบิล

<<tiddler "securityAllow">>
ค่าเพิ่มเติม
| +100 |ยกเว้น ให้แก้ไขส่วนลดของ สินค้า ที่มี * นำหน้าชื่อ |

ตัวอย่าง 
[[_InvAllowChangeDsc]]=104 เป็นการ lock ไม่ยินยอมให้ระดับที่ต่ำกว่า user4 แก้ไข แต่มีข้อยกเว้น กรณีที่สินค้านั้นมีชื่อ นำหน้าด้วย * (ดอกจัน) สามารถเข้าไปใส่ส่วนลดได้

ดูเพิ่มเติม: [[_InvDspDscNSize]] ซ่อนคอลัมน์ส่วนลด ไม่ให้ user เห็น
|Default|@@""" _InvAllowChangeNo = ON """@@|
ใช้สำหรับ lock ฟิลด์__เลขที่__บิลขาย ไม่ให้ user ที่มีระดับต่ำกว่าระดับที่กำหนด เข้าไปเปลี่ยนเลขที่ในฟิลด์ดังกล่าว ดังนั้นเลขที่จะต้องเกิดจากการ run เลขอัตโนมัติของโปรแกรมเท่านั้น 
<<tiddler "securityAllow">>
|Default|@@""" _InvAllowChangePrice = ON """@@|

กำหนดระดับ user ต่ำสุด ที่ยินยอมให้แก้ไขช่อง__ราคา__สินค้าในบิล
<<tiddler "securityAllow">>
ค่าเพิ่มเติม
| +100 |ยกเว้น ให้แก้ไขราคาของ สินค้า ที่มี * นำหน้าชื่อ |
| +200 |ยกเว้น ให้ใส่ราคาในบรรทัดที่ราคาเป็น 0 แต่ถ้าใส่ราคาแล้ว จะกลับไปแก้ไขอีกไม่ได้ |

ตัวอย่าง 
[[_InvAllowChangePrice]]=304 เป็นการ lock ไม่ยินยอมให้ระดับที่ต่ำกว่า user4 แก้ไข แต่มีข้อยกเว้น กรณีที่สินค้านั้นมีชื่อ นำหน้าด้วย * (ดอกจัน) หรือ บรรทัดที่ยังไม่ราคา สามารถเข้าไปใส่ราคาได้
|Default|@@""" _InvAllowChangeQty = ON """@@|
ใช้สำหรับ lock ฟิลด์__จำนวน__ในบิลขาย ไม่ให้ user ที่มีระดับต่ำกว่าระดับที่กำหนด เข้าไปแก้ไขจำนวนในฟิลด์ดังกล่าว 
<<tiddler "securityAllow">>
|Default|@@""" _InvAllowPostPrint = ON """@@|
ใช้ควบคุมเพื่อ lock ไม่ให้พิมพ์บิลขาย ที่ผ่านรายการแล้ว
<<tiddler "securityAllow">>
/%
| ON |อนุญาตให้พิมพ์บิลทุกเงื่อนไข (default) |
| 0 หรือ OFF |ไม่อนุญาตให้พิมพ์บิลที่ผ่านรายการแล้ว |
| 1 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายสต็อกไม่สมบูรณ์ |
| 2 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายการภาษีไม่สมบูรณ์ |
| 3 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายสต็อกหรือภาษีไม่สมบูรณ์ |
| 4 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายการลูกหนี้ไม่สมบูรณ์ |
| 5 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายสต็อกหรือลูกหนี้ไม่สมบูรณ์ |
| 6 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายภาษีหรือลูกหนี้ไม่สมบูรณ์ |
| 7 |ไม่อนุญาต ให้พิมพ์บิลที่ผ่านรายการไม่สมบูรณ์ |

กรณี ค่า 0-7 รวมถึง ไม่อนุญาตกรณีบิลที่ยกเลิก หรือยังไม่ผ่านรายการด้วย
%/
|Default|@@""" _InvAllowPrintEdit = ON """@@|
ใช้ lock ไม่ให้แก้ไขหรือลบ บิลที่พิมพ์แล้ว
<<tiddler "securityAllow">>
|Default|@@""" _InvAllowReprint=ON """@@|
สำหรับ lock ไม่ให้สั่งพิมพ์บิลที่เคยพิมพ์ไปแล้ว
<<tiddler "securityAllow">>
|Default|@@""" _InvAutoListBOM = OFF """@@|
(บิลซื้อใช้@@ [[_PoAutoListBOM]] @@)
ใช้สำหรับหน้าจอเปิดบิลขาย เวลาป้อนรายการ ส่วนประกอบสินค้าต้องขึ้นอัตโนมัติ (ไม่ต้องกด)
เพราะป้องกันลืม หรือสับสน สินค้าบางตัวมีส่วนประกอบ บางตัวไม่มี

เดิมการใส่ส่วนประกอบสินค้าในบิล จะต้องลงที่ช่องรายการบรรทัดถัดจากสินค้าหลัก ใส่@@'' *. ''@@หรือ@@'' *- ''@@แล้วกด@@'' F1 ''@@โปรแกรมจึง list ส่วนประกอบมาใส่ลงมาให้ (ถ้ามี)
> บิลซื้อจะใช้'' *. ''หรือ@@'' *+ ''@@แล้วกด'' F1 ''
| 0 หรือ OFF |ไม่ใส่ส่วนประกอบอัตโนมัติ ทำงานแบบเดิม |
| 1 |ใส่ส่วนประกอบอัตโนม้ติ เมื่อกด F1 ในบรรทัดติดกับสินค้าหลัก โดยไม่ถาม |
| 2 |ตรวจสอบส่วนประกอบ หลังจากกด F1 เลือกชื่อสินค้าหลัก แล้วขึ้นมาถาม เหมือนกับกรณี กด F1 ที่บรรทัดติดกับสินค้าหลัก |
| 3 |ตรวจสอบส่วนประกอบ หลังจากด F1 เลือกชื่อสินค้าหลัก ถ้ามีส่วนประกอบ จะใส่ให้อัตโนมัติโดยไม่ถาม |

> นอกจากนี้ยังปรับปรุงเพิ่มเติม ปกติ ในบรรทัดที่ติดกับสินค้าหลัก จะต้องใส่@@'' *. ''@@หรือ@@'' *- ''@@ก่อนกด@@'' F1 ''@@แต่ถ้ามีคำสั่ง [[_InvAutoListBOM]] ที่ไม่ใช่ 0 การกด@@'' F1 ''@@ใส่ส่วนประกอบ จะทำงาน เมื่อบรรทัดนั้นมีแค่@@'' * ''@@ตัวเดียว ได้อีกเงื่อนไขหนึ่งด้วย
|Default|@@""" _InvAutoPackItm = -1 """@@|
ใช้สำหรับยุบรวมจำนวนสินค้า สำหรับบรรทัดที่มีรายการเหมือนกันหลายบรรทัด มารวมเป็นบรรทัดเดียวกัน
>เคยเห็นตัวอย่างจากระบบบิลของ ห้างแมคโคร พนักงานเขาจะยิง barcode ไปเรื่อยๆ พอตอนปิดบิล โปรแกรมจะตรวจยุบรายการที่เหมือนกัน ให้เหลือบรรทัดเดียว 

คำสั่งนี้ มีหลักการดังนี้
ตรวจบรรทัดรายการที่มี ชื่อเหมือนกัน, หน่วยนับเหมือนกัน, ส่วนลดเหมือนกัน, ราคาต่อหน่วยเหมือนกัน ถ้าพบก็ยุบไปรวมกับบรรทัดที่อยู่ก่อน ด้วยบวกจำนวนเข้าด้วยกัน

| -1 (default) |สำหรับบิลทั่วไป ไม่ยุบรวม ยกเว้นบิลที่มีคำสั่ง ~FCB ในหมวดบิล |
| 0 หรือ OFF |ไม่ต้องยุบรวม |
| 1 หรือ ON |ให้ยุบรวม |
ใช้สำหรับกำหนดให้ โปรแกรมแสดง list ของบิลต้นแบบ มาให้เลือกก่อน ตั้งแต่ตอนกด F5 เพื่อเปิดบิลใหม่  สามารถกำหนดดังนี้

@@"""
_InvAutoReferNo=หมวดบิล [ +wildcard ของเลขที่บิลต้นแบบ ]
"""@@
* ค่าที่กำหนด หมวดบิล 2 ตัวอักษรแรก ห้ามเป็น wildcard
* ถ้าไม่เจอบิลต้นแบบ ตาม wildcard ที่กำหนด โปรแกรมจะไม่แสดง list
* เมื่อตั้งต้นแบบแล้ว สามารถลบหมวดบิลต้นแบบออกไป เพื่อไม่ให้ผู้ใช้เห็นหมวดบิลดังกล่าว และเข้าไปแก้ไข เปลี่ยนแปลงบิลต้นแบบ
* สามารถกำหนดหมวดบิลเป็น ?? เช่น ??01* แปลว่า ต้นแบบใช้หมวดบิลเดียวกับ หมวดบิลที่กำลังเปิดบิลอยู่
* กรณี ค่าที่กำหนดไม่ใช่ wildcard โปรแกรมจะไม่แสดง list และ copy ต้นแบบมาใส่ให้อัตโนมัติ
* กรณี ค่าที่กำหนดลงท้ายด้วย & โปรแกรมจะไม่แสดง list ให้เลือก และ copy ต้นแบบเลขที่สูงสุด ตามเงื่อนไข wildcard มาใส่ให้อัตโนมัติ
ตัวอย่าง
| [[_InvAutoReferNo]] | คำอธิบาย |h
|TT |ให้แสดง list ของบิลต้นแบบ หมวด TT |
|~TT01* |ให้แสดง list ของบิลต้นแบบ หมวด TT ที่มีเลขที่ขึ้นต้นด้วย 01 |
|~TT00000000 |ใช้ บิลหมวด TT เลขที่ 00000000 มาใส่เป็นต้นแบบอัตโนมัติ |
|TT& |ใช้ บิลหมวด TT เลขที่สูงสุด มาใส่เป็นต้นแบบอ้ตโนมัติ |
|~TT01*& |ใช้ บิลหมวด TT เลขที่สูงสุด ที่ขึ้นต้นด้วย 01 มาใส่เป็นต้นแบบอ้ตโนมัติ |
|??000000 |ใช้ บิลหมวดเดียวกันกับบิลที่เปิด  เลขที่ 00000000 มาใส่เป็นต้นแบบอัตโนมัติ |
|TT& |ใช้ บิลหมวดเดียวกันกับบิลที่เปิด เลขที่สูงสุด มาใส่เป็นต้นแบบอ้ตโนมัติ |
<<<
เดิมการทำงานใช้บิลต้นแบบ สามารถทำได้ 2 วิธีคือ 
* ค้นหาบิลต้นแบบที่ต้องการ กด Space ให้แสดงรายละเอียด แล้วกด F5 ครั้งแรกเพื่อเพิ่มบิลใหม่ กด F5 ครั้งที่สอง เพื่อคัดลอกบิลที่แสดงรายละเอียด มาใส่ลงในบิลใหม่
* เลือกต้นแบบผ่านช่อง อ้างถึง โดยจะต้องเข้าไปเพิ่มบิลใหม่ก่อน แล้วไปที่ช่องอ้างถึง กด F1 เลือกหมวด+เลขที่บิลที่ต้องการ แล้วโปรแกรมก็จะถาม ให้คัดลอกรายละเอียดของบิลนั้นมาใส่
<<<

ดูเพิ่มเติม: [[_InvAutoReferStatus]] กำหนดเงื่อนไขเพิ่มเติม สำหรับสถานะของบิลที่ใช้เป็นต้นแบบ (ต.ค.52)

<<<(มิ.ย.53)
ปรับปรุงให้สามารถใช้ตัวแปร &Y&M ผสมกับ wildcard ได้ เช่น
[[_InvAutoReferNo]].IV=~V&Y&M*

นอกจากนี้ยังสามารถใช้ &y&m สำหรับตัวแปรเลข ปีเดือน ของเดือนที่แล้ว เช่น
[[_InvAutoReferNo]].IV=~V&y&m*

ใช้ประโยชน์กรณี ใบแจ้งหนี้ กิจการอพาร์ตเมนต์
ที่ต้องการอ้างถึงต้นแบบจากใบแจ้งหนี้เดือนที่แล้ว มาใช้

หมายเหตุ กรณี &y ไม่ได้หมายถึงเลขปีของปีที่แล้ว ไม่ได้หมายถึงเลขปี ของเดือนที่แล้ว
ดังนั้น กรณีปัจจุบันคือ ปีเดือน 5401 เลขปีเดือนของเดือนที่แล้วคือ 5312
สำหรับตัวแปรเลขปี ของปีที่แล้วนั้น ไม่มีให้ใช้<<<
ใช้สำหรับซ่อนหมวดบิล ไม่ให้เลือกใช้ตอนเปิดบิล 
สามารถกำหนดได้ดังนี้
@@"""
_InvCfgHideList = หมวดบิล1[,หมวดบิล2,..]
"""@@

ค่าหมวด สามารถกำหนดเป็น wildcard ก็ได้ เช่น
[[_InvCfgHideList]]=?2,SX,Q?
แปลว่า ซ่อนหมวดบิลที่ลงท้ายด้วย 2, หมวดบิล SX และหมวดบิลที่ขึ้นต้นด้วย Q

ผลของการซ่อนหมวดบิล ทำให้ไม่สามารถเข้าไปดูรายการบิลของหมวดนั้น จึงไม่สามารถแก้ไข ลบ หรือเพิ่มบิลของหมวดนั้นได้ มีผลเหมือนกับการ lock ข้อมูลของบิลหมวดนั้น

> เราสามารถสั่งซ่อนหมวดบิล ได้อีกวิธีหนึ่ง โดยใส่@@'' {H} ''@@ไว้ที่ข้อมูลหมวดบิล ในส่วนของชื่อบิล

ดูเพิ่มเติม: [[_InvLimitInvType]] [[_InvCustomListType]]
{{{
ขออภัย อธิบายผิดพลาดไปนิด
สรุปดังนี้
คำสั่งนี้มีผล ในจังหวะที่กด F10 Save ข้อมูล
ดังนี้จึงสามารถใช้กับบิลที่ ไม่ได้ผ่านรายการอัตโนมัติด้วย ~L

ถ้ากำหนด =OFF หรือ =0 จะมีผลคือ ไม่ฟ้องเตือนในขั้นตอน Save
ดังนั้น ถ้ากำหนด =1 คือ ยินยอมให้ save แต่ฟ้องเตือนตั้งแต่ user1
ถ้ากำหนด =10 คือ ยินยอมให้ save เฉพาะ supervisor ส่วนระดับต่ำลงมาฟ้อง error และไม่ยอมให้ save
ถ้ากำหนด =ON จะมีค่าเท่ากับ =9 ยินยอมให้ save ตั้งแต่ ผู้จัดการ ขึ้นไป

ถ้ากำหนด _InvChkCusLimit ให้ตรวจสอบในขั้นตอน F10 Save
จะมีผลกระทบ ทำให้โปรแกรม ไม่ตรวจจสอบวงเงิน ในขั้นตอนผ่านรายการซ้ำซ้อน

กลับกัน
ถ้ากำหนด =OFF หรือ =0 หรือไม่ได้กำหนด _InvChkCusLimit เลย
จะมีผลให้โปรแกรม ตรวจวงเงิน ในขั้นตอน ผ่านรายการแทน
และบันทึก error การผ่านรายการไว้ด้วย
ยกเว้น จะมีการกำหนด _PInvChkCusLimit 
เพื่อควบคุมแทน เช่น

ถ้าเราไม่ต้องการให้ตรวจสอบ วงเงินเลย
จะต้องกำหนด
_PInvChkCusLimit=OFF
_InvChkCusLimit=OFF (หรือไม่กำหนดเลย)

ถ้าเราต้องการให้ตรวจสอบ ทั้ง 2 จังหวะ คือเมื่อ F10 Save และเมื่อผ่านรายการ
จะต้องกำหนด
_InvChkCusLimit=ON
_PInvChkCusLimit=ON
}}}

{{{
ปรับปรุงล่าสุด 7-6-49

สำหรับลูกค้าที่มีการ ระบุยอดวงเงิน ในข้อมูลเพิ่มเติม
และบิลหมวดนั้น มีการระบุให้ผ่านรายการลูกหนี้

1. กรณีไม่ระบุอะไรเลย
(มีผลเหมือนกับระบุ _PInvChkCusLimit = ON)

2. กรณีระบุ _InvChkCusLimit แต่ไม่ระบุ _PInvChkCusLimit
โปรแกรมจะตรวจวงเงิน ในขั้นตอน F10 Save และจะไม่ตรวจสอบซ้ำในขั้นตอนผ่านรายการ
สามารถกำหนดเป็นค่า level ของ user ที่ยินยอมให้ผ่าน 
โดยโปรแกรมฟ้องเตือน ให้ตอบยืนยัน
หรือ กำหนดเป็น ON ซึ่งมีผลเหมือนกับกำหนดค่า _InvChkCusLimit=9 
ยอมให้ผู้ใช้ระดับตั้งแต่ ผู้จัดการ ขึ้นไป สามารถ Save บิลที่เกินวงเงินได้

ส่วนผู้ใช้ที่ระดับต่ำกว่า ที่กำหนด
ถ้ามีการกำหนดรหัสอนุมัติไว้ที่ชื่อผู้ใช้ ~CLIMIT 
โปรแกรมจะขึ้นมาถาม รหัสอนุมัติ ก่อนที่จะยอมให้ Save

3. กรณีระบุ _PInvChkCusLimit
ล่าสุดสามารถกำหนดเป็นค่า level ของ user ที่ยินยอมให้ผ่าน
หรือกำหนดเป็น ON ซึ่งมีผลเหมือนกับ _PInvChkCusLimit=9
นั่นคือ ยอมให้ user ระดับตั้งแต่ระดับ ผู้จัดการ ขึ้นไป ยืนยันผ่านยอดเกินวงเงินได้

ส่วนผู้ใช้ที่ระดับต่ำกว่า ที่กำหนด
ถ้ามีการกำหนดรหัสอนุมัติไว้ที่ชื่อผู้ใช้ ~CLIMIT 
โปรแกรมจะขึ้นมาถาม รหัสอนุมัติ ก่อนที่จะยอมให้ทำงานผ่านรายการต่อไป 
}}}

{{{
11/6/2551

โปรแกรมเดิม มีการตรวจสอบ ยอดหนี้ล่วงหน้า ที่เกิดจากการเปิดบิลล่วงหน้า 
โดยแสดงในคำเตือน แต่ยอดล่วงหน้าดังกล่าว ถูกบวกกลับ เสมือนหนี้นั้นยังไม่เกิดขึ้น 
ดังนั้นผลที่เกิดขึ้นกับบิลที่มีการแก้ไข ย้อนหลัง 
โปรแกรมจะใช้ยอดหนี้จนถึงวันที่ตามบิลนั้น 
ไม่เอายอดหนี้ของบิลที่ลงวันที่หลังจากนั้น 

ปรับปรุงโปรแกรมใหม่ แยกโปรแกรมเป็น 3 แบบ
ย้ายส่วน support _InvChkCusLimit ไปเป็นงาน MODIFY
เพื่อลดขนาดโปรแกรมลง
และทำเพิ่ม ส่วนเพิ่มบวกยอดหนี้จากเช็คที่ยังไม่เคลียร์/เด้ง

_Inv1_  ไม่มีการตรวจสอบเพื่อบวกกลับหนี้ล่วงหน้า จะใช้ยอดหนี้ล่าสุดเสมอ ทำงานเร็ว และไฟล์ .EXE มีขนาดเล็กกว่าเดิม สำหรับผู้ที่ไม่ใช้ _InvChkCusLimit
_Inv2_ มีบวกกลับยอดหนี้ล่วงหน้า เหมือนโปรแกรมรุ่นเดิม
_Inv3_ มีบวกกลับยอดหนี้ล่วงหน้า และบวกเพิ่มยอดหนี้จากเช็คที่ยังไม่เคลียร์/เด้ง ไฟล์ .EXE ขนาดใหญ่กว่า

ดังนั้น default ของโปรแกรมส่วนบิลขายจะเปลี่ยนไป 
ผู้ที่ต้องการใช้ _InvChkCusLimit จะต้องเข้าไป set ใน __FSYS.INI ว่า
_Inv1_=_Inv2_ สำหรับตรวจวงเงินแบบเดิม
หรือ 
_Inv1_=_Inv3_ สำหรับตรวจวงเงินแบบรวมยอดเช็คไม่เคลียร์
}}}
ให้โปรแกรมหา maxrunno ระหว่างหมวดบิลใน [[Work Flow]] เพื่อแก้ปัญหา กรณี move บิลจาก หมวด A ไป หมวด B จะเกิดปัญหา runno ของ หมวด A กลับไปใช้เลขเดิม เนื่องจาก เลขบิลล่าสุด ถูก move หายไปแล้ว ดังนั้นถ้าให้โปรแกรมใช้ runno ที่มากที่สุด ของทั้งหมวด A และ B ก็จะทำให้สามารถ หมวด A run ได้ถูกต้อง

@@"""
_InvCrossRunNo.หมวดบิล=หมวดบิล,หมวดบิล,...
"""@@
@@"""
_PoCrossRunNo.หมวดบิล=หมวดบิล,หมวดบิล,...
"""@@

การทำงาน
โปรแกรมจะหา runno ของหมวดบิลที่ใช้อยู่ก่อน เหมือนปกติ หลังจากนั้น จะตรวจสอบคำสั่ง ~CrossRunNo ถ้ามี ก็จะหา runno ของหมวดบิล ตามที่ระบุ และเลือกใช้ runno ที่เป็นเลขมากที่สุด

> ตัวอย่าง 
> [[_InvCrossRunNo]].I0=I1,I2
ใช้สำหรับเปลี่ยนรายการหมวดบิล ที่ใช้เลือกตอนก่อนเข้าไปเปิดบิล ซึ่งเรียงตามตัวอักษร ให้เป็นเมนูให้เลือกหมวดบิล ที่จัดลำดับหมวดบิลตามต้องการได้
สามารถกำหนดได้ดังนี้
@@"""
_InvCustomListType = หมวดบิล1,หมวดบิล2,...
"""@@

เมนูที่ขึ้นมาให้เลือก จะเรียงตามลำดับรายการหมวดบิลที่กำหนดนั้น ช่วยให้การใช้งานเปิดบิลสะดวกยิ่งขึ้น เนื่องจากเราสามารถลำดับ ให้หมวดบิลที่ใช้บ่อย ขึ้นมาอยู่เป็นรายการแรก นอกจากนี้ยังช่วยซ่อนหมวดบิลที่ไม่ได้ใช้งานแล้ว ให้เหลือแต่หมวดบิลที่ใช้งานอยู่ปัจจุบัน

* ค่าหมวดบิลที่กำหนด ไม่สามารถใช้ wildcard 
* สามารถใส่ ,* ต่อท้ายรายการหมวดบิล มีผลให้สามารถเข้าไปเลือกหมวดบิลทั้งหมด โดยใช้รายการหมวดบิลแบบเดิมได้อีกด้วย ตัวอย่างเช่น [[_InvCustomListType]]=IN,TI,*

(ปรับปรุง ส.ค.52)
สามารถกำหนดเงื่อนไข filter และข้อความที่ต้องการแสดงต่อท้ายหมวดบิล เพื่อให้แสดงเฉพาะบิลที่มีสถานะตามเงื่อนไขที่กำหนด ดังนั้น เราสามารถสร้างเมนูเพื่อให้เลือก บิลหมวดเดียวกัน แต่แยกเป็นบิลที่ชำระแล้ว กับบิลที่ค้างชำระก็ได้ เป็นต้น
@@"""
_InvCustomListType = หมวดบิล1[:filter ข้อความ],หมวดบิล2[:filter ข้อความ],...
"""@@

รหัสสำหรับเงื่อนไข filter
| กลุ่ม | รหัสกลุ่ม | flag | รายละเอียด |h
| การผ่านรายการ | L | - |ไม่ผ่านรายการ |
|~|~| * |ผ่านรายการแล้ว |
|~|~| e |error |
|~|~| ? |รอคืนรายการ |
|~|~| c |ยกเลิก |
|~|~| x |ไม่ยกเลิก มีค่าเท่ากับกำหนด -*e? |
|~|~| @ |ผ่านรายการขั้นสอง (ใบเสร็จ,ใบจ่าย) |
| การพิมพ์ | P | - |ยังไม่พิมพ์ |
|~|~| * |พิมพ์แล้ว |
| การชำระ | R | - |ไม่ชำระ/ยังไม่วางบิล |
|~|~|  n |วางบิลแล้ว มีค่าเท่ากับ ud |
|~|~| * |ชำระแล้ว มีค่าเท่ากับ sfo |
|~|~| s |ชำระแล้ว/ไม่ครบ some paid |
|~|~| f |ชำระแล้ว/ครบ full paid |
|~|~| o |ชำระแล้ว/เกิน over paid |
|~|~| u |วางบิลแล้ว/ยังไม่นัด unknown due |
|~|~| d |วางบิลแล้ว/นัดแล้ว already due |

ตัวอย่าง
[[_InvCustomListType]]=IN'':- (ไม่ผ่านรายการ)'',IN'':* (ผ่านรายการ)'',TI,*'':e (error)''

เราสามารถกำหนดเงื่อนไข filter โดยใช้ @@'' : ''@@ ต่อท้ายหมวดบิลที่กำหนด ตามด้วย รหัสกลุ่ม ''L'' หรือ ''P'' หรือ ''R'' (ต้องใช้ตัวอังกฤษใหญ่) ตามด้วย flag ที่ต้องการ (ใช้ตัวอังกฤษเล็ก) และเราสามารถกำหนดหลายกลุ่มในคราวเดียวกัน เช่น
* ''""":Le?"""'' (ผ่านรายการ error ''หรือ'' รอคืนรายการ)
* ''""":Le?R*"""'' (ผ่านรายการ error ''หรือ'' รอคืนรายการ ''และ'' ชำระแล้ว) 
> สังเกต logic ภายในกลุ่มจะเป็น ''หรือ'' แต่ logic ระหว่างกลุ่มจะเป็น ''และ''

* default ถ้าไม่ได้กำหนดรหัสกลุ่ม โปรแกรมจะ default เป็นกลุ่ม L เช่น TI:- คือ บิลที่ไม่ผ่านรายการ
* เงื่อนไขสำหรับหมวดบิลใดๆ ใช้ ''*:เงื่อนไข'' ซึ่งต้องเข้าไปเลือกหมวดบิล จากรายการหมวดบิลแบบเดิมอีกชั้นหนึ่ง เสร็จแล้วก็จะได้จอรายการบิล ที่แสดงเฉพาะตามเงื่อนไข filter ที่กำหนด
* เราจะต้องใส่เงื่อนไขทั้งหมดติดกันโดยไม่มีเคาะวรรค หลังจากที่เคาะวรรคจะเป็นข้อความที่เราต้องการให้ขึ้นในเมนู custom list เช่น TI:- (ไม่ผ่าน)
* สำหรับเงื่อนไข กลุ่มพิมพ์ และ กลุ่มชำระ จะมีผลกับบิลที่ไม่ยกเลิกเท่านั้น ดังนั้นถ้ามีการกำหนดให้ filter เงื่อนไขใน 2 กลุ่มนี้ บิลที่ยกเลิก จะไม่แสดงออกมาอัตโนมัติอยู่แล้ว เช่น :Lx มีผลเท่ากับ :P-*
|Default|@@""" _InvDspDscNSize = 8 """@@|
สำหรับบังคับค่า ขนาด(กว้าง) ของคอลัมน์__ส่วนลด__ ในจอบิลที่ป้อนข้อมูล กำหนดค่าได้ 0 - 10
* ถ้ากำหนดเป็น 0 จะซ่อนคอลัมน์นี้ ไม่ให้ผู้ใช้เห็น มีผลให้ไม่สามารถเข้าไปแก้ หรือใส่ข้อมูลช่องนี้ แต่โปรแกรมก็ยังเห็น ถ้ามีข้อมูลอยู่แล้ว ก็จะนำไปคำนวณมูลค่า หรือใช้พิมพ์ออกมาได้ด้วย
* ถ้าลดขนาดของช่องนี้ จะมีผลให้ ช่องรายการสินค้าขยายกว้างยิ่งขึ้น
ดูเพิ่มเติม: [[ปรับขนาด column หน่วยนับ, ส่วนลด ในจอบิล]]
|Default|@@""" _InvDspUnitNSize = 4 """@@|
สำหรับบังคับค่า ขนาด(กว้าง) ของคอลัมน์__หน่วยนับ__ ในจอบิลที่ป้อนข้อมูล กำหนดค่าได้ 0 - 12
* ถ้ากำหนดเป็น 0 จะซ่อนคอลัมน์นี้ ไม่ให้ผู้ใช้เห็น มีผลให้ไม่สามารถเข้าไปแก้ หรือใส่ข้อมูลช่องนี้ แต่โปรแกรมก็ยังเห็น ถ้ามีข้อมูลอยู่แล้ว ก็จะนำไปคำนวณมูลค่า หรือใช้พิมพ์ออกมาได้ด้วย
* ถ้าหดขนาดของช่องนี้ จะมีผลให้ ช่องรายการสินค้าขยายกว้างยิ่งขึ้น

ดูเพิ่มเติม: [[ปรับขนาด column หน่วยนับ, ส่วนลด ในจอบิล]]
{{{
10/3/2551

เดิมค่า default ของ _InvGetLastPrice=2
มีผลให้กรณีที่ สินค้าที่ไม่มีราคาขาย ในข้อมูลสต็อก/ตารางราคา
เมื่อเปิดบิล โปรแกรมจะเอาราคาที่เคยขายในบิลหมวดนั้น+ของลูกค้ารายนั้น มาใช้อัตโนมัติ

แก้ไขใหม่ ค่า default เป็น _InvGetLastPrice=-2
กรณีนี้ เมื่อเปิดบิล โปรแกรมจะไม่เอาราคาที่เคยขายมาใช้อัตโนมัติ
ดังนั้นในช่องราคา จะมีค่าเป็น 0 
แต่ ถ้ากด F1 ในช่องราคา จะขึ้น list ของราคาที่เคยขายมาให้เลือก

ข้อดีคือ เมื่อเปิดบิล เลือกชื่อสินค้า โปรแกรมจะไม่เสียเวลาหน่วงค้นหาบิลที่เคยขาย

นอกจากนี้ยังแก้ไขข้อผิดพลาด
กรณีที่ สินค้าที่ไม่มีราคาขาย ในข้อมูลสต็อก/ตารางราคา

เดิม เมื่อมีการเปลี่ยนแปลงจำนวนที่ขายในบิล
โปรแกรมจะพยายาม sync เอาราคาที่เคยขาย หรือราคา 0 มาเปลี่ยนใส่ให้

ใหม่ โปรแกรมจะพยายาม sync ราคาใหม่ตามจำนวนที่เปลี่ยนไป
ถ้ามีราคาในข้อมูลสต็อก/ตารางราคาเท่านั้น
ดังนั้นถ้าไม่มีราคาขายจากข้อมูลสต็อก/ตารางราคา
ราคาในหน้าบิล จะไม่เปลี่ยนแปลง


}}}
|Default|@@""" _InvLimitInvType = ON """@@|
ใช้สำหรับกำหนดให้โปรแกรม ต้องแสดงรายการหมวดบิล มาให้เลือก ก่อนเข้าไปเปิดบิลหรือไม่

| 0 หรือ OFF |ปิด ไม่ต้องแสดงรายการหมวดบิลมาให้เลือก ดังนั้นจึงเข้าไปในจอบิล แบบแสดงบิลรวมทุกหมวด |
| 1 หรือ ON |เปิด แสดงรายการหมวดบิล มาให้เลือก ก่อนเข้าไปเปิดบิล สามารถกด Esc ไม่เลือกหมวดบิล เพื่อเข้าจอบิล แบบแสดงบิลรวมทุกหมวด |
| -1 |บังคับ แสดงรายการหมวดบิล มาให้เลือก ไม่สามารถกด Esc เพื่อเข้าจอบิลรวมทุกหมวด เราสามารถใช้ค่านี้ ร่วมกับคำสั่งซ่อนหมวดบิล [[_InvCfgHideList]] เพื่อ lock ซ่อนข้อมูลบิลหมวดที่ต้องการได้ |
|Default|@@""" _InvLockBeforePrint = OFF """@@|
อ้างอิงจากคำสั่ง [[_InvAllowPrintEdit]]=OFF
ที่ใช้สำหรับ lock ไม่ให้แก้ไขบิลที่พิมพ์แล้ว

ปรากฏว่าเกิดช่วงสูญญากาศ
ระหว่างที่ user เครื่องหนึ่ง เปิดบิลและ save ไปแล้ว
กำลัง preview บิลที่จะสั่งพิมพ์
ซึ่งจังหวะนั้น โปรแกรมยังไม่ได้เปลี่ยนสถานะของบิลนั้น ว่าพิมพ์แล้ว
เนื่องจาก user อาจกด Esc เพื่อยกเลิก ไม่ได้สั่งพิมพ์จริงก็ได้

ขณะเดียวกัน ระหว่างที่ preview นั้น
ปรากฏว่า user อีกเครื่องหนึ่งเข้าไปแก้ไขบิลใบเดียวกันนั้นได้

ก็เลยเกิดปัญหา อ้าว... บิลที่สั่งพิมพ์ออกมา กับบิลที่ save (ถูกแก้ไขโดยอีก user)
มีรายการไม่ตรงกัน ทั้งๆ ที่มีการ lock ไม่ให้ใครแก้ไข

ถ้ากำหนด [[_InvLockBeforePrint]]=ON
เมื่อสั่งพิมพ์ โปรแกรมจะเปลี่ยนสถานะของบิลเป็นพิมพ์แล้ว ก่อนที่จะ preview
ดังนั้น อีกเครื่องหนึ่งก็จะติดเงื่อนไข [[_InvAllowPrintEdit]] ไม่สามารถแก้ไข
และถ้า user ที่สั่งพิมพ์นั้นตัดสินใจยกเลิก กด Esc ไม่ได้สั่งพิมพ์
โปรแกรมก็จะเปลี่ยนสถานะพิมพ์แล้วกลับมากลายเป็น ยังไม่พิมพ์
{{{
ใช้สำหรับ ให้ซ่อนไม่ให้ผู้ใช้เห็น บิลขาย วันที่ย้อนหลัง โดยสามารถกำหนดจำนวนวันที่ยินยอมให้ผู้ใช้เห็นวันที่ย้อนหลังได้ เช่น

_InvLockHiDate = 0
หมายความว่า สามารถเห็นบิลขายที่เปิดไปแล้วย้อนหลังได้ 0 วัน แปลว่าเห็นได้แค่บิลของวันนี้เท่านั้น

และเช่นเดียวกับคำสั่ง เกี่ยวกับควบคุมบิลขายอื่นๆ สามารถเจาะจงหมวดบิลได้เช่น

_InvLockHiDate.T1 = 1
หมายความว่า สำหรับบิลหมวด T1 ให้เห็นบิลย้อนหลังได้แค่ 1 วัน

ถ้ากำหนดให้เป็นค่าติดลบ เช่น -1 ก็จะกลายเป็นว่า บิลที่เปิดไปแล้ว user จะไม่สามารถมองเห็น

สามารถใช้กับระบบที่ออกแบบ ในลักษณะของ POS (Point Of Sale) ป้องกันไม่ให้ user ย้อนเข้าไปแก้ไขหรือเปลี่ยนแปลงบิลเก่าที่เปิดไปแล้ว

คำสั่งนี้ ไม่มีผลกับ user ที่มีระดับเป็น supervisor

update สิงหาคม 2009
สำหรับบิลซื้อก็ใช้ได้เหมือนกัน
_PoLockHiDate

เพิ่มคำสั่ง LockHiDate สำหรับ ใบเสร็จและใบจ่าย ตามนี้

_RcpLockHiDate ใช้กับ "วันที่" ของใบเสร็จ
_MrcLockHiDate ใช้กับ "วันที่" ของใบจ่าย

_RcpLockHiPDate ใช้กับ "วันที่นัด/ชำระ" ของใบเสร็จ
_MrcLockHiPDate ใช้กับ "วันที่นัด/ชำระ" ของใบจ่าย
}}}
{{{
คำสั่ง _InvSaleAsLogin มีการปรับปรุงใหม่ เมื่อ 15-3-48 ดังนี้

ส่วนของค่าที่เหมือนเดิมคือ
= 1 (หรือ ON) ใส่ชื่อ user ที่ login ลงใน field พนักงานขาย
= 2 ใส่ชื่อ user ที่ login ลงใน field หมายเหตุ
= 3 ใส่ชื่อ user ที่ login ลงใน field ชื่อสรุปบิล
= 4 ใส่ชื่อ user ที่ login ลงใน field ส่งของ
= 5 ใส่ชื่อ user ที่ login ลงใน field อ้างถึง
= 10 ให้ copy ชื่อพนักงานขายจากข้อมูลบิลครั้งก่อนมาใส่

ส่วนที่เพิ่มใหม่คือ 
ใช้ = 9 แทน = 10 ให้ copy ชื่อพนักงานขายจากข้อมูลบิลครั้งก่อนมาใส่

สำหรับค่า 1 - 5 ถ้า +20 เช่น =21 จะเป็นการใส่ เวลาปัจจุบันต่อท้ายไปด้วย และถ้า +40 เข้าไป จะเป็นการใส่ วันที่+เวลา ปัจจุบันต่อท้าย

ค่าดังกล่าวถ้าใช้เป็นค่าติดลบ เช่น = -21 ก็จะเป็นการสั่ง lock field ไม่ให้เข้าไปแก้ไข
}}}
| รอเรียบเรียง |
{{{
23/9/51

- ผลสรุปจากการประชุม เรื่อง ปัญหา การป้อนใบ OF (ป้อนจากเอกสาร Sale)
แล้ว ป้อน ราคาสินค้า ผิด เพราะโปรแกรม ดึงราคาเก่าที่เคยเปิดมาให้ (มีการเปลี่ยนราคา)
ผู้เปิด ลืม แก้ไข ราคา สรุป ให้โปรแกรม ขึ้นกรอบ ยืนยัน ยอดรวมของบิลก่อนถึงจะ Save ได้ โดยตอนถามให้ยืนยันห้ามโชว์ ยอดรวมให้ผู้ใช้ดู
(ผิดพลาดได้ไม่เกิน 1 บาท) รบกวนโปรแกรมเมอร์ช่วย พัฒนาด้วยครับ ลูกค้าขอด่วน
}}}

{{{
25/9/51

ทำโปรแกรมเพิ่ม ดังนี้
ใช้ _InvSaveReconcile=ON
กรณี เจาะจงหมวดบิล ก็ใช้ _InvSaveReconcile.XX=ON
XX คือ หมวดบิล

หรือใช้ _InvSaveReconcile=2
ให้ เทียบกับ ยอดมูลค่าสินค้า ไม่รวม VAT ครับ

note: ไฟล์ที่ update คือ _Inv*.exe 
}}}

{{{
อ้างจาก: adm ที่ ตุลาคม 30, 2008, 03:09:00 am
30/10/51

- Work ครับ ติดตั้งให้ลูกค้า ทดสอบ มานานพอควรแล้ว วันนี้ผมเข้าไปที่ Cinnamon
ลูกค้าแจ้ง ขอปรับสักนิดได้ไหมครับ คือขอให้ ยืนยันยอด ทุกครั้ง ที่ Save เพราะมีกรณี
ไปแก้ไข แล้วโปรแกรมไม่เช็ค (เพราะ Save ได้แล้ว) ซึ่งผมแจ้ง ลูกค้าแล้ว ว่าการแก้ไข
ีมีหลายกรณี อาจจะไม่ได้แก้ไข ยอดเงินก็ได้ แล้วยืนยันทุกครั้ง จะไม่รำคาญ เหรอ 
ซึ่งลูกค้าก็ยังยืนยัน ที่จะให้ ยืนยัน ทุกครั้ง เพราะกลัวพนักงานทำงานหลุด รบกวนแ้ก้ไขให้อีกครั้งนะครับ



1/11/2551

ปรับโปรแกรม _Inv*.EXE
กรณีต้องการให้ตรวจตอน แก้ไขด้วย ให้บวก 100 เข้าไป
ใช้ _InvSaveReconcile==101 หรือ _InvSaveReconcile=102
}}}
คำสั่งต่อเนื่องจาก Work Flow
@@"""
_InvWFAutoCmd=ชื่อไฟล์.WFC
"""@@
สำหรับกำหนดให้ ทำงานคำสั่ง Work Flow อัตโนมัติ
เมื่อมีการกด F10 เพื่อ save หลังจาก เพิ่ม (F4) หรือ แก้ไข (F5)

ดูคำอธิบายที่ [[Work Flow]]
custom Work Flow Menu (ไฟล์ .INI หรือ .MNU)
กำหนดใน """__FSys.INI"""
@@"""
_InvWFMenu=ชื่อไฟล์ [.INI หรือ .MNU]
"""@@
หรือ
@@"""
_InvWFMenu.หมวดบิล=ชื่อไฟล์ [.INI หรือ .MNU]
"""@@
ภายในเมนูไฟล์ ใช้ format เหมือนกับเมนูรายงาน ฯลฯ เช่น
<<<
ไฟล์คำสั่ง1.WFC=ข้อความที่แสดงในเมนู1 
ไฟล์คำสั่ง1.WFC=ข้อความที่แสดงในเมนู1 
<<<

ดูคำอธิบายที่ [[Work Flow]]
{{{
สำหรับโปรแกรม update ล่าสุด 18-4-48

เป็นคำสั่งสำหรับเพิ่มความฉลาดและความยืดหยุ่นให้โปรแกรม
ผู้ใช้สามารถป้อนรหัสลูกค้า หรือ รหัสพนักงาน
ในบรรทัดรายการ บรรทัดไหนของบิลก็ได้

โปรแกรมจะตรวจสอบ กรณีที่ไม่พบว่าเป็นข้อมูลสินค้า
ก็จะตรวจสอบว่า เป็นรหัสลูกค้าหรือไม่

ถ้าเป็นรหัสลูกค้า ก็จะย้ายข้อความบรรทัดดังกล่าวมาใส่ที่ฟิลด์ลูกค้าให้อัตโนมัติ


ถ้าไม่ใช่ ก็จะตรวจสอบต่อว่าเป็นรหัสพนักงานหรือไม่
ถ้าเป็นรหัสพนักงาน ก็จะย้ายข้อความบรรทัดดังกล่าวมาใส่ที่ฟิลด์พนักงานให้อัตโนมัติ

ความหมายของค่าต่างๆ ที่ใช้
 1 = ตรวจสอบรหัสกับข้อมูลลูกค้า
 2 = ให้โปรแกรมถามยืนยัน ก่อนใส่ข้อมูลลูกค้า
 4 = ตรวจสอบรหัสกับข้อมูลพนักงาน
 8 = ให้โปรแกรมถามยืนยัน ก่อนใส่ข้อมูลพนักงาน

วิธีใช้คือ เอาค่าดังกล่าวมาบวกกัน เช่น
_InvXCodeItm = 3  แปลว่าให้ตรวจสอบเฉพาะรหัสลูกค้า และต้องถามยืนยันก่อนเติมฟิลด์ลูกค้า

_InvXCodeItm = 5  แปลว่าให้ตรวจสอบทั้งรหัสลูกค้าและรหัสพนักงาน ถ้าตรงให้ย้ายไปใส่อัตโนมัติทันที โดยไม่ต้องรอถามยืนยัน

พิเศษ สำหรับบิลที่กำหนด ~FCB ปกติ ไม่ต้องกำหนด _InvXCodeItm ก็จะทำงานตรวจสอบรหัสลูกค้าและพนักงานให้อัตโนมัติ
}}}
|Default|@@""" _LOGLevel = 1 """@@|
กำหนดความลึกของการเก็บรายละเอียด LOG การทำงาน ถ้าเป็น 0 หรือ OFF คือไม่เก็บ LOG สามารถกำหนดได้ตั้งแต่ 1 ถึง 10 

> LOG file คือ hidden file .LOG เก็บอยู่ใน directory ของชุดข้อมูลนั้นๆ แยกตามวัน โดยตั้งชื่อ ใช้เลข ปีเดือนวัน (yymmdd).LOG เราสามารถใช้โปรแกรม text editor เช่น notepad เปิดไฟล์เพื่อดูได้
|Default|@@""" _LeaveTmpFile = OFF """@@|
กำหนดให้โปรแกรม ปล่อย temp file (.TMP) ที่สร้างในระหว่างประมวลผลรายงานทิ้งไว้ โดยไม่ต้องลบ เมื่อจบรายงาน 

ใช้ประโยชน์ในการ debug โปรแกรม ตรวจสอบข้อมูลที่ได้ระหว่างประมวลผลรายงาน 
|Default|คำสั่งนี้ยกเลิกไปแล้ว|
เป็นคำสั่งสำหรับ แปลงข้อความของโปรแกรมให้เป็นคำอื่น เช่น แปลเป็นอังกฤษ โปรแกรมปัจจุบันถอดออกไปแล้ว เนื่องจากมีปัญหากิน Memory มาก 
|Default|@@""" _LowDate = 0/0/0 """@@|
กำหนดวันที่ต่ำสุดของข้อมูลในช่อง ที่ยินยอมให้ user เข้าไปใส่

โดยถ้า user ใส่วันที่ในช่องดังกล่าว ก่อนวันที่ที่กำหนด โปรแกรมจะฟ้อง error ไม่ยอม save ข้อมูล

สามารถกำหนดได้ 2 แบบ คือ
แบบคงที่ ว้น/เดือน/ปี เช่น 1/1/51 
หรือ แบบอ้างอิงกับวันที่ปัจจุบัน โดยใช้เครื่องหมาย + หรือ - เช่น 

| [[_LowDate]] | ความหมาย |h
| -0 หรือ +0 |วันที่ปัจจุบัน (วันนี้) |
| +1 |+1 วัน จากวันที่ปัจจุบัน (พรุ่งนี้) |
| -7 |-7 วัน จากวันที่ปัจจุบัน |
| 0/-1 หรือ /-1 |-1 เดือน จากวันที่ปัจจุบัน |
| 0/0/-1 หรือ """//-1""" |-1 ปี จากวันที่ปัจจุบัน  |
| 1/- หรือ 1/-0 |วันที่ 1 ของเดือนนี้ |
| 1/1/- หรือ 1/1/-0 |วันที่ 1 เดือน 1 ของปีนี้ |
| 1/1/-1 |วันที่ 1 เดือน 1 ของปีที่แล้ว |

<<<
คำสั่งนี้ ไม่มีผลโดยตรง แต่ใช้เป็นค่า default กรณีมีการกำหนดคำสั่งต่อไปนี้ โดยไม่ระบุค่าใดๆ เช่น @@[[_BnkInLowDate]] = @@

* ภาษีมูลค่าเพิ่ม : [[_BuyTaxLowDate]] [[_SellTaxLowDate]]
* สต็อก : [[_StkInLowDate]] [[_StkOutLowDate]]
* ลูกค้า-ลูกหนี้ : [[_CusCrLowDate]] [[_CurDrLowDate]]
* เจ้าหนี้ : [[_VndCrLowDate]] [[_VndDrLowDate]]
* บิลขาย : [[_InvLowDate]] [[_InvPaidLowDate]]
* ใบเสร็จ : [[_RcpLowDate]] [[_RcpLowPDate]]
* บิลซื้อ : [[_PoLowDate]] [[_PoPaidLowDate]]
* ใบจ่าย : [[_MrcLowDate]] [[_MrcLowPDate]]
* บัญชีธนาคาร : [[_BnkInLowDate]] [[_BnkInLowClrDate]] [[_BnkOutLowDate]] [[_BnkOutLowClrDate]] 
* พิมพ์เช็ค : [[_PrnChqLowDate]]
<<<
|Default|@@""" _LowDateLockLevel = 8 """@@|
กำหนดระดับผู้ใช้ ที่ไม่สามารถป้อนวันที่ย้อนหลัง ก่อนวันที่ที่กำหนด 
วันที่ที่กำหนดแตกต่างกัน ขึ้นอยู่กับ module ที่ทำงาน (ดูคำสั่ง [[_LowDate]]) 
ค่า default 8 คือ lock ตั้งแต่ ผู้ใช้ที่มีระดับ user1 - user7, บัญชี 

| 0 |ไม่ lock |
| 1 |lock user1 |
| 2 |lock user2 หรือ ต่ำกว่า |
| 3 |lock user3 หรือ ต่ำกว่า |
| 4 |lock user4 หรือ ต่ำกว่า |
| 5 |lock user5 หรือ ต่ำกว่า |
| 6 |lock user6 หรือ ต่ำกว่า |
| 7 |lock user7 หรือ ต่ำกว่า |
| 8 |lock บัญชี หรือ ต่ำกว่า |
| 9 |lock ผู้จัดการ หรือ ต่ำกว่า |
| 10 |lock supervisor หรือ ต่ำกว่า |
|Default|@@""" _MoneyTxtSep = 0 """@@|
กำหนดรูปแบบของตัวเชื่อมคำภาษาอังกฤษของข้อความจำนวนเงิน 
> ข้อความจำนวนเงินเป็นภาษาอังกฤษ ในกรณีต่อไปนี้
> เมื่อใช้ระบบวันที่เป็น ค.ศ. [[_ThaiDate]] = OFF ใน """__FSys.INI"""
> หรือในไฟล์รายงาน .FRM  ใช้คำสั่ง [[ThaiPrint]] = OFF 
> หรือในไฟล์รายงาน .FRM  ใช้คำสั่ง [[_Money]] = 0

| [[_MoneyTxtSep]] | คำอธิบาย |h
| 0 |ไม่ใช้ตัวเชื่อม |
| 1 |ใช้ "and" เป็นตัวเชื่อม |
| 2 |ใช้ "," เป็นตัวเชื่อม |
| +4 |เชื่อมบาท/สตางค์แบบไทย xxx BAHT yyy STG. |
| +8 |สตางค์ใช้เป็นตัวเลข yyy/100 |
| +16 (0x10) |แปลงตัวอักษรเป็น lower case |
| +32 (0x20) |แปลงตัวอักษรเป็น upper case |
| +16+32 (0x30) |แปลงตัวอักษรเป็น lower case ยกเว้นตัวอักษรตัวแรกเป็น upper case |

การกำหนดค่า จะต้องใช้ค่าผลลัพธ์ที่บวกกันแล้ว หรือใช้ค่าเลขฐาน 16 ตามที่แสดงในวงเล็บ เช่น [[_MoneyTxtSep]] = 0x3C 

ตัวอย่าง 

* default BAHT xxxx and yyy STGS 
| 0 |BAHT One Thousand Two Hundred Twenty Five and Fifty STGS. |
| 1 |BAHT One Thousand and Two Hundred and Twenty Five and Fifty STGS. |
| 2 |BAHT One Thousand, Two Hundred Twenty Five and Fifty STGS. |

* +4 xxx BAHT yyy STGS. 
| 0+4 (4) |One Thousand Two Hundred Twenty Five BAHT Fifty STGS. |
| 1+4 (5) |One Thousand and Two Hundred and Twenty Five BAHT Fifty STGS. |
| 2+4 (6) |One Thousand, Two Hundred Twenty Five BAHT Fifty STGS. |

* +8 BAHT xxx and ##/100 
| 0+8 (8) |BAHT One Thousand Two Hundred Twenty Five and 50/100 |
| 1+8 (9) |BAHT One Thousand and Two Hundred and Twenty Five and 50/100 |
| 2+8 (10) (0xA) |BAHT One Thousand, Two Hundred Twenty Five and 50/100 |

* +4+8 xxx BAHT ##/100 
| 0+4+8 (12)(0xC) |One Thousand Two Hundred Twenty Five BAHT 50/100 |
| 1+4+8 (13)(0xD) |One Thousand and Two Hundred and Twenty Five BAHT 50/100 |
| 2+4+8 (14)(0xE) |One Thousand, Two Hundred Twenty Five BAHT 50/100 |

* +16 lower case 
|0+16 (16)(0x10) |BAHT one thousand two hundred twenty five and fifty STGS. |
| 0+4+16 (20)(0x14) |one thousand two hundred twenty five BAHT fifty STGS. |
| 0+4+8+16 (28)(0x1C) |one thousand two hundred twenty five BAHT 50/100 |

* +32 upper case 
| 0+32 (32)(0x20) |BAHT ONE THOUSAND TWO HUNDRED TWENTY FIVE AND FIFTY STGS. |
| 0+4+32 (36)(0x24) |ONE THOUSAND TWO HUNDRED TWENTY FIVE BAHT FIFTY STGS. |
| 0+4+8+32 (44)(0x2C) |ONE THOUSAND TWO HUNDRED TWENTY FIVE BAHT 50/100 |

* +16+32 First letter captitalize 
| 0+16+32 (48)(0x30) |BAHT One thousand two hundred twenty five and fifty STGS. |
| 0+4+16+32 (52)(0x34) |One thousand two hundred twenty five BAHT fifty STGS. |
| 0+4+8+16+32 (60)(0x3C) |One thousand two hundred twenty five BAHT 50/100 |

> note โปรแกรมบางรุ่นจะมี bug กรณีคำสั่ง xxx BAHT yyy STGS. ใช้ +4 ไม่ได้ จะต้องใช้ +64 แทน 
ใช้สำหรับซ่อนหมวดใบจ่าย ไม่ให้เลือกใช้ตอนเปิดใบจ่าย
ดูคำอธิบายจาก [[_InvCfgHideList]]
ใช้สำหรับเปลี่ยนรายการหมวดใบจ่าย ที่ใช้เลือกตอนก่อนเข้าไปเปิดใบจ่าย ซึ่งเรียงตามตัวอักษร ให้เป็นเมนูให้เลือกหมวดบิล ที่จัดลำดับหมวดบิลตามต้องการได้

ดูคำอธิบายจา [[_InvCustomListType]]
@@"""
_MrcLimitMrcType = ON 
"""@@

ใช้สำหรับกำหนดให้โปรแกรม ต้องแสดงรายการหมวดใบจ่าย มาให้เลือก ก่อนเข้าไปเปิดใบจ่ายหรือไม่
ดูคำอธิบายจาก [[_InvLimitInvType]]
|Default|@@""" _OffCnt = 15 """@@|
กำหนดเวลา (นาที) ที่โปรแกรมรอ โดยไม่มีการกดคีย์ใดๆ เพื่อเข้าสู่โหมด พักหน้าจอ โดยคำสั่ง [[_OffScr]]
default คือ 15 นาที
|Default|@@""" _OffScr = ON """@@|
พักหน้าจอ (Off Screen) แสดงเป็นตัวหนังสือวิ่ง เมื่อไม่มีการกดคีย์ใด ในช่วงเวลา ตามที่กำหนดใน [[_OffCnt]] (นาที)

> เพิ่มเติม (ยังไม่ test) คำสั่ง [[_OffScr]] สำหรับโปรแกรมรุ่น IDS สามารถใช้ค่า +4 เพื่อให้เช็คสถานะของ remote database ระหว่างที่ไม่มีการกดคีย์ เช่น [[_OffScr]] = 4 ไม่ใช้ Off Screen แต่ให้ตรวจสอบ IDS หรือ [[_OffScr]] = 5 ใช้ Off Screen และตรวจสอบ IDS 
{{{
cin ต้องการขอ modify pgogram 
ขั้นตอนผ่านรายการ ใบจอง (OF) ไม่ได้ Set ให้ผ่านรายการอะไร
กำหนด {~O-} เวลาผ่านรายการให้ตรวจสอบยอดคงเหลือ
"ที่สามารถขายได้" โดย คำนวณจากยอดคงเหลือ(stock.btr) หักลบกับรายจอง
ถ้ายอดที่เหลือน้อยกว่ายอดในใบจอง ให้ฟ้อง Error รอยืนยัน
โดยแสดงข้อมูล ของสาเหตุดังกล่าว เช่น 
เลขที่ใบจอง/ชื่อพนักงานขาย/วันที่จอง/จำนวนที่จอง
}}}

{{{
28/2/2550

กลัวติดค้างข้ามเดือน เลยต้องรีบเคลียร์ ฮิ ฮิ

ใช้ PInvA หรือ PInvB ก็ได้ แทน PInv
จะต้องกำหนด __FSYS.INI

_PInvChkStkBo=ON

ทดสอบคร่าวๆ แล้ว ข้อมูลไม่กี่รายการ ทำงานถูกต้อง
}}}

{{{
30/10/51

- ผ่านไปปีกว่า .. กับ Add-on ระบบนี้ ตอนนี้ลูกค้า แจ้งปัญหาการใช้งาน มาดังนี้ครับ
1. เวลาแสดงรายการ ค้าง อยากให้เรียง ตามลำดับ เลขที่เอกสาร 
(ของเก่าเรียงตามชื่อลูกค้าก่อนแล้วค่อยเรียงวันที่) เพราะเขาบอกดูยาก เพราะตอนนี้ข้อมูลเยอะ
ปล. เรียงตามวันที่ไม่ได้นะครับ เพราะ วันที่อาจมีกระโดดไปกระโดด มา เน้นเรื่องลำดับเอกสาร
2.อยากพิมพ์ สิ่งที่โปรแกรมเตือนได้.. เพราะรายการมันเยอะ จนไม่สะดวกที่จด 
ผมลองแล้ว ทั้ง Ctrl+KP, printsrc ก็ไม่ Work ฟ้อง memory ไม่พออย่างเดียว (mem 460)

พอมีทางแก้ไขไหมครับ..!!

1/11/2551

ให้เรียงตามเลขที่บิล ใช้
_PInvChkStkBo=101

เรื่องพิมพ์ ทดลองแล้ว ไม่น่าจะพิมพ์ได้ ติดปัญหาเรื่อง memory
เลยแก้โปรแกรมให้ กด Ctrl+KS เพื่อ save ไฟล์ไว้แทน
}}}

{{{
อ้างจาก: adm ที่ ธันวาคม 18, 2008, 05:56:43 am
18/12/51

-ที่ยังไม่ work คือการเรียงลำดับ เพราะยังเรียง 
"ลูกค้า" ก่อนที่จะเรียง "เลขที่ี" อีกครั้ง ส่วนอย่างอื่น work หมดแล้ว ครับ

ปล.ต้องการให้เรียงเลขที่เป็นลำดับแรกก่อนอย่างอื่น 
ขอโทษที่ที่ไม่ได้แนบไฟล์มาให้ดู เนื่องจากหาอะไร copy ไม่ได้เลย


18/12/51
ทดลองสร้างข้อมูล เพื่อทดสอบแล้ว พบข้อผิดพลาดตามที่แจ้งแล้วครับ
แก้ไขให้ถูกต้องแล้ว พร้อมทั้งแถมเพิ่มคำสั่ง 
_PInvChkStkBo=201 
สำหรับเรียงตามวันที่ครับ

note 
ข้อจำกัดที่อาจเกิดขึ้น คือปัญหากรณีที่มี lot ค้างส่งมากๆ อาจไม่สามารถแสดงออกมาได้ครบ
เนื่องจาก เดิม load ข้อมูล lot ค้างแยกตามลูกค้า ทีละรายทำให้ไม่ต้อง load lot ทั้งหมดมาไว้ใน memory แต่การสั่งให้เรียงทำให้ต้อง load lot ที่ค้างทั้งหมดของลูกค้าทุกรายเข้ามาไว้ใน memory เพื่อทำการเรียงใหม่ก่อน ดังนั้นวิธีนี้จึงกิน memory มากกว่าเดิม

เพิ่ม
_PInvChkStkBo=501

ไม่แสดงรายละเอียด lot ที่ค้าง 
ขึ้นแสดง error แค่ข้อความ ยอดสต็อกคงเหลือ น้อยกว่า ยอดค้างส่ง
}}}
{{{
update 15 กันยายน 2548

ผ่านรายการบิลขาย (PInv)
ผ่านรายการบิลซื้อ (PPo)
สามารถกำหนดให้ 
ผ่านรายการสต็อก
โดยปรับยอดคงเหลือ ที่ข้อมูลสต็อก
โดยไม่บันทึก Transaction 
รายการออกสต็อก หรือ รายการเข้าสต็อก

ใช้คำสั่ง 
_PInvSkipTrn = 1
หรือ
_PPoSkipTrn = 1

การทำงานยังสามารถ
ผ่านรายการ และคืนรายการ เหมือนปกติ
แต่จะไม่ปรากฏข้อมูล รายการเข้า/ออก สต็อก

ใช้ประโยชน์อะไร (วะ) นะหรือ
เป็น feature หนึ่งใน Project 
Light Weight POS Client
}}}
|Default|@@""" _PRNCode = 1 """@@|
กำหนดรหัสภาษาไทย ของ export ไฟล์ .PRN
* 0 = รหัส เกษตร 
* 1 = รหัส สมอ. (windows-tis) 
|Default|@@""" _PViewColorMode = ON """@@|
การแสดงผลรายงานออกหน้าจอ เป็นแบบสี ถ้าเครื่องพิมพ์ที่จะพิมพ์ออกเป็นเครื่องพิมพ์สี 
ถ้าเปลี่ยนค่านี้เป็น OFF หรือเครื่องพิมพ์ที่ใช้เป็นขาวดำ การแสดงรายงานบนหน้าจอก็จะเป็น ขาว-ดำ
|Default|@@""" _PViewSaveMode = ON """@@|
กรณีแสดงรายงานออกทางหน้าจอ โปรแกรมจะเก็บรายงานหน้าที่ผ่านมาแล้ว เพื่อย้อนกลับมาดูได้โดยกด ~PgUp
ถ้าเปลี่ยนคำสั่งนี้เป็น OFF การดูรายงานทางหน้าจอจะดูได้เฉพาะรายงานหน้าล่าสุด ไม่สามารถย้อนไปดูหน้าก่อนได้ 
|Default|@@""" _PageNoPic = #9-4 """@@|
กำหนด picture สำหรับการพิมพ์เลขหน้า (Page No) ลงบนรายงาน  ค่า default คือ "9999" 
|Default|@@""" _PoAllowChangeDsc = ON """@@|
กำหนดระดับ user ต่ำสุด ที่ยินยอมให้แก้ไขช่อง__ส่วนลด__ต่อรายการสินค้าในบิล

ดูคำอธิบายจาก [[_InvAllowChangeDsc]]
|Default|@@""" _PoAllowChangeNo = ON """@@|
ใช้สำหรับ lock ฟิลด์__เลขที่__บิลซื้อ ไม่ให้ user ที่มีระดับต่ำกว่าระดับที่กำหนด เข้าไปเปลี่ยน

ดูคำอธิบายจาก [[_InvAllowChangeNo]]
|Default|@@""" _PoAllowChangePrice = ON """@@|
กำหนดระดับ user ต่ำสุด ที่ยินยอมให้แก้ไขช่อง__ราคา__สินค้าในบิล

ดูรายละเอียดจาก [[_InvAllowChangePrice]]
|Default|@@""" _PoAllowChangeQty = ON """@@|
ใช้สำหรับ lock ฟิลด์__จำนวน__ในบิลซื้อ ไม่ให้ user ที่มีระดับต่ำกว่าระดับที่กำหนด เข้าไปแก้ไขจำนวนในฟิลด์ดังกล่าว 

ดูคำอธิบายจาก [[_InvAllowChangeDsc]]
|Default|@@""" _PoAllowPostPrint = ON """@@|
ใช้ควบคุมเพื่อ lock ไม่ให้พิมพ์บิลซื้อ ที่ผ่านรายการแล้ว

ดูคำอธิบายจาก [[_InvAllowPostPrint]]
|Default|@@""" _PoAllowPrintEdit = ON """@@|
ใช้ lock ไม่ให้แก้ไขหรือลบ บิลที่พิมพ์แล้ว

ดูคำอธิบายจาก [[_InvAllowPrintEdit]]
|Default|@@""" _PoAutoListBOM = OFF """@@|
ใช้สำหรับหน้าจอเปิดบิลซื้อ เวลาป้อนรายการ ส่วนประกอบสินค้าต้องขึ้นอัตโนมัติ 

ดูคำอธิบายจาก [[_InvAutoListBOM]]
|Default|@@""" _PoAutoPackItm = -1 """@@|
ใช้สำหรับยุบรวมจำนวนสินค้า สำหรับบรรทัดที่มีรายการเหมือนกันหลายบรรทัด มารวมเป็นบรรทัดเดียวกัน

ดูคำอธิบายจาก [[_InvAutoPackItm]]
ใช้สำหรับกำหนดให้ โปรแกรมแสดง list ของบิลต้นแบบ มาให้เลือกก่อน ตั้งแต่ตอนกด F5 เพื่อเปิดบิลใหม่  สามารถกำหนดดังนี้

@@"""
_InvAutoReferNo=หมวดบิล [ +wildcard ของเลขที่บิลต้นแบบ ]
"""@@

ดูคำอธิบายจาก [[_InvAutoReferNo]]
ใช้สำหรับซ่อนหมวดบิล ไม่ให้เลือกใช้ตอนเปิดบิล
ดูคำอธิบายจาก [[_InvCfgHideList]]
ใช้สำหรับเปลี่ยนรายการหมวดบิล ที่ใช้เลือกตอนก่อนเข้าไปเปิดบิล ซึ่งเรียงตามตัวอักษร ให้เป็นเมนูให้เลือกหมวดบิล ที่จัดลำดับหมวดบิลตามต้องการได้

ดูคำอธิบายจา [[_InvCustomListType]]
|Default|@@""" _PoDspDscNSize = 4 """@@|
สำหรับบังคับค่า ขนาด(กว้าง) ของคอลัมน์__ส่วนลด__ ในจอบิลที่ป้อนข้อมูล กำหนดได้ 0 - 10

ดูคำอธิบายจาก [[_InvDspDscNSize]]
@@"""
_PoDspUnitNSize = 8 
"""@@

สำหรับบังคับค่า ขนาด(กว้าง) ของคอลัมน์__หน่วยนับ__ ในจอบิลที่ป้อนข้อมูล กำหนดค่าได้ 0 - 12

ดูคำอธิบายจาก [[_InvDspUnitNSize]]
|Default|@@""" _PoLimitPoType = ON """@@|
ใช้สำหรับกำหนดให้โปรแกรม ต้องแสดงรายการหมวดบิล มาให้เลือก ก่อนเข้าไปเปิดบิลหรือไม่
ดูคำอธิบายจาก [[_InvLimitInvType]]
|Default|@@""" _PrintBreakChk = 0 """@@|
กำหนดจำนวนเวลาของรอบ ที่ให้โปรแกรมตรวจสอบการกดคีย์ Esc เพื่อยกเลิกการพิมพ์ 
ค่า default 0 โปรแกรมจะตรวจสอบการกดคีย์ทุกรอบ 1 วินาที
ถ้ากำหนดค่า 1 โปรแกรมจะตรวจสอบการกดคีย์ทุก 3 วินาที (1 x 3)
ถ้ากำหนดค่า 2 โปรแกรมจะตรวจสอบการกดคีย์ทุก 6 วินาที (2 x 3)
|Default|@@""" _PrintChkReady = ON """@@|
กำหนดให้โปรแกรมตรวจสอบสถานะของเครื่องพิมพ์ก่อนเริ่มพิมพ์ว่า เครื่องพิมพ์พร้อมที่จะพิมพ์ (Ready หรือ Online) หรือไม่

มีกรณีที่ใช้มีกล่องสลับสายบางรุ่น เชื่อมเครื่องคอมพิวเตอร์ 2 เครื่อง ต่อเครื่องพิมพ์ 1 เครื่อง  ไม่ยอมแสดงสถานะว่าเครื่องพิมพ์พร้อม ดังนั้นจึงต้องสามารถสั่งให้โปรแกรมเริ่มพิมพ์โดยไม่ต้องตรวจสอบก่อน โดยเปลี่ยนคำสั่งนี้เป็น OFF 
|Default|@@""" _PrintControl = OFF """@@|
การควบคุมลักษณะการพิมพ์บางอย่าง สามารถทำได้ทั้งจาก panel ของเครื่องพิมพ์ หรือโดยจากโปรแกรมส่งคำสั่งควบคุม คำสั่งนี้ใช้เพื่อกำหนดว่า ผู้ใช้จะเป็นผู้ควบคุมลักษณะการพิมพ์เอง (ON) หรือปล่อยให้โปรแกรมควบคุม (OFF) โดยกำหนดจากไฟล์ .FRM 

* คุณภาพตัวอักษร LQ/Draft (ดู - [[LQ]] หรือ [[QUALITY]]) 
* ฟอนท์ รูปแบบตัวอักษร (ดู - [[LQFONT]]) 
* ขนาดตัวอักษร (10/12/15/17/20 cpi) (ดู - [[12CPI]], [[15CPI]], [[COMPRESS]]) 
|Default|@@""" _PrintExportToTemp = OFF """@@|
ปกติโปรแกรมจะ export ไฟล์ (.PRN, .TXT, .CSV) ไว้ที่ directory ของชุดข้อมูลที่ทำงานอยู่

ถ้ากำหนดค่าเป็น ON โปรแกรมจะ export ไฟล์ ไปไว้ที่ temp directory ซึ่งกำหนดโดยคำสั่ง [[_UseLocalTmp]]
|Default|@@""" _PrintMaskStyle = 15 """@@|
ใช้สำหรับปิดไม่ให้ใช้ style ของ font ในเครื่องพิมพ์บาง style เนื่องจาก เครื่องพิมพ์บางรุ่นเมื่อใช้ style นั้นแล้ว ทำให้ตัวหนังสือไม่สวย หรืออ่านไม่ออก 

| +1 |ขีดเส้นใต้ (underline) |
| +2 |ตัวเอียง (italic) |
| +4 |ตัวเงา (shadow) |
| +8 |ตัวเจาะ (outline) |

ค่า default คือ เปิดใช้ทุก style (1+2+4+8 = 15) 
|Default|@@""" _PrintMaxOutput = 0 """@@|
ใช้สำหรับจำกัดช่องทางการแสดงผลของรายงาน ค่า default=0 คือ ไม่จำกัด 

ช่องทางการแสดงผล มีดังนี้ 
| 1 | หน้าจอ  |
| 2 | 1 + เครื่องพิมพ์ |
| 3 | 2 + PRN File |
| 4 | 3 + TXT File |
| 5 | 4 + CSV File |

ตัวอย่าง 
ถ้ากำหนด [[_PrintMaxOutput]] = 2 คือ จำกัดให้แสดงรายงานได้แค่ลำดับที่ 2 คือ หน้าจอ และ เครื่องพิมพ์ ไม่สามารถ export รายงานออกมาเป็นไฟล์ 
|Default|@@""" _PrintOutput = 0 """@@|
กำหนดค่า default output ของรายงาน 

| 0 | หน้าจอ  |
| 1 | เครื่องพิมพ์ |
| 2 | PRN File |
| 3 | TXT File |
| 4 | CSV File |
|Default|@@""" _PrintPortNo = 0 """@@|
กำหนด printer port ที่ใช้เป็น default port สำหรับพิมพ์รายงาน
| 0 | lpt1 |
| 1 | lpt2 |
| 2 | lpt3 (server) |
|Default|@@""" _PrintServer = OFF """@@|
ใช้สำหรับกรณีที่เครื่องพิมพ์ต่อกับ Netware Server และมีปัญหา server จะ delay ไม่ยอมพิมพ์รายงานออกเครื่องพิมพ์ทันที เพราะรอรับคำสั่ง end of job ก่อน ถ้ากำหนดค่านี้เป็น ON โปรแกรมจะส่งคำสั่ง EOJ หลังจากจบรายงาน 
ใช้สำหรับซ่อนหมวดใบเสร็จ ไม่ให้เลือกใช้ตอนเปิดใบเสร็จ
ดูคำอธิบายจาก [[_InvCfgHideList]]
ใช้สำหรับเปลี่ยนรายการหมวดใบเสร็จ ที่ใช้เลือกตอนก่อนเข้าไปเปิดใบเสร็จ ซึ่งเรียงตามตัวอักษร ให้เป็นเมนูให้เลือกหมวด ที่จัดลำดับหมวดใบเสร็จตามต้องการได้

ดูคำอธิบายจา [[_InvCustomListType]]
|Default|@@""" _RcpLimitRcpType = ON """@@|
ใช้สำหรับกำหนดให้โปรแกรม ต้องแสดงรายการหมวดใบเสร็จ มาให้เลือก ก่อนเข้าไปเปิดใบเสร็จหรือไม่
ดูคำอธิบายจาก [[_InvLimitInvType]]
|Default|@@""" _RecoverMode = 1 """@@|
กำหนดให้โปรแกรมทำการกู้ข้อมูลแบบ auto correction สำหรับกรณี error code 22 (ขนาด record ไม่พอดี) รวมทั้งการตรวจสอบและปรับขนาดของ field ที่เป็น string อัตโนมัติ, ตรวจสอบ field วันที่ 
|Default|@@""" _ReopenDelay = 5 """@@|
กำหนดให้โปรแกรม ปิด database แล้ว เปิดใหม่ ถ้าตรวจเช็คแล้ว พบว่า database ได้เปิด file ทิ้งไว้นานกว่าเวลาที่กำหนด (default คือ 5 นาที)โดยไม่มีการกดคีย์ใดๆ ก่อนที่จะทำงาน save data หรือ delete data ป้องกันปัญหากรณี connection lost โดยเฉพาะกรณี IDS ที่มี database เป็นแบบ remote 
|Default|@@""" _ScrMgrAutoCorrectLen = 6 """@@|
การทำงานจอป้อนข้อมูล สั่งให้ตรวจจับและแปลงตัวอักษรภาษาไทยเป็นตัวเลขอัตโนมัติ โดยกำหนดขนาดความยาวของตัวอักษรภาษาไทยอย่างน้อย (default คือ 6) ที่โปรแกรมจะตรวจและแปลงตัวอักษรภาษาไทยเป็นตัวเลข อัตโนมัติ ใช้ประโยชน์ในกรณีเครื่องอ่านบาร์โค้ดอ่านรหัสตัวเลข ขณะที่คีย์บอร์ดของโปรแกรมย้งอยู่ในโหมดภาษาไทย 
|Default|@@""" _ScrMgrMultiplier = 1000 """@@|
การทำงานจอป้อนข้อมูล สำหรับ field ป้อนตัวเลข ถ้ากดปุ่ม * จะเป็นการคูณตัวเลขใน field นั้น ด้วยค่า Multiplier ที่กำหนด (เหมือนกับเป็นการ Quick Input เติม 000) และกลับกัน ถ้ากดปุ่ม / ก็จะเป็นการหารตัวเลขใน field ด้วยค่า Multiplier 
|Default|@@""" _ScrMgrZoomFlag = 1 """@@|
การทำงานจอป้อนข้อมูล สำหรับ field ป้อนข้อความยาว ที่ยาวกว่าขนาดของ field ที่แสดงบนจอ 
| 0 |ไม่ขึ้น field ขยาย จะต้อง scroll ภายใน field |
| 1 |ขึ้น field แบบขยายขึ้นมายาวเต็มให้ป้อน ซ้อนทับตำแหน่ง field เดิม |
| 2 |ขึ้นหน้าต่างขยายขึ้นมายาวเต็มให้ป้อน พร้อมกรอบ และชื่อ field ซ้อนตำแหน่ง field เดิม |
| 3 |ขึ้นหน้าต่างขยายขึ้นมายาวเต็มให้ป้อน พร้อมกรอบ และชื่อ field ใต้ตำแหน่ง field เดิม |
นอกจากนี้ ในระหว่างป้อนข้อมูล เราสามารถกด Ctrl+Z เพื่อสลับ Zoom Mode ของ field ได้อีกด้วย 
|Default|@@""" _SearchPathPrg = -1 """@@|
กำหนดให้โปรแกรม search หาไฟล์ทำงาน (.EXE) จาก ค่าของ PATH ที่กำหนดใน environment ของเครื่องที่ใช้งานอยู่ หรือไม่ (DOS API - searchpath) ปกติเมื่อมีการเรียกใช้ระบบงานย่อย โปรแกรมก็จะไปเรียกไฟล์ .EXE ของระบบงานนั้นขึ้นมาทำงาน โดยหาไฟล์ที่ต้องการ 2 วิธีดังนี้

* หาภายใน directory ต่างๆ ของโปรแกรมเอง
*# จาก program directory หรือ directory เดียวกันกับที่ไฟล์โปรแกรมเมนู (~MN_*.EXE)
*# จาก home directory หรือ working directory (กรณีอยู่คนละที่กับ program directory)
*# จาก data directory คือ directory ของชุดข้อมูลที่กำลังเปิดใช้อยู่
*# จาก frm directory คือ directory ชื่อ (FRM) ปกติใช้สำหรับเก็บไฟล์ฟอร์มรายงานเพิ่มเติม (.FRM)
* หาโดยใช้ DOS API searchpath จาก directory ที่กำหนดโดย environment PATH
| [[_SearchPathPrg]] | คำอธิบาย |h
| -1 |หาจากภายใน directory ของโปรแกรมเองก่อน ถ้าไม่พบจึงหาโดยใช้ DOS API |
| 0 หรือ OFF |หาจากภายใน directory ของโปรแกรมเองเท่านั้น |
| 1 หรือ ON |หาโดยใช้ DOS API ก่อน ถ้าไม่พบจึงหาจากภายใน directory ของโปรแกรมเอง |

> ข้อดีของ การใช้ DOS API searchpath คือ เราสามารถ จัดที่อยู่ของไฟล์โปรแกรมได้ยืดหยุ่น แต่ข้อเสียคือ การทำงาน DOS API searchpath อาจทำให้เสียเวลา ยิ่งถ้า directory ที่ต้อง search มีไฟล์มาก เช่น windows ก็ยิ่งเสียเวลา
|Default|@@""" _SellTaxChain = OFF """@@|
กรณีที่ใช้ [[_SellTaxChain]]=ON จะมีผลเปลี่ยน sequence การป้อนข้อมูลภาษีขาย โดยหลังจากที่ save ข้อมูลภาษีขาย โปรแกรมจะข้ามไปให้ใส่ข้อมูล เพิ่มหนี้ลูกหนี้ และสต็อกสินค้าออก โดยใส่ค่าในช่อง วันที่ และ เลขที่เอกสาร เป็นค่าเดียวกันกับในรายการภาษีขายที่เพิ่ง save นั้น 

นอกจากนี้ ยังมีผลกับ หน้าจอแสดงรายการภาษีขาย (browse list) เราสามารถใช้วิธีกด Space แล้วกด F1 เพื่อเข้าไปป้อนข้อมูล เพิ่มหนี้ลูกหนี้ หรือสต็อกสินค้าออก ที่มี วันที่/เลขที่เอกสารเดียวกัน 

คำสั่ง [[_SellTaxChain]]=2 ทำงานเหมือน [[_SellTaxChain]]=ON และเพิ่มกรณีสั่งลบรายการภาษีซื้อ โปรแกรมจะส่งคำสั่งไปลบรายการ เพิ่มหนี้เจ้าหนี้ และสต็อกสินค้าเข้า อัตโนมัติ 

ดูเพิ่มเติม: [[_BuyTaxChain]] 
|Default|@@""" _ShowLogo = ON """@@|
แสดง Logo ของ Soft-Craft ตอนเริ่มโปรแกรม 
|Default|@@""" _ShowTitle = ON """@@|
แสดงกรอบ ชื่อ/ที่อยู่ ของกิจการ ตรงกลางจอ 
|Default|@@""" _SleepDelay = 5 """@@|
ในระหว่างที่อยู่ในจอ Browse ข้อมูล ถ้าไม่มีการกดคีย์ใดๆ ภายในกำหนดเวลา (default คือ 5 นาที) โปรแกรมจะปิด database และเข้าสู่ mode sleep (idle) เพื่อปล่อยให้ process อื่นทำงาน จนกว่าจะมีการกดคีย์ 
@@"""
_StkXListStock=hide#,ข้อความเมนู[,hide#,ข้อความเมนู...] 
"""@@

กำหนดเมนูเพื่อแยกแสดงรายการสินค้า ตามเงื่อนไขที่อ้างถึงโดยคำสั่ง [[_StockXListHide]] 

ตัวอย่าง เราสามารถกำหนด channel # ของ [[_StockXListHide]] เตรียมไว้ก่อนได้ไม่จำกัด 
|[[_StockXListHide]].1 = 3 |(ซ่อนรายการสินค้าที่ยอดคงเหลือเป็น 0 ทุกกรณี) |
|[[_StockXListHide]].2 = 0x182,*สีดำ* |(ซ่อนรายการสินค้าที่ยอดเป็น 0 แบบมี transaction และ มีคำว่า "สีดำ" ในชื่อสินค้า) |
|[[_StockXListHide]].3 = 0x102,*สีดำ* |(ซ่อนรายการสินค้าที่ยอดเป็น 0 แบบมี transaction หรือ สินค้าที่มีคำว่า "สีดำ" ในชื่อสินค้า) |
|[[_StockXListHide]].4 = 0x100,!*สีแดง* |(ซ่อนรายการสินค้าที่ไม่มีคำว่า "สีแดง" ในชื่อสินค้า) |

แล้วเราก็กำหนดรายการเมนู ดังนี้ 
|[[_StkXListStock]] = 0,สินค้าทั้งหมด,1,เฉพาะสินค้าที่เหลือในสต็อค,4,สินค้าสีแดง |

> คำสั่งนี้ มีผลให้ขึ้นเมนูเลือกแสดงสินค้าตามเงื่อนไข ก่อน ที่จะเข้าหน้าจอแสดงรายการสินค้าปกติ และไม่สามารถเข้าหน้าจอแสดงรายการปกติ ยกเว้นในคำสั่งนี้มีกำหนด channel 0 ให้เลือกด้วย 

> นอกจากนี้ เมื่อกด F1 เพื่อเลือกชื่อสินค้า จากหน้าจอป้อนข้อมูล สินค้าเข้า, สินค้าออก, บิลขาย, บิลซื้อ โปรแกรมก็จะแสดงเมนูให้เลือกแสดงสินค้าตามเงื่อนไขก่อนเช่นกัน 
|Default|@@""" _StockAltPrice = 0 """@@|
กำหนดให้คำนวณราคาขาย ของสินค้าจากตารางราคา มาแสดงในจอ browse list ของสินค้า แทนราคาที่เก็บใน master ข้อมูลสินค้า 

| default 0 |ใช้ราคาจาก master ข้อมูลสินค้า |
| 1 ถึง 99 |ใช้ราคาตามดัชนี (1-99) จาก ตารางราคา? ที่ตั้งไว้ในข้อมูล สินค้าเพิ่มเติม |
| 100 |ใช้ราคาตามดัชนี 0 จาก ตารางราคา ที่ตั้งไว้ในข้อมูล สินค้าเพิ่มเติม กรณีที่ไม่พบราคาตามดัชนีที่กำหนด โปรแกรมจะใช้ราคาจาก master ข้อมูลสินค้า |
|Default|@@""" _StockFilterChoice = OFF """@@|
ถ้ากำหนดค่าเป็น ON หรือ 1 โปรแกรมจะขึ้นเมนูให้เลือกแสดงรายการสินค้า ระหว่าง สินค้าที่เป็นสต็อก 
สินค้าที่ไม่เป็นสต็อก (มี * นำหน้าชื่อ หรือมี \ ภายในชื่อ หรือมี {N} ภายในหน่วยนับ/หรือชื่อหมวด) 
ก่อนที่จะเข้าสู่จอแสดงรายการสินค้า (browse list) 

> note: คำสั่งใหม่ [[_StkXListStock]] สามารถกำหนดเงื่อนไขเลือกแสดงรายการสินค้าได้หลากหลาย และยีดหยุ่นกว่า 
|Default|@@""" _StockShowQInf=-3 """@@|
(มี.ค.53) ใช้ควบคุมการแสดงข้อมูลเพิ่มเติม (QuickInfo) ที่ด้านล่างของจอ Browse ของข้อมูลสต็อก ผู้ใช้สามารถกด ''Alt + I'' เพื่อสลับระหว่าง ซ่อน/แสดง QuickInfo
| 0 หรือ OFF |ปิด ไม่แสดง QuickInfo |
| -1 |เปิดใช้ QuickInfo แต่ซ่อนเมื่อเริ่ม |
| 1 หรือ ON |เปิดใช้ QuickInfo และแสดงเมื่อเริ่ม |
| 100 |เปิดใช้ QuickInfo และแสดง ไม่สามารถใช้ ~Alt-I เพื่อสั่งซ่อน |
 
|Default|@@""" _TXTCode = 1 """@@|
กำหนดรหัสภาษาไทย ของ export ไฟล์ .TXT
* 0 = รหัส เกษตร 
* 1 = รหัส สมอ. (windows-tis) 
|Default|@@""" _ThaiDate = ON """@@|
แสดงวันที่แบบปี พ.ศ. 
ถ้าใช้ [[_ThaiDate]] = OFF จะเปลี่ยนเป็นปี ค.ศ. มีผลทำให้ การแสดงข้อความจำนวนเงินในรายงานเป็นภาษาอังกฤษด้วย
|Default|@@""" _TransactionDelay = 0 """@@|
กำหนดให้หน่วงเวลา (millisec = 1/100 วินาที) เมื่อโปรแกรมทำการ commit transaction กรณีที่ database server ต้องรับ load ปริมาณมาก มีจำนวน client ที่ connect มาก อาจเกิดปัญหาคอขวด ไม่สามารถ update database ได้ทัน เมื่อมีการ commit 
|Default|@@""" _UseLocalTmp = OFF """@@|
ปกติโปรแกรมจะใช้ directory ''(BTR)'' ที่อยู่ภายใน directory ของโปรแกรมที่ทำงานอยู่ เป็น temp directory

แต่ถ้ากำหนดเป็น ON โปรแกรมจะใช้ directory ตามที่ตั้งไว้ที่ TEMP ใน DOS environment ของเครื่องนั้นๆ เป็นที่เก็บ temp file (.TMP) และ export file (.PRN, .TXT, .CSV) ถ้ากำหนด [[_PrintExportToTemp]] = ON
|Default|@@""" _UseRTC = OFF """@@|
กำหนดให้อ่านเวลาของระบบ (system time) จาก Bios Realtime Clock แทนที่จะอ่านเวลาจาก DOS Time สำหรับ Windows 98 นั้นไม่แตกต่าง แต่สำหรับ Windows 2000, Windows XP ซึ่ง DOS เป็น Virtual Platform จะข้อแตกต่างของเวลาจาก RTC กับ DOS Time นั่นคือ DOS สามารถ Pause หรือหยุดการทำงานชั่วคราวได้ ทำให้เวลาอาจเดินๆ หยุดๆ จนทำให้เวลาของ DOS ไม่ตรงกับเวลาจริง 
ให้โปรแกรมซ่อนชื่อเจ้าหนี้ โดยกำหนดเงื่อนไขจากข้อความในช่องผู้ติดต่อ สามารถใช้ wildcard และ , (comma) เช่น @@"""_VendorHideContact"""=*ซ่อน*,*{H}*@@
ให้โปรแกรมซ่อนชื่อเจ้าหนี้ โดยกำหนดเงื่อนไขจากข้อความในช่องหมวดเจ้าหนี้ สามารถใช้ wildcard และ , (comma) เช่น @@"""_VendorHideVndGroup""" = ส่วนตัว,*เลิก@@
|Default|@@""" _VendorShowQInf=-1 """@@|
(มี.ค.53) ใช้ควบคุมการแสดงข้อมูลเพิ่มเติม (QuickInfo) ที่ด้านล่างของจอ Browse 

ดูรายละเอียดที่ [[_StockShowQInf]]
ใช้สำหรับซ่อนเล่มใบหักภาษี ไม่ให้เลือกใช้ตอนเปิดใบหักภาษี
ดูคำอธิบายจาก [[_InvCfgHideList]]
|Default|@@""" _WTaxLimitWTGNo = ON """@@|
ใช้สำหรับกำหนดให้โปรแกรม ต้องแสดงรายการเล่มใบหักภาษี มาให้เลือก ก่อนเข้าไปเปิดใบหักภาษีหรือไม่
ดูคำอธิบายจาก [[_InvLimitInvType]]
|Default|@@""" _XMem = ON """@@|
กำหนดให้โปรแกรมใช้ XMS เป็น Memory ส่วนขยาย ไม่ควรเปลี่ยนค่านี้ ใช้สำหรับ test debug เท่านั้น 
|Default|@@""" __BDirSkip = OFF """@@ฅ
ใช้สำหรับกำหนด ให้ user สามารถเลือกเปลี่ยน ชุดข้อมูลที่ต้องการ ก่อนเข้าโปรแกรมหรือไม่

| 0 หรือ OFF |ให้โปรแกรมหยุดรอถามชื่อชุดข้อมูลก่อนเข้าโปรแกรม |
| 1 หรือ ON |ไม่ต้องหยุดรอถามชื่อชุดข้อมูลก่อนเข้าโปรแกรม |
| 2 |ไม่ต้องหยุดรอถามชื่อชุดข้อมูลก่อนเข้าโปรแกรม และไม่สามารถเปลี่ยนชุดข้อมูลด้วย |

ใช้คำสั่งนี้คู่กับ [[__BDirStart]] และสามารถใช้ชื่อย่อของ user ที่ต้องการต่อท้าย ช่วยให้สามารถควบคุมการใช้งานชุดข้อมูลที่แตกต่างกัน ของ user แต่ละคนได้ เช่น

[[__BDirStart]].TOM = SALE
[[__BDirSkip]].TOM = 2
มีผลให้ user TOM เข้าโปรแกรมโดยใช้ชุดข้อมูลชื่อ SALE อัตโนมัติ และไม่สามารถเปลี่ยนไปใช้ชุดข้อมูลอื่น

{{{
สำหรับโปรแกรม update ล่าสุด 31/3/48 
(load โปรแกรม update ล่าสุดได้จาก ftp อยู่ใน scaft/small business pack/latest built)

__BDIRSKIP=ON ใน __FSYS.INI
ใช้สำหรับควบคุมให้ข้าม ไม่ต้องหยุดรอถามชุดข้อมูล
ปกติควรใช้คู่กับ __BDIRSTART ดูรายละเอียดได้ที่ http://xpcnet/phpbb2/viewtopic.php?t=388 และเช่นเดียวกับ __BDIRSTART สามารถระบุเจาะจงเฉพาะ user ต่อท้ายได้ เช่น 
__BDIRSTART.TOM=SALE
__BDIRSKIP.TOM=ON

ตามตัวอย่างข้างต้น เมื่อ TOM login เข้ามา จะบังคับชุดข้อมูลสำหรับ TOM เป็นชุดที่ชื่อ SALE และไม่หยุดรอถามชุดข้อมูล ดังนั้น TOM จะถูกบังคับให้เข้าสู่ชุดข้อมูล SALE อัตโนมัติ ไม่สามารถเลือกเปลี่ยนไปชุดข้อมูลอื่น
}}}

{{{
update ล่าสุด built 1/4/48

เพิ่มความหมายกรณี __BDIRSKIP=2 จะบังคับไม่ให้เลือกเปลี่ยนชุดข้อมูลหลังจากเข้าโปรแกรมไปแล้ว ในเมนู เลิก

ที่จริงแล้วมีอีกวิธีที่ผมเคยใช้อยู่คือ บังคับโดยใช้ command line บังคับชื่อ user ที่ใช้ และชื่อชุดข้อมูล เป็นการบังคับเจาะจงในระดับเครื่องเลยว่าต้องเป็น user ชื่อนี้ และใช้ชุดข้อมูลนี้ นั่นคือเมื่อเรียกโปรแกรมผ่าน icon จะทะลุเข้าโปรแกรมอัตโนมัติ ไม่ต้อง login หรือ เลือกชุดข้อมูล ตัวอย่างเช่นที่ BTG ผมใช้วิธีบังคับ login เป็น POS1 ที่เครื่องหนึ่ง และ POS2 อีกเครื่องหนึ่ง ด้วย command line ช่วยให้ตรวจสอบรู้ได้ว่าเป็นบิลที่ออกจาก คอมพิวเตอร์เครื่องไหน
}}}
ใช้สำหรับกำหนด ชื่อชุดข้อมูลเริ่มต้น เมื่อเข้าโปรแกรม ตัวอย่างเช่น
@@"""
__BDirStart = MYDATA
"""@@

> ปกติ โปรแกรมจะใช้ชุดข้อมูลเริ่มต้นชื่อ WORK ถ้าไม่พบก็จะใช้ DATA ถ้าไม่พบอีกก็จะเว้นว่าง

นอกจากนี้ เราสามารถกำหนดชุดข้อมูลเริ่มต้น สำหรับ user แต่ละคนให้แตกต่างกันได้ โดยใช้
[[__BDirStart]]''.ชื่อย่อ'' = ชุดข้อมูล

ต้วอย่างเช่น
[[__BDirStart]].TOM=SALE สำหรับ user TOM จะใช้ชุดข้อมูลชื่อ SALE เป็นชุดข้อมูลเริ่มต้น

คำสั่งนี้ จะต้องอยู่ในไฟล์ """__FSys.INI""" ที่อยู่ใน directory ของโปรแกรมเท่านั้น
| 0 หรือ OFF |lock ไม่อนุญาต user ทุกระดับ |
| 1 หรือ ON |อนุญาตให้ user ทุกระดับ (default) |
| 2 |อนุญาตตั้งแต่ระดับ user2 ขึ้นไป |
| 3 |อนุญาตตั้งแต่ระดับ user3 ขึ้นไป |
| 4 |อนุญาตตั้งแต่ระดับ user4 ขึ้นไป |
| 5 |อนุญาตตั้งแต่ระดับ user5 ขึ้นไป |
| 6 |อนุญาตตั้งแต่ระดับ user6 ขึ้นไป |
| 7 |อนุญาตตั้งแต่ระดับ user7 ขึ้นไป |
| 8 |อนุญาตตั้งแต่ระดับ บัญชี ขึ้นไป |
| 9 |อนุญาตตั้งแต่ระดับ ผู้จัดการ ขึ้นไป |
| 10 |อนุญาต เฉพาะระดับ supervisor |
{{{
ปรับปรุง 21/7/2548

เป็นการปรับปรุง
เพื่อให้เกิดความยืดหยุ่นสำหรับ 
รายงานที่ไม่ Fix รูปแบบ

เราสามารถกำหนดคำสั่งควบคุมรายงาน บางส่วน
ที่เดิมต้องไว้ใน .FRM
เอามาไว้ใน __FSYS.INI แทน
ซึ่งทำให้มีผลกับทุกรายงานอัตโนมัติ
ยกเว้นภายใน .FRM มีคำสั่งนั้นมาซ้อน เพื่อบังคับใช้ค่าตาม .FRM
เพื่อความสะดวกในการเปลี่ยนแปลงค่าในอนาคต

การใช้คำสั่งจะเป็น คำสั่งเดียวกันกับใน .FRM
แต่ต้องเพิ่ม FRM. นำหน้าคำสั่ง เมื่อนำมาใช้ใน __FSYS.INI
เช่น FRM.PORTNO = 2

มีคำสั่งที่เพิ่มขึ้นมาคือ NLINE_PAGE 
เพื่อรองรับกรณีปรับค่า PAGELENGTH
และต้องการปรับจำนวนบรรทัดต่อหน้า ให้สัมพันธ์กัน

มีผลก็คือ ถ้ามีการกำหนด LINE_PAGE จาก __FSYS.INI
สำหรับ .FRM ที่มีกำหนด NLINE_PAGE=ON
ก็ให้ใช้ค่าตาม __FSYS.INI นั้น 
แต่ถ้าไม่มีกำหนดใน __FSYS.INI
จะใช้ค่าตาม LINE_PAGE ใน .FRM

กลับกันถ้ากำหนด NLINE_PAGE=OFF
ก็จะมีผลบังคับใช้ค่า LINE_PAGE ใน .FRM
และยัง reset ค่า PAGELENGTH ให้เป็น 11 อีกด้วย

สำหรับโปรแกรมรุ่นเก่า ที่ไม่รู้จัก NLINE_PAGE 
ก็จะใช้ค่าตาม LINE_PAGE เดิม 
แต่โปรแกรมรุ่นใหม่ก็จะใช้ค่า NLINE_PAGE 
ช่วยให้เราออกแบบ .FRM รองรับทั้งโปรแกรมรุ่นเก่าและรุ่นใหม่


อ้างถึง

คำสั่งที่สามารถใช้ได้ใน __FSYS.INI ได้แก่
LEAVETMPFILE
LOCALELBL
THAIPRINT
MONEYTXTSEP
15PAPER
OUTSCR
OUTPUT
ADJCPI
PRNCOPY
PGPAUSE
PORTNO
TTLPGPRINT
PGPRINT
PRINTGRID
FEED
THAIFEED
ONEPASS
COLOR
EMPHASIZE
LQ
QUALITY
LQFONT
BORDER
SHOWDATETIME
SHOWFRMNAME
SHOWTITLE
SHOWQUERY
WAITSETUP
EXPORTNAME
TITLEMASK
QUERYMASK
TXTCODE
PRNCODE
CSVCODE
CSVDATE
_MONEY
PAGENO_PIC
UNIDIR
PRINTNODATA
INSLEFTMARGIN
PAGELENGTH
PAGESUBLENGTH
LINE_PAGE
TTLPAGENO_ROW
TTLPAGENO_COL
PAGENO_ROW
PAGENO_COL
PAGEDATE_ROW
PAGEDATE_COL
PAGETIME_ROW
PAGETIME_COL
FRMCODE_ROW
FRMCODE_COL

และคำสั่ง set ค่าสี ในส่วนต่างๆของรายงาน
C.NORMAL
C.DATANUM+
C.DATANUM-
C.DATADATETIME
C.BORDER
C.PAGENO
C.TTLPAGENO
C.PAGEDATE
C.PAGETIME
C.FRMCODE
C.HEAD
C.DETAIL
C.FOOT
C.TITLE0
C.TITLE1
C.REPTITLE
C.COUNT
C.BRK
C.TOTAL
C.QUERY
}}}
{{{
ปรับปรุง 26 ก.ค. 2548

ใครที่เคยติดปัญหา
การวางระบบ คาใจอยู่นาน
อ่านเรื่องนี้จบ
รับรองว่าจะต้อง ซี๊ด..ปาก กัน   

ก่อนอื่นต้องขอยอมรับก่อน
ว่า เรื่องนี้ ไม่ได้อยู่ในแผนมาก่อน
และก่อนหน้านั้น 
ก็เคยคิดหาทางแก้ปัญหาอยู่นาน

อยู่ดีๆ พอ เพิ่มคำสั่ง _InvCfgHideList
ก็เลย ปิ๊งไอเดีย
ว่าเราน่าจะกำหนดได้ว่า
คำสั่งนี้ มีผลเฉพาะกับ user คนไหนได้

ตอนแรกก็ทำใช้ เฉพาะกับ _InvCfgHideList
พอทำแล้ว ก็เลยเปลี่ยน
ย้ายมันมาตรวจกันตอนที่ load ไฟล์ __FSYS.INI กันซะเลย
ทำให้มันกลายเป็นคำสั่งมาตรฐาน
สามารถใช้ควบกับคำสั่งอะไรก็ได้

ลองดูตัวอย่าง


โค๊ด:
<sale>@_InvCfgHideList = !S*
<acc>@_InvCfgHideList = !*
_InvCfgHideList = T*
แปลว่า 
สำหรับ user sale จะถูกซ่อนหมวดบิลทุกหมวด 
ยกเว้นหมวดที่ขึ้นต้นด้วย S
สำหรับ user acc จะเห็นหมวดบิลทุกหมวด
ส่วน user อื่นๆ ที่เหลือ จะไม่เห็นบิลหมวดที่ขึ้นต้นด้วย T

กลไกของมันเป็นอย่างนี้ครับ

สำหรับบรรทัดที่ขึ้นต้นด้วย
<.....>@
โปรแกรมจะตรวจสอบว่า
ไอ้ที่อยู่ใน <....> มันตรงกับ
ชื่อย่อ ของคนที่ login เข้ามาหรือเปล่า

ถ้าไม่ใช่
โปรแกรมก็จะ ตัดบรรทัดนั้นทั้งบรรทัดออกไป จากรายการ INI
แต่ถ้าใช่
โปรแกรมก็จะคงบรรทัดนั้นไว้ 
โดยตัดท่อน <....>@ ที่นำหน้าทิ้งไป

จากตัวอย่างข้างบน
ถ้าผู้ใช้คือ sale เป็นผู้ login เข้ามา

ก็จะได้ __FSYS.INI หลังจากกรองผู้ใช้แล้วดังนี้

โค๊ด:
_InvCfgHideList = !S*
_InvCfgHideList = T*
ตามกลไกปกติของ INI ในโปรแกรม คำสั่งที่มีผล ก็คือคำสั่งที่อยู่ก่อน

เราสามารถใช้ได้ แม้กระทั่งกับคำสั่ง #include
เช่น
<sale>@#include=salecfg.ini
<acc>@#include=acccfg.ini

ตอน ไอนสไตน์ คิด ทฤษฏี สัมพันธภาพ
ก็ไม่เคยคิดว่า e = mc2 
จะกลายเป็น ระเบิดปรมาณู

ขอให้พวกคุณ บรรเจิด เตลิดเปิดเปิง สักพัก
เอาโปรแกรมไป upgrade แล้ว set กันให้มันส์
ผสมกันกลายเป็น ระเบิด หรือ กลายเป็น โรงไฟฟ้า
ก็แล้วแต่ จินตนาการ นะครับ
ผมหว้งแต่ว่า จะได้มีเวลาพัก ทำอย่างอื่นบ้าง
 :idea:  :idea:
}}}

{{{
ปรับปรุง 1 ส.ค. 48

เพิ่มเติมอีกเล็กน้อย
แต่ประโยชน์ มหาศาล

สามารถใช้กับไฟล์ .INI, .FRM ทุกไฟล์

สามารถใช้ wildcard ภายใน <....>@ 
หรือใส่เป็น list ก็ได้ เช่น

<ADM,JOE>@SHOWCOST = OFF
<!ADM*>@SHOWCOST = 5

เพิ่มเงื่อนไขกับ รหัสแผนก ของผู้ใช้
<depcode:.....>@....
เช่น
<depcode:FIN*,ACC*>@....

เพิ่มเงื่อนไข ระดับผู้ใช้
<userlevel:...>@...
เช่น
<userlevel:1,2,3>@...

เพิ่มเงื่อนไข ชุดข้อมูล
<bdir:...>@...
เช่น
<bdir:DEMO>@...
}}}
{{{
13/1/2553

คำสั่ง _GlbLowDate

สามารถใช้ได้เหมือนกับคำสั่ง LockHiDate

แต่ _GlbLowDate จะมีผลกับข้อมูลทั้งหมด
สำหรับ บิลซื้อ, บิลขาย, ใบจ่าย, ใบเสร็จ สามารถกำหนดขยาย แบบเจาะจงเฉพาะหมวดบิลได้ 
เช่น _GlbLowDate.xx 

เราสามารถกำหนดวันที่ ที่ยินยอมให้ user เห็นข้อมูล ได้ 3 ลักษณะดังนี้

1. days count 
กำหนดจำนวนวัน ที่ยินยอมให้ย้อนหลัง นับจากวันที่ปัจจุบัน
เหมือนกับคำสั่ง LockHiDate เดิม
เช่น _GlbLowDate=7 แสดงข้อมูลย้อนหลังได้ 7 วัน
คำสั่งซ่อนนี้ มีผลกับ ระดับ user1 - ผู้จัดการ (ไม่มีผลกับ supervisor)
ยกเว้น เราบวกค่า bias สำหรับกำหนด ระดับ user ที่มีผลกับคำสั่งนี้ ดังนี้
+1000 ซ่อนข้อมูล สำหรับ ระดับต๋ำกว่า user1 (แปลว่า ไม่ซ่อนข้อมูล สำหรับ  user ทุกระดับ)
+2000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า user2 (ซ่อนสำหรับ user1)
+3000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า user3 (ซ่อนสำหรับ user1 - user2)
+4000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า user4 (ซ่อนสำหรับ user1 - user3)
+5000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า user5 (ซ่อนสำหรับ user1 - user4)
+6000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า user6 (ซ่อนสำหรับ user1 - user5)
+7000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า user7 (ซ่อนสำหรับ user1 - user6)
+8000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า บัญชี (ซ่อนสำหรับ user1 - user7)
+9000 ซ่อนข้อมูล สำหรับ ระดับต่ำกว่า ผู้จัดการ (ซ่อนสำหรับ user1 - บัญชี)
ตัวอย่างเช่น 
_GlbLowDate=9007 (คำสั่งซ่อน 7 วัน สำหรับ user1 - บัญชี)

2. absolute date
กำหนดวันที่แบบเจาะจง 
เช่น _GlbLowDate=1/1/53 ให้แสดงข้อมูล เริ่มตั้งแต่ วันที่ 1/1/53
คำสั่งนี้ มีผลกับ user1 - ผู้จัดการ
เราสามารถบวก bias ในค่า วัน เพื่อมีผลกับระดับ user ที่ต้องการ เช่นเดียวกันข้อ 1
เช่น _GlbLowDate=9001/1/53

นอกจากนี้ สามารถใช้ bias +1000 ในค่าเดือน เพื่อให้มีผลกับ user ทุกระดับ ไม่เว้นแม้แต่ supervisor
เช่น _GlbLowDate=1/1001/53

3. relative date
แบบกำหนด -วัน/-เดือน/-ปี ที่บวกลบ อ้างอิงจาก วันที่ปัจจุบัน
โดยใช้รูปแบบคล้ายวันที่ แต่มีค่าเป็นลบ 
เช่น _GlbLowDate=0/-1/0  ให้แสดงข้อมูลย้อนหลังไม่เกิน 1 เดือน
หรือ _GlbLowDate=-7/0/0 ให้แสดงข้อมูลย้อนหลังไม่เกิน 7 วัน (มีผลเช่นเดียวกับ _GlbLowDate=7)

กรณีใช้ bias เพื่อควบคุมระดับ user ที่ต้องการ
เช่น _GlbLowDate=9000/-1/0 หรือ -9007/0/0

ใช้ bias เพื่อให้มีผลกับทุก user
เช่น _GlbLowDate=0/-1001/0 หรือ _GlbLowDate=-7/1000/0

note:
กรณี days count การกำหนด จำนวนวันย้อนหลังจะใช้ค่าเป็นบวก ปกติ
แต่กรณี relative จะใช้ค่าเป็นลบ เพื่อย้อนหลัง

note: 
ผลจากการ update นี้ ทำให้คำสั่ง LockHiDate เดิม 
สามารถใช้รูปแบบการกำหนดค่า แบบเดียวกับ _GlbLowDate ได้อีกด้วย

update 15/1/53

bias พิเศษ
+10000 
  สำหรับ บิลขาย/บิลซื้อ กรณีหมวดบิลที่ใช้ผ่านรายการลูกหนี้/เจ้าหนี้ ให้แสดงบิลที่ค้างชำระด้วย
  สำหรับ ใบเสร็จ/ใบจ่าย กรณีหมวดบิลที่ใช้ชำระ ให้แสดงบิลที่ค้างชำระด้วย

+20000 
  สำหรับ ใบจ่าย/ใบเสร็จ กรณีหมวดบิลที่ใช้ผ่านรายการวางบิล ให้แสดงบิลที่ไม่มีวันที่นัดด้วย

bias พิเศษ ใช้บวกเพิ่มในค่า "วัน" 
เช่น _GlbLowDate=19007
หรือ _GlbLowDate=-10007/1000/0

note: สำหรับรายการชำระ บิลขาย/บิลซื้อ มีแนวคิด 2 แบบคือ
1. แสดงข้อมูล ตามเงื่อนไขวันที่ชำระ ของข้อมูลตัวเอง (default)
2. แสดงข้อมูลโดยอิงกับ วันที่ของ บิลขาย/บิลซื้อ ที่ refer อยู่ (ยังไม่ implement)

module suffix extension
ใช้สำหรับกำหนด _GlbLowDate ในระดับ module เช่น _GlbLowDate._Inv_
สามารถใช้ร่วมกับหมวดบิลได้ด้วย เช่น _GlbLowDate._Inv_.RC 
(แก้ปัญหากรณีมีการใช้รหัสหมวดบิลเหมือนกัน ในระบบต่างกัน เช่น ใบเสร็จ กับ บิลขาย ให้หมวด RC เหมือนกัน)

_Tax_ ภาษีมูลค่าเพิ่ม
_Stk_ สต็อก
_Cus_ ลูกค้า
_Vnd_ เจ้าหนี้
_Inv_ บิลขาย
_Po_ บิลซื้อ
_Rcp_ ใบเสร็จ
_Mrc_ ใบจ่าย
_Bnk_ บัญชีธนาคาร
_Chq_ เช็ค
_WTax_ หักภาษี ณ ที่จ่าย
_WS_ ใบค่าจ้าง
_Wkr_ พนักงาน
_GV_ บัญชีแยกประเภท
_SOBo_ ค้างส่ง
_SIBo_ ค้างรับ
_FA_ ทรัพย์สินถาวร

}}}

{{{
อ้างจาก: jojosati ที่ มกราคม 13, 2010, 01:02:46 pm
bias พิเศษ
+10000 
  สำหรับ บิลขาย/บิลซื้อ กรณีหมวดบิลที่ใช้ผ่านรายการลูกหนี้/เจ้าหนี้ ให้แสดงบิลที่ค้างชำระด้วย
  สำหรับ ใบเสร็จ/ใบจ่าย กรณีหมวดบิลที่ใช้ชำระ ให้แสดงบิลที่ค้างชำระด้วย

+20000 
  สำหรับ ใบจ่าย/ใบเสร็จ กรณีหมวดบิลที่ใช้ผ่านรายการวางบิล ให้แสดงบิลที่ไม่มีวันที่นัดด้วย


1. เมื่อใช้เป็นใบจ่าย/ใบเสร็จ ผ่านรายการไปชำระบิล ใบที่มีวันที่ก่อน LowDate ปกติจะถูกซ่อนทั้งหมด แต่สามารถสั่งให้ยกเว้น แสดงใบที่ค้างชำระ โดยใช้ +10000

ืnote: ใบที่ไม่มีวันที่นัด ย่อมไม่สามารถใส่รายละเอียดการชำระ ดังนั้นจึงเข้าเงื่อนไข เป็นใบที่ค้างชำระ แต่ความหมายของ เงื่อนไขค้างชำระ รวมถึงใบที่มีวันที่นัด แต่ยังไม่มีรายละเอียดการชำระด้วย

2. เมื่อใช้วางบิล ผ่านรายการไปปรับสถานะวางบิล ใบที่มีวันที่ก่อน LowDate ปกติจะถูกซ่อนทั้งหมด แต่สามารถสั่งให้ยกเว้น แสดงใบที่ไม่มีวันที่นัด  โดยใช้ +20000

3. เมื่อใช้เป็นใบลอยๆ ไม่ได้เชื่อมโยงไปผ่านรายการใดๆ bias พิเศษจะไม่มีผล ใบที่มีวันที่ก่อน LowDate ปกติจะถูกซ่อนทั้งหมด 

}}}
| รอเรียบเรียง |
{{{
28/11/2551

ขอ note ไว้ก่อน

ซ่อนรายชื่อสินค้าที่ไม่มีในสต็อค ตอนกด F1 ในบิลขาย และบิลซื้อ
ให้แสดงเฉพาะสินค้าที่ยอดคงเหลือเป็น 0

พิเศษคือ จะต้องกำหนดเงื่อนไขอื่นๆ เช่น
ซ่อนเฉพาะสินค้าที่ เคยมีรายการเข้า-ออก
ซ่อนเฉพาะสินค้า บางกลุ่ม

1-12-51
ทบทวนความสามารถเดิม
ความสามารถที่โปรแกรมทำได้อยู่แล้ว
1. เราสามารถใช้ {H} ในหมวดสินค้า หรือหน่วยนับ 
เพื่อซ่อนรายการสินค้าที่ไม่ต้องการให้แสดงตอนกด F1 เลือกสินค้าตอนเปิดบิล

2. เราสามารถกำหนดใน __FSYS.INI 
เพื่อสั่งซ่อนสินค้าบางหมวด หรือสินค้าบางชื่อ ตอนกด F1  
_StockHidePGroup=wildcard และ _StockHideUnit=wildcard
}}}

{{{
1/12/51

-หารือ หน่อยครับ ผมนอนคิดไป คิดมา คิดว่า น่าจะเพิ่มการกด แบบพิเศษ ด้วย
น่าจะดี เช่น กด Ctrl+F1(เป็นการเรียกแบบดั้งเดิม)  
เพราะบางครั้ง อาจต้องการดึงรายการ บางอย่างมาทำรายการ หรือตรวจสอบรายการปกติ
เวลาลูกค้ามีปัญหา หรือสอบถามได้ (สามารถปิด/เปิด ความสามารถนี้ได้ด้วย)

รบกวนฝากให้พิจารณาด้วยครับ

5/12/51

-อ้อ อย่าลืม แยกระหว่างบิลซื้อกับบิลขายด้วยครับ (ต้องรองรับแบบเฉพาะหมวดด้วย)
อิ อิ ขอทีเดียว เยอะ เลย
}}}

{{{
6-12-51
คำสั่งสำหรับซ่อนรายการสินค้า
_StockXListHide.#=mode,sign (wildcard)
เช่น
_StockXListHide.1 = 3
ซ่อนรายการสินค้าที่ยอดคงเหลือเป็น 0 ทุกกรณี
_StockXListHide.2 = 0x182,*สีดำ*
ซ่อนรายการสินค้าที่ยอดเป็น 0 แบบมี transaction และ มีคำว่า "สีดำ" ในชื่อสินค้า
_StockXListHide.3 = 0x102,*สีดำ*
ซ่อนรายการสินค้าที่ยอดเป็น 0 แบบมี transaction หรือ สินค้าที่มีคำว่า "สีดำ" ในชื่อสินค้า
_StockXListHide.4 = 0x100,!*สีแดง*
ซ่อนรายการสินค้าที่ไม่มีคำว่า "สีแดง" ในชื่อสินค้า


เราสามารถกำหนด # ของ xlist เตรียมไว้ก่อนได้ไม่จำกัด

ส่วนทางด้านบิลขาย
สามารถกำหนดเมนูเวลากด F1 เลือกสินค้าดังนี้
_InvXListStock=#xlist,ข้อความเมนู,#xlist,ข้อความเมนู,...
เช่น
_InvXListStock=0,สินค้าทั้งหมด,1,เฉพาะสินค้าที่เหลือในสต็อค,4,สินค้าสีแดง
ถ้าผู้ใช้เลือก "เฉพาะสินค้าที่เหลือในสต็อค"
โปรแกรมก็จะแสดงรายการสินค้าตามคำสั่ง _StockXListHide.1
ตามตัวอย่างข้างบน กำหนดไว้เป็น 3 นั่นคือ ให้ซ่อนรายการสินค้าที่จำนวนเป็น 0 ทั้งหมด


อ้างถึง
_StockXListHide.#=mode,sign (wildcard)
qty status
1 = hide if qty is zero trn (ซ่อน จำนวนเป็น 0 ที่มีรายการเข้า-ออก)
2 = hide if qty is zero non-trn (ซ่อน จำนวนเป็น 0 ที่ไม่มีรายการเข้าออก)
4 = hide if qty is negative (ซ่อนจำนวนติดลบ)
8 = hide if qty is positive (ซ่อนจำนวนบวก)

extend nstk
+16 (0x10) = exclusive mode ****
+32 (0x20) = hide if stockable (สินค้าที่ตัดสต็อกได้)
+64 (0x40) = hide if non-stockable (สินค้าที่ไม่ตัดสต็อค เช่น มี * ข้างหน้า หรือมี comment \ )

extend sign string checking
+128 (0x80) = exclusive mode ****
+256 (0x100) = hide if match wildcard in PName
+512 (0x200) = hide if match wildcard in PGroup
+1024 (0x400) = hide if match wildcard in UnitName
+2048 (0x800) = hide if match wildcard in  AltName

**** note: exclusive mode means
hide if qty status is TRUE 'and' extend checking also TRUE
example: 
mode 35 (3+32) = (not exclusive) hide if zero qty 'or' stockable 
  (hide all zero + hide all stockable)   -> show only 'non-zero with non-stockable' 
mode 51 (3+16 + 32) = (exclusive) hide if zero qty 'and' stockable
 (hide only zero in stockable product)  -> show all except 'zero with stockable' 


note เพิ่มเติม
ความสามารถสร้างเมนู ตัวเลือกสต็อค ในบิลขายจะต้องใช้ _Inv2_ หรือ _Inv3_ เท่านั้น ส่วน _Inv1_ ที่เป็นโปรแกรมมาตรฐานใช้ไม่ได้ ดูข้อแตกต่างระหว่าง _Inv1_, _Inv2_, _Inv3_ ได้ที่นี่
http://admthai.homeip.net/forum/index.php/topic,338.msg3117.html#msg3117

คำสั่งซ่อนที่ตรวจสอบจำนวน จะไม่ทำงานถ้ามีการใช้คำสั่ง StkKeepQty
http://admthai.homeip.net/forum/index.php/topic,530.0.html
และตรวจสอบจากจำนวนรวม(ที่ฟิลด์ข้อมูลสต็อค) โดยไม่มีการคำนวณแยกโกดัง

ไฟล์ที่ update
_Stk*.EXE
H_Stk*.EXE
_Inv*.EXE

สำหรับบิลซื้อ ใช้คำสั่งลักษณะเดียวกับบิลขายคือ

_PoXListStock=#xlist,ข้อความเมนู,#xlist,ข้อความเมนู,...


และจะต้องใช้ไฟล์ใหม่คือ _Po2_ แทน _Po1_ เดิม
}}}
{{{
2 กันยายน 2548

เพิ่มให้ใส่ ที่เก็บ หรือ โกดัง เข้าไปด้วย

ผลก็คือ
เปลี่ยนโครงสร้าง database ของระบบนี้
ใหม่ทั้งหมด

พูดง่ายๆ ก็คือ
รื้อใหม่หมด

ข้อมูลของเก่า
ต้องทิ้ง
สั่งสร้างแฟ้มข้อมูลใหม่ ทับไปเลย

การใช้งาน ทั่วๆ ไปก็ เหมือนเดิม

เพียงแต่ต้องระบุ ชื่อที่เก็บ (ถ้ามี)
ช่วยให้ สามารถตรวจนับ 
แยกทีละโกดัง

การ Post รายการปรับปรุงยอด
ไปที่บิลขาย
สามารถกำหนดให้
ผ่านรายการบิลอัตโนมัติ 
ในคราวเดียวกันได้ด้วย

สามารถสร้างรายการ
ปิดงวด เพื่อ Lock ยอดของโกดังนั้น
ได้อีกด้วย

คำสั่งใน __FSYS.INI ค่า default

_StkCntAdjDirect = ON
กำหนดว่า ยินยอม 
ให้สั่ง Post รายการปรับปรุง
ไปที่ รายการสินค้าออก โดยตรง 
โดยไม่ผ่านบิลขาย ได้หรือไม่

_StkCntMaxInvItm = 16
กำหนด จำนวนบรรทัด ต่อบิล 1 ใบ
เมื่อ Post รายการปรับปรุงไปบิลขาย
ถ้ากำหนดเป็น 0 ก็จะมีผล
ไม่ยินยอม ให้ Post ไปเข้าบิลขาย

_StkCntPostInv = OFF
กำหนดให้ ทำการ ผ่านรายการบิลขาย
เพื่อ ตัดสต๊อก ทันทีพร้อมกันไปเลยหรือไม่

_StkCntAddTrnClosed = OFF
กำหนดให้ สร้าง Closed Transaction
(รายการสินค้าเข้า ที่เลขที่เอกสารเป็น ดอกจัน
ชื่อสินค้าเป็นดอกจัน ใช้สำหรับ lock ไม่ให้ทำรายการย้อนหลัง)
อัตโนมัติ เมื่อ Adjust เสร็จหรือไม่
}}}
{{{
กำหนด __Fsys.ini

InvTabPic = 99
InvTabPicX = 99
InvTabWinRow = 8
InvTabWinCol = 10
InvTabMaxRow = 10
InvTabMaxCol = 10

_Inv_ = _Invcs_

การป้อนข้อมูลใน Stock \{frm=cs} (ถ้าจำไม่ผิด)
}}}

{{{
INVCS.TXT
_INVCS_: Inv module add-on

Usage:
_Inv1_ = _InvCS_

Inv Table Input
---------------
\{FMT=CS} -> <F1>
or
\{FMT=CC} -> <F1>

__FSYS.INI
----------
InvTabPic=99  ;item picture
InvTabPicX=99 ;total picture
InvTabWinRow=8 ;position of popup window
InvTabWinCol=10 ;position of popup window
InvTabMaxRow=10 ; max row
InvTabMaxCol=10 ; max col
}}}
{{{
_InvTypeAsLastPrice=""	// use same invtype to search last price
_InvPrcTabByDsc=OFF		// ON or 1 = use dsc for -xx, 2 = for +xx, 3= for - and +
_InvGetLastPrice=2		// scan last price for 2 last times
_InvUsePrcTab=-1000		// use prc tab for only exist customer/with loosely prc choice

InvUsePrcTab combine value
---------------------------
1000= use prc tab for only customer that exist in customer file
2000= disable if stock has no prctab use prctab name="*stkgrp" instead
4000= disable prctab (check for lastprice only)
1xx	= if customer has prcidx=0 then use prcidx=xx instead
	all value can combine by adding
	value can be negative for loosely price choice
	negative -> show all possible price of other prcidx
	positive -> show all price within prcidx only

InvGetLastPrice combine value
-----------------------------
deep bit:	0-3 = deep level (max=16) (0x01 - 0x0F)
			4-5 = reserved
			6	= skip same price (0x40)
			7	= prefer as default choice (0x80)
			8	= filter on salename (0x100)
			9	= filter on invbrief (0x200)
			10	= reserved
			11	= reserved
			12	= (0x1000) for '*' cusname (or empty) = not check (default is check)
			13	= (0x2000) for any cusname (except '*' or empty) = not check   (default is check)
			14  = (0x4000) for '*' filter (or empty) = not check (default is check)
			15	= (0x8000)- for any filter (except '*' or empty) = not check  (default is check)
}}}
[[ต้นทุนแฝง]] เป็นส่วนหนึ่งของ ~MOD-CIN ต้องการบันทึกค่าใช้จ่ายเกี่ยวกับการนำเข้า เอาไปบวกเพิ่มกับมูลค่าต้นทุนสินค้า 
[[ต้นทุนแฝง]]คือ การบันทึกค่าใช้จ่ายหรือต้นทุนเพิ่มเติมที่เกิดขึ้นภายหลัง กับสินค้าที่เข้าสต็อกไปแล้ว เช่น ค่าพิธีการศุลกากร, ค่าขนส่ง ฯลฯ ซึ่งค่าใช้จ่ายหรือต้นทุนเพิ่มเติมที่เกิดขึ้นนั้นอาจจำเป็นต้องคำนวณเฉลี่ยให้กับสินค้าหลายๆ ตัวที่เข้าสต็อกในล็อตเดียวกันนั้น เพื่อไปบวกเพิ่มเข้าเป็นต้นทุนแฝงให้กับสินค้านั้น 
!!!!คำอธิบายเพิ่มเติม เกี่ยวกับ concept ของ[[ต้นทุนแฝง]] (กุมภาพันธ์ 2548) 
การบันทึก[[ต้นทุนแฝง]] คือ การเปิดบิลซื้อต้นทุนแฝงใบใหม่ เพื่อนำยอดจำนวนเงินในบิลซื้อใหม่ ไปผนวกกับต้นทุนสินค้า ของบิลซื้อที่รับสินค้าเข้าสต็อกใบเดิม ดังนั้น ในบิลซื้อต้นทุนแฝงใบใหม่ จะต้องระบุเงื่อนไขอ้างถึงบิลซื้อใบเดิม ซึ่งอาจมีหลายใบ เช่น เลขที่บิล, วันที่ของบิลซื้อ ฯลฯ
* บิลซื้อ โปรแกรมใช้บิลซื้อทำการบันทึกค่าใช้จ่ายหรือต้นทุนเพิ่มที่เกิดขึ้น โดยภายในบิลซื้อสามารถกำหนดเงื่อนไข ที่สั่งให้โปรแกรมไปหารายการสินค้าที่ต้องการให้ทำการเฉลี่ยยอดต้นทุนแฝง 
** ในหมวดบิลซื้อที่ใช้ต้นทุนแฝงต้องใส่ ~X 
** ภายในบิลซื้อจะต้องอ้างถึงเลขที่เอกสาร ที่เป็นเลขที่เดียวกับบิลซื้อที่ใช้ผ่านรายการเข้าสต็อกใช้อ้างถึง เช่น ถ้าบิลซื้อหมวดที่ผ่านรายการเข้าสต็อกอ้างถึงเลขที่ใบกำกับภาษี บิลซื้อที่ใช้ผ่านต้นทุนแฝงก็ต้องอ้างถึงเลขที่ใบกำกับภาษีเหมือนกัน (ไม่ใช่เลขที่ของบิลซื้อ) 
** ใส่รายละเอียดหรือรายการของค่าใช้จ่ายที่เกิดขึ้น เหมือนกับบิลค่าใช้จ่ายปกติ เมื่อสั่งผ่านรายการ บิลซื้อจะไปตรวจหารายการเข้าสต็อก ที่มีเลขที่เอกสารตรงกับที่อ้างถึง นำมาคำนวณเฉลี่ยต้นทุนแฝง โดยใช้วิธีปัน cost ตามมูลค่าของสินค้า (นำมูลค่าทั้งหมดมารวมกัน เพื่อ weight สัดส่วนของมูลค่าต้นทุน) 
* """__FSys.INI""" จะต้องกำหนด XCostLevel=128 เพื่อเปิดให้โปรแกรมรับรู้การใช้งานต้นทุนแฝง 
* รายการเข้าสต็อก ที่หน้าจอ browse สำหรับสินค้าที่มีต้นทุนแฝง จะมีเครื่องหมาย + ขึ้นที่หน้าชื่อสินค้า และเมื่อเคาะ space โปรแกรมจะแสดงรายละเอียดของต้นทุนแฝงที่บันทึกไว้กับรายการสินค้าเข้าสต็อกดังกล่าว 
* รายงานสต็อกสินค้า(ใหม่) มีตัวอย่างรายงานเพิ่มเติม ที่แสดงยอดต้นทุนแบบคำนวณรวมต้นทุนแฝง 

ดูเพิ่มเติม [[ต้นทุนแฝง-advance1]]
!!!!เทคนิค การใช้งาน[[ต้นทุนแฝง]] advance (กุมภาพันธ์,เมษายน 2548) 
* [[ต้นทุนแฝง]] ภายในโปรแกรมแยกเป็น 3 level (เดิมมี 2 level)
** level 1 (~XCost) เป็นต้นทุนแฝงที่ "มี" ผลเปลี่ยนแปลงมูลค่าที่ใช้ในการคำนวณเฉลี่ยต้นทุนแฝงในครั้งถัดไป กรณีที่มีการทำต้นทุนแฝงมากกว่า 1 ครั้ง 
** level 2 (Expense) เป็นต้นทุนแฝงที่ "ไม่มี" ผลเปลี่ยนแปลงกับการคำนวณเฉลี่ยต้นทุนแฝงในครั้งถัดไป 
** level 3 (Deposit) เหมือนกับ level 2 ใช้สำหรับ deposit (ปรับปรุงเพิ่มเข้ามา เมษายน 2548) update ล่าสุด 23-4-2548 ตกลงการแก้ไขโปรแกรมเพื่อรองรับ deposit ทำดังนี้ บิลซื้อ/ผ่านรายการ สำหรับบรรทัดค่าใช้จ่ายที่มีคำว่า ''{XDEP}'' เมื่อผ่านรายการจะไปเข้ายอดต้นทุนแฝงชนิดใหม่ เป็นยอด Deposit 
* รายการเข้าสต็อค เมื่อกด space หรือ Enter เพื่อแสดงรายละเอียดของรายการเข้าสต็อค ในจอย่อยที่แสดงรายการต้นทุนแฝง เดิมจะมีคอลัมน์ของยอด ~XCost กับ Expense จะมีคอลัมน์เพิ่มอีกเป็นยอดของ Deposit แต่มูลค่าของต้นทุนต่อหน่วยเฉลี่ยที่คำนวณรวมต้นทุนแฝง จะคำนวณรวมเฉพาะยอดของ ~XCost กับ Expense 
* รายงานสต็อก เพิ่ม field ~IXDeposit (เฉลี่ย deposit ต่อหน่วย) และ ~IXDAmnt (มูลค่า deposit) 
* กรณีที่ต้องการให้ต้นทุนแฝงมีผลเหมือนกับเป็นต้นทุนจริงๆ เช่น ในรายงานสต็อก(ใหม่) ให้รายงานมาตรฐาน ทุกตัวใช้มูลค่าต้นทุนที่รวมมูลค่าแฝง แล้ว, หรือรายงานวิเคราะห์การขาย ให้เปรียบเทียบกับต้นทุนที่รวมมูลค่าแฝง สามารถกำหนดค่า XCostLevel ใน """__FSys.INI""" 
** 1 ให้รวมคำนวณมูลค่าแฝงของ ต้นทุน level1 
** 2 ให้รวมคำนวณมูลค่าแฝงของ ต้นทุน level2 
** 3 ให้รวมคำนวณมูลค่าแฝงของ ต้นทุนทั้ง 2 level 
** ปรับปรุงเพิ่ม เมษายน 2552 เพิ่ม level 4 - 7 ดูที่ XCostLevel 
* คำสั่งเงื่อนไขพิเศษ ภายในบิลซื้อ นอกจากกำหนดเลขที่อ้างถึง เพื่อเลือกล็อตของสินค้าที่ต้องการเฉลี่ยแล้ว เราสามารถกำหนดเงื่อนไขที่ซับซ้อนภายในบรรทัดรายการของบิลซื้อ แทนการกำหนดที่เลขที่อ้างถึงก็ได้ มีคำสั่งดังต่อไปนี้ 
> \''{DOC}[''wildcards'']'' 
> \''{DATE}[''99/99/99'']'' หรือ \''{DATE}[''99/99/99''-]'' หรือ \''{DATE}[''99/99/99''-''99/99/99'']'' 
> \''{PGRP}[''wildcards'']'' 
ตัวอย่าง 
> \ใบสั่งซื้อ''{DOC}[''I0523*,X0507*'']'' 
> \วันที่''{DATE}[''1/1/48''-]'' 
* กรณีที่มีเงื่อนไขมากกว่า 1 เงื่อนไข ทุกเงื่อนไขที่ระบุจะต้องเป็นจริงทั้งหมด 
** ''inclusive case'' บรรทัดที่มี \{+} ถ้าเงื่อนไขเฉพาะบรรทัดนั้นบรรทัดเดียวเป็นจริง ให้รวมข้อมูลสินค้าเข้านั้น มาคำนวณกระจายต้นทุน 
** ''exclusive case'' บรรทัดที่มี \{-} ถ้าเงื่อนไขเฉพาะบรรทัดนั้นบรรทัดเดียวเป็นจริง ให้ยกเว้นข้อมูลสินค้าเข้านั้น ออกจากรายการคำนวณกระจายต้นทุน 
** ''continue line'' (ใช้กับกรณีของ inclusive หรือ exclusive) กรณีที่ช่องรายการบรรทัดเดียวไม่พอ บรรทัดต่อไปใช้ \& จะมีผลให้เงื่อนไขของบรรทัดนั้นต่อกันเป็นบรรทัดเดียวกันกับบรรทัดก่อนหน้านั้น 
* กรณี เงื่อนไขของชื่อสินค้า (ชื่อสินค้าเต็มหรือ wildcard) เป็นกรณีพิเศษ ให้ระบุ \''{PNAME}'' ในบรรทัดก่อนหน้า แล้วบรรทัดต่อมาให้ระบุชื่อสินค้าในช่องรายการตามปกติ โดยไม่ต้องใส่ จำนวน หรือ ราคาต่อหน่วย 
* กรณียกเว้นบรรทัดที่ ไม่ต้องการใช้เป็นเงื่อนไขสินค้า ให้ใส่ \ ในช่องหน่วยนับ หรือใช้ * นำหน้าชื่อสินค้า 
กรณีที่มีคำสั่งเงื่อนไขกระจายต้นทุนภายในบิล โปรแกรมจะไม่ใช้ค่าในช่องอ้างถึง เป็นเงื่อนไขกรองรายการสินค้าเข้า 
* ชื่อรายการในบิลซื้อ ปกติจะผ่านรายการเข้าเป็นต้นทุนแฝง level 2 (ไม่มีผลต่อเปลี่ยนแปลงมูลค่าต้นทุนที่ใช้เป็นฐานคำนวณปัน cost)ยกเว้นรายการบรรทัดนั้น มีการระบุว่า ''{XCOST}'' อยู่ด้วย จึงผ่านรายการเข้าเป็นต้นทุนแฝง level 1 เช่น 
> *ค่าพิธีการภาษี ''{XCOST}'' 
> *ค่าล่วงเวลา ''{XCOST}'' 

ดูเพิ่มเติม [[ต้นทุนแฝง-advance2]]
!!!!ปรับปรุงเพิ่มเติม เกี่ยวกับการทำรายการ[[ต้นทุนแฝง]]ในบิลซื้อ (กุมภาพันธ์ 2548) 
การกำหนดจับคู่ ระหว่างรายการค่าต้นทุน กับเงื่อนไขเลือกกระจายต้นทุน ใช้ สัญญลักษณ์ ''!'' แล้วตามด้วยเลข ใส่ในช่องหน่วยนับ ของบรรทัดค่าต้นทุน และบรรทัดเงื่อนไข เช่น 
| ที่ | รายการ | หน่วยนับ | จำนวน | ราคา | เป็นเงิน |h
| 1 |ค่าพิธีการศุลกากร {XCOST} | | | 2500.-| 2500.-|
| 2 |DUTY FEE | ''!1'' | | 1000.-| 1000.-|
| 3 |DUTY FEE | ''!2'' | | 2000.-| 2000.-|
| 4 |ค่าน้ำร้อนน้ำชา | ''!!'' | | 500.-| 500.-|
| 5 |ใบรับสินค้า{DOC}[PO47012] | | | | |
| 6 |สินค้า IMPORT ABC | ''!1'' | | | |
| 7 |สินค้า IMPORT XYZ | ''!2'' | | | |
จากตัวอย่าง จะเห็นว่า 
* "ค่าพิธีการศุลกากร" ไม่มีกำหนดกลุ่ม ซึ่งจะกระจายต้นทุน ตามเงื่อนไข ที่ไม่มีระบุกลุ่ม นั่นคือ {DOC}[PO47012] 
* "ค่า DUTY FEE" ในบรรทัดที่ 2 ระบุเป็นกลุ่ม 1 จะต้องกระจายตามเงื่อนไขไม่ระบุกลุ่ม {DOC}[PO47012] แต่เพิ่มด้วยเงื่อนไขของกลุ่ม 1 คือ สินค้า IMPORT ABC 
* "ค่า DUTY FEE" ในบรรทัดที่ 3 ระบุเป็นกลุ่มที่ 2 จะเป็นการกระจายด้วยเงื่อนไข {DOC}[PO47012] และ สินค้า IMPORT XYZ 
* "ค่าน้ำร้อนน้ำชา" ในบรรทัดที่ 4 เป็นการระบุกลุ่มที่ผิดเงื่อนไข ซึ่งจะมีผลให้ยอดของค่าน้ำร้อนน้ำชา ในบรรทัดนี้ ไม่ถูกกระจายไปรายการใดทั้งสิ้น 
ปรับปรุง 4 กันยายน 2548

เพิ่มคำสั่งใน """__FSys.INI"""

[[_InvDspUnitNSize]]=4
สำหรับบังคับค่า ขนาด(กว้าง) ของคอลัมน์__หน่วยนับ__ ในจอบิลที่ป้อนข้อมูล

[[_InvDspDscNSize]]=8
สำหรับบังคับค่า ขนาด(กว้าง) ของคอลัมน์__ส่วนลด__ ในจอบิลที่ป้อนข้อมูล

ถ้ากำหนดเป็น 0 ก็จะมีผลทำให้คอลัมน์นั้นหายหายไป
แปลว่า user มองไม่เห็น แต่โปรแกรมยังมองเห็น 

เช่น ถ้าตอน F1 เลือกสินค้ามา หน่วยนับสินค้านั้นติดมาด้วย มีการตั้งไว้ว่า @NV ที่หน่วยนับ (ยกเว้นภาษี-No VAT) โปรแกรมก็ยังเห็น @NV ที่หน่วยนับนั้นได้ สินค้ารายการนั้น ก็คือ สินค้าที่ยกเว้นภาษี

> note: 
> ช่องรายการสินค้าจะ ปรับขนาดอัตโนมัติ 
> สามารใช้ควบคุม ในระดับหมวดบิลได้ เช่น [[_InvDspDscNSize]].IN = 0

> สำหรับบิลซื้อ ใช้ [[_PoDspUnitNSize]] [[_PoDspDscNSize]]
{{{
คำสั่งใน __FSys.INI

สำหรับผ่านรายการบิลขาย
_PInvAddNewCus = ON
ถ้าไม่พบชื่อลูกค้า โปรแกรมจะเพิ่มให้อัตโนมัติ

_PInvAddNewStk = ON
ถ้าไม่พบชื่อสินค้า โปรแกรมจะเพิ่มให้อัตโนมัติ

สำหรับผ่านรายการบิลซื้อ
_PPoAddNewVnd = ON
ถ้าไม่พบชื่อเจ้าหนี้ โปรแกรมจะเพิ่มให้อัตโนมัติ

_PPoAddNewStk = ON
ถ้าไม่พบชื่อสินค้า โปรแกรมจะเพิ่มให้อัตโนมัติ
}}}
{{{
MOD ผ่านรายการบิลขาย

ยี่ปั๊ว ใช้ฟิลด์ ชื่อลูกค้า
ลูกค้าย่อย ใช้ ฟิลด์ผู้ขาย

ตรวจสอบ ราคาของยี่ปั๊ว แต่ละราย ที่ขายให้ลูกค้าย่อย 
และฟ้องเตือนเมื่อเจอราคาไม่เท่ากัน
}}}

{{{
PInvA ใช้แทน PINV
ให้ตรวจสอบราคาขาย กับราคาซื้อครั้งก่อน (ต้องใส่ ~E ในชื่อบิล)

__FSYS.INI (default)
=====================
_PInvAddNewCus=OFF ;(for movinv)
_PInvAddNewStk=OFF  ;(for movinv)
_PInvECostType=4,5,5
_PInvECostLev=0,0,2003

_PInvECostInvType=xx,xx,xx

แปลความหมาย จับคู่ระหว่าง CostType กับ CostLev
4(0) -> ราคาซื้อสูงสุด (ย้อนไปจนถึงรายการซื้อแรกสุด)
5(0) -> ราคาซื้อต่ำสุด (ย้อนไปจนถึงรายการซื้อแรกสุด)
5(2003) -> ราคาขายต่ำสุด (ย้อนไม่เกิน 3 ครั้ง)


คำอธิบายเพิ่มเติม
=============
CostType
      -1  ไม่คำนวณ
      (0-3 ต้นทุนจริง) 0=เฉลี่ย 1=หลังสุด 2=มาตรฐาน 3=FIFO
      (4-8 ต้นทุนประเมินจากราคาซื้อ ข้อมูลเข้าสต๊อค หรือราคาขาย ขึ้นอยู่กับ Lev)
          4=สูงสุด 5=ต่ำสุด 6=เฉลี่ย 7=Mean 8=Median

กรณี costtype บวกด้วยค่าต่อไปนี้ เพื่อกำหนดเงื่อนไขการเปรียบเทียบ เพื่อฟ้อง error
0 (ไม่บวก) = ราคานั้น ต่ำกว่า ราคาที่คำนวณมาได้
1xx = ราคานั้น สูงกว่า ราคาที่คำนวณมาได้
2xx = ราคานั้น ไม่เท่ากับ ราคาที่คำนวณได้


CostLev
ส่วนของหลักสิบ/หน่วย
      lev=0 ค้นย้อนไปจนถึงแรกสุด (ไม่เกิน 500 รายการ)
      lev=1-99  ค้นย้อนไปตามค่าที่กำหนด

กรณีบวกด้วยหลักร้อย แสดง ราคาซื้อ/ส่วนลด
      กรณี 1xx เหมือนกับกรณี 0 - 99
          ยกเว้นการแสดงListที่มาของต้นทุนจะแสดงแบบ ราคาที่ขาย(ส่วนลด%)
      กรณี 2xx เหมือนกับกรณี 0 - 99 (ต้องใช้ R_INVB.EXE)
          ยกเว้นการแสดงListที่มาของต้นทุนจะย้อนไปค้นจากบิลซื้อแทน (หรือบิลขาย) หลักพันจะเป็นตัวกำหนด ว่าใช้บิลซื้อ หรือบิลขาย
      กรณี 3xx เหมือน 2xx การตรวจสอบชื่อสินค้า จะใช้ชื่อตามบิลขาย(ถ้ามี comment จะต้องมีชื่อที่มี comment ตรงกัน)

              
      กรณี 6xx,7xx เหมือนกับกรณี 2xx,3xx
          แต่รายการที่ไม่พบในบิลซื้อ(หรือบิลขาย)จะไม่นำมาคำนวณ

     

กรณีบวกด้วยหลักพัน
    กรณีไม่บวก จะเป็นการตรวจจากต้นทุนหรือราคาซื้อ
    กรณี  2xxx จะเป็นการตรวจจากราคาขายตามสต๊อคออก
    กรณี 12xxx จะเป็นการตรวจจากราคาขายตามสต๊อคออก ไม่เจาะจงลูกค้า

เพิ่มเติม 26/5/44 
  ใช้คู่กับ _PInvECostInvType=xx,xx,xx (กำหนดหมวดบิลที่ตรวจสอบ)
    กรณี  3xxx จะเป็นการตรวจจากราคาขายตามบิลขายโดยตรง(ไม่ตรวจผ่านสต๊อคออก)
    กรณี 13xxx จะเป็นการตรวจจากราคาขายตามบิลขายโดยตรง ไม่เจาะจงลูกค้า 

  ใช้คู่กับ _PInvECostPoType=xx,xx,xx
    กรณี 4xxx จะเป็นการตรวจจากราคาซื้อตามบิลซื้อโดยตรง 


15/9/49
    กรณี 5xxx จะเป็นการตรวจจากราคาขายตามบิลขายโดยตรง เจาะจงผู้ขาย และเจาะจงลูกค้า
    กรณี 15xxx จะเป็นการตรวจจากราคาขายตามบิลขายโดยตรง เจาะจงผู้ขาย แต่ไม่เจาะจงลูกค้า 
}}}

{{{
27/10/49

ทดสอบละเอียดวันนี้
ได้ข้อสรุป
1. ต้องใช้ PINVB (ที่แก้ไขล่าสุด) ไม่ใช่ PINVA
2. หมวดบิลต้องมี ~E
3. กำหนด __FSYS.INI
_PInvECostType=208 (ราคาไม่เท่ากับ Median) 
_PInvECostLev=15004 (ตรวจย้อน 4 บิลล่าสุด เฉพาะที่ชื่อผู้ขายตรง ไม่เจาะจงชื่อลูกค้า)

มีการปรับปรุง วิธีคิดค่า Median เดิม 
เอาจำนวนสินค้าที่ขายมาใช้เป็น ประชากร ซึ่งอาจมีปัญหา 
บิลที่ขาย volumn สูงลดราคาพิเศษ กลายเป็นราคาที่ถูกเลือก
เปลี่ยนใหม่เป็น ใช้นับจำนวนบิลแทน โดยไม่สนใจจำนวนสินค้า
ดังนั้นถ้าพบว่า มี 3 บิล ขาย 200 บาท บิลละตัว
แต่มี 1 บิลขาย 190 บาท 100 ตัว
ราคาที่เป็นค่า median คือ 200 เพราะมี 3 บิล มากกว่า 1 บิล
ถ้าคำนวณอย่างเดิม จะได้ราคา 190 เพราะมีจำนวน 100 ตัว มากกว่า 3 ตัว
}}}

{{{
วันศุกร์ 3/11/49 ไปรุ่งแสง 
กะจะปิดงานสักหน่อย ...
Up program ปุ๊ป เล่นให้ดูทันที ..
ผู้บริหาร 6 ท่านมามุงดู และประชุม
5 คนบอกเป็นเสียงเดียวกัน เยี่ยม ..!!! คราวนี้น่าจะช่วยทำให้งาน
ปัญหาน้อยลงทันที แต่... ช้าก่อน ผู้บริหารคนที่ 6
มีข้อท้วงติง ว่าน่าจะใช้ไม่ได้ ผมก็ถามว่าทำไม..
เขาบอกว่าข้อมูลของรายย่อยของเขา (ที่อยู่ช่องพนักงานขาย)
1 รายมีหลาย ๆ ชื่อ โดยแยกกันจาก {} เช่น
กิจรุ่งเรืองเมืองตรัง {พ}
กิจรุ่งเรืองเมืองตรัง {ป}
กิจรุ่งเรืองเมืองตรัง {ม}
ซึ่งตามข้อเท็จจริงคือรายเดียวกันหมด..
ผมเลยสรุปว่าทำงานยังนี้ไม่ได้ ต้องแก้ไขชื่อทั้งหมดให้ยุบเหลือแค่ 1
เพราะไม่งั้นโปรแกรมตรวจไม่ได้ .. 
เขาก็เลยบอกว่างั้นเขาก็ต้องเริ่มใหม่หมด และข้อมูลในเครื่องก็มีค่อนข้างมาก
คร่าว ๆ มีประมาณเกือบ พัน ชื่อ !!!
นั่นคือประเด็นที่ 1

คราวนี้มาเรื่องของการฟ้องตอนตรวจ
(ขอแทรกนิดหนึ่ง การผ่านรายการบิลแต่ละใบ ใช้เวลาประมาณเกือบ 30 วินาที)
โปรแกรมฟ้องกรณี Error ว่าราคาไม่เท่ากับ xxx และมีรายละเอียด
ให้ 1 บรรทัด เช่น 150(3/10/49 ล49100080),160(3/12/49 ล491200012)
ประมาณ นี้ เขาก็ถามว่าแล้วเป็นผู้ขาย(ยี่ปั๊ว/ลูกค้าตามโปรแกรม)รายไหน 
ผมบอกว่าก็ไปดูตามบิลที่โปรแกรมโชว์ว่ารายไหน ให้ไปค่อย ๆ เคาะดูทีละบิล
เขาบอกว่ายังนี้ก็ยังช้าอยู่ดี แทนที่จะทำงานได้เลย พนักงานเปิดบิลต้องไป
เข้าหน้าโน้นหน้านี้อีก ไม่สะดวกเท่าที่ควร .. สุดท้ายผมเสียเวลาไป 1 วัน
ไม่สามารถปิดงานได้ (วันนั้นนัดลูกค้าไว้ 3 ราย สุดท้ายแห้วหมด..เก็บเงิน
ไม่ได้อีกต่างหาก)..นั่นคือประเด็นที่ 2

สรุป...แล้ว.. หลังปรึกษาเฮียธงชัย อยากขอให้ช่วยปรับโปรแกรมเพิ่มเติม
1.สามารถกำหนดการตรวจ โดยมองชื่อรายย่อย(ช่องพนักงานขาย) ว่าให้ตัด
{xxx} ด้านหลังออกและถือว่าเป็นชื่อเดียวกัน ในการตรวจราคา..
2.หน้าจอฟ้อง Error ตอนตรวจสอบขอให้มีชื่อลูกค้าตามบิลขึ้นมาด้วย 
คนทำงานจะได้รู้ว่ายี่ปั๊วคนไหน ทำงานต่อได้ทันที่และเรียงเป็นบรรทัด
เพื่อสะดวกในการดูดังนี้ ตาม Format ดังนี้
1.ลูกค้าa ราคา(จำนวน วันที่ เลขที่บิล)
2.ลูกค้าb ราคา(จำนวน วันที่ เลขที่บิล)
รบกวนช่วยแก้ไขให้ด้วยครับ (จะได้ปิดงานซะที) งานนี้ค่า mo ผมคิดเขา 9,000

เพิ่มเติมนิดหนึ่ง ประวัติการขาย ทีเรากด F1 ช่องราคา
ถ้าสามารถ
1.ตรวจเฉพาะช่องพนักงานขายตรงกัน (โดยที่ตัดข้อมูล {xxx} แล้วมองเป็นราย
เดียวกันแบบที่สรุปให้แก้ไข) ก่อน
2.เวลาโชว์ราคา ให้ใช้ราคาหลังสุด พอกด F1 ให้แสดงประวัติ โดย มี 
ราคา,ชื่อลูกค้า,วันที่,เลขที่บิล ได้หรือไม่ ถ้าได้ละ เยี่ยม มากเลย สำหรับผู้บริหาร
ท่านนี้เพราะเขาไม่ยอมไปดูรายงาน(บอกว่าช้าต้องเข้าหน้านี้ออกหน้านั้น)และทำ
งานต่อเนื่องได้ทันที จะเอาไว้ดูราคาว่ารายย่อยรายนี้ มีปั๊วคนไหนขายบ้าง, และ
ขายราคาเท่าไร สะดวกสุด ๆ ฝากไว้เพิ่มเติมละกันครับ !!!
}}}

{{{
7/11/49

เพิ่มคำสั่งต่อไปนี้

StkEstFormatCus = mode
StkEstFormatSale = mode
StkEstFormatStk = mode

สำหรับตัดต่อชื่อ ก่อนที่จะทำการ compare


โค๊ด:
mode มีดังนี้

10 code only
40 name+comment (without code)
41 name without code & comment
42 comment only
43 code + name (without comm)
 
+100 = also strip hidden 

ตัวอย่างชื่อ A125|MY STOCK {+} \RED

10 = A125
40 = MY STOCK {+} \RED
41 = MY STOCK {+}
42 = \RED
43 = A125|MY STOCK {+}

100 = A125|MY STOCK  \RED
140 = MY STOCK  \RED
141 = MY STOCK
143 = A125|MY STOCK

กรณีรุ่งแสง จะต้องกำหนดเพิ่ม สั่งให้ตัด {xxx} ออก คือ
StkEstFormatSale = 100
}}}
{{{
void	Inv_getINI (char *invtype)
{
#ifndef	_MINIMUM_SIZE_
	char	*cache,*s ;

	if (invtype && _APIsEmptyField (STRING,invtype))
		invtype	= NULL ;

	//cache	= am_loadINICache (NULL) ;
	cache = am_cacheFSys () ;



	InvTypeAsLastPrice[0]= 0 ;
	InvAutoReferNo[0]= 0 ;
#ifdef	_LIMITSTATUS_MODIFY_
	InvAutoReferStatus = 0 ;
#endif
	InvChkCusLimit= OFF ;
	InvShipToAutoFill= ON ;
	InvDscAutoFill= OFF ;
	InvSaleAsLogin= OFF ;
	InvSaleAsCustomer= OFF ;
	InvAllowPostPrint= ON ;
	InvAllowPrintEdit= 7 ;
	InvPrcTabByDsc= OFF ;
	InvUsePrcTab= -1000 ;
	InvAllowChangeNo= ON ;
	InvAllowChangeQty= ON ;
	InvAllowChangePrice = ON ;
	InvAllowChangeDsc = ON ;
	InvGetLastPrice= -2 ;
	InvAutoPackItm = -1 ;
	InvDspDscNSize = -1 ;
	InvDspUnitNSize = -1 ;
	InvFixBug85 = 0 ;
	InvSaveReconcile = 0 ;
	InvAutoListBOM = 0 ;
	InvTraceCurLoc = OFF ;
#ifdef	_MULTILINE_MODIFY_
	InvMultilineEdit = 0 ;
#endif
	InvRemarkList[0]= 0 ;
#ifdef	_WF_MODIFY_
	InvWFAutoCmd[0]= 0 ;
#endif
	
	am_switchINICache (&InvShipToAutoFill,"_InvShipToAuto",invtype,cache) ;
	am_switchINICache (&InvDscAutoFill,"_InvDscAuto",invtype,cache) ;
	am_switchINICache (&InvSaleAsLogin,"_InvSaleASLogin",invtype,cache) ;
	am_switchINICache (&InvSaleAsCustomer,"_InvSaleASCustomer",invtype,cache) ;
	s	= am_switchINICache (&InvChkCusLimit,"_InvChkCusLimit",invtype,cache) ;
	if (s)
	{
		if (strcmpi(s,"ON")==0)
			InvChkCusLimit	= 9 ;
	}
	am_switchINICache (&InvAllowPostPrint,"_InvAllowPostPrint",invtype,cache) ;
	am_switchINICache (&InvAllowPrintEdit,"_InvAllowPrintEdit",invtype,cache) ;
	am_switchINICache (&InvAllowChangeNo,"_InvAllowChangeNo",invtype,cache) ;
	am_switchINICache (&InvAllowChangeQty,"_InvAllowChangeQty",invtype,cache) ;
	am_switchINICache (&InvAllowChangePrice,"_InvAllowChangePrice",invtype,cache) ;
	am_switchINICache (&InvAllowChangeDsc,"_InvAllowChangeDsc",invtype,cache) ;
	am_switchINICache (&InvPrcTabByDsc,"_InvPrcTabByDsc",invtype,cache) ;
	am_switchINICache (&InvUsePrcTab,"_InvUsePrcTab",invtype,cache) ;
	am_switchINICache (&InvGetLastPrice,"_InvGetLastPrice",invtype,cache) ;
	am_switchINICache (&InvAutoPackItm,"_InvAutoPackItm",invtype,cache) ;

	InvXCodeItm = -15 ;	// autofill customer & sale with confirm 
	if (((InvSaleAsLogin%10)==-1 || (InvSaleAsCustomer%10)==-1))
		InvXCodeItm = -3 ;	// autofill customer with confirm 
	am_switchINICache (&InvXCodeItm,"_InvXCodeItm",invtype,cache) ;
	am_switchINICache (&InvDspDscNSize,"_InvDspDscNSize",invtype,cache) ;
	am_switchINICache (&InvDspUnitNSize,"_InvDspUnitNSize",invtype,cache) ;

	s	= am_switchINICache (NULL,"_InvTypeAsLastPrice",invtype,cache) ;
	if (s)
	{
		_ResizeStr (s,2) ;
		strcpy (InvTypeAsLastPrice,s) ;
	}

	s	= am_switchINICache (NULL,"_InvAutoReferNo",invtype,cache) ;
	if (s)
	{
		if (strlen(s) > sizeof(InvAutoReferNo)-1)
			s[sizeof(InvAutoReferNo)-1] = 0 ;
//		_ResizeStr (s,10) ;
		strcpy (InvAutoReferNo,s) ;
	}
#ifdef	_LIMITSTATUS_MODIFY_
	s	= am_switchINICache (NULL,"_InvAutoReferStatus",invtype,cache) ;
	if (s)
		InvAutoReferStatus = Inv_parselimitstatus (s) ;
#endif
	am_switchINICache (&InvFixBug85,"_InvFixBug85",invtype,cache) ;
	am_switchINICache (&InvSaveReconcile,"_InvSaveReconcile",invtype,cache) ;
	am_switchINICache (&InvAutoListBOM,"_InvAutoListBOM",invtype,cache) ;
	am_switchINICache (&InvTraceCurLoc,"_InvTraceCurLoc",invtype,cache) ;
#ifdef	_MULTILINE_MODIFY_
	am_switchINICache (&InvMultilineEdit,"_InvMultilineEdit",invtype,cache) ;
#endif
	s	= am_switchINICache (NULL,"_InvRemarkList",invtype,cache) ;
	if (s)
	{
		if (strlen(s) > sizeof(InvRemarkList)-1)
			s[sizeof(InvRemarkList)-1] = 0 ;
		strcpy (InvRemarkList,s) ;
	}
#ifdef	_WF_MODIFY_
	s	= am_switchINICache (NULL,"_InvWFAutoCmd",invtype,cache) ;
	if (s)
	{
		if (strlen(s) > sizeof(InvWFAutoCmd)-1)
			s[sizeof(InvWFAutoCmd)-1] = 0 ;
		strcpy (InvWFAutoCmd,s) ;
	}
#endif
#endif	// MINIMUM
	//am_freeINICache (cache) ;
}
}}}

{{{
	s	= am_switchINICache (NULL,"_InvWFMenu",InvRec.data.InvType,am_cacheFSys()) ;
}}}

{{{
	InvMaxBItm	= -InvMaxItm ;	// force to use XMS
	if ((mdltype>=0 && mdltype<10) || mdltype==81)
	{
		char	*s ;
		int	max ;

		s	= am_getINI (NULL,"_InvMaxItm") ;
		if (s)
		{
			max	= atoi (s) ;
			InvMaxBItm	= -max ; // force to use XMS
		}
	}
}}}

{{{
	if (access (am_prgFileName ("IP-Pos",NULL),0)==0)
	{
		char	*cache ;
		
		//cache	= am_loadINICache (NULL) ;
		cache = am_cacheFSys () ;

		am_switchINICache (&pflag,"IPPosCall",NULL,cache) ;
		am_switchINICache (&pflag,"_IPPosCall",((char **)InvPaidXParm)[0],cache) ;
		
		//am_switchINICache (&dflag,"IPDrwCall",NULL,cache) ;
		//am_switchINICache (&dflag,"_IPDrwCall",NULL,cache) ;
		//am_freeINICache (cache) ;
	}
}}}

{{{
int	IPDrwCall (void)
{
	int	flag ;
	char	*cache ;
	char	sendstr[41],portstr[41] ;

	flag= TRUE ;
	strcpy (portstr,"com1,8,n,1") ;
	strcpy (sendstr,"00000000") ;

	//cache	= am_loadINICache (NULL) ;
	cache = am_cacheFSys () ;
	if (cache)
	{
		am_switchINICache (&flag,"IPDrwCall",NULL,cache) ;
		am_switchINICache (&flag,"_IPDrwCall",NULL,cache) ;
		if (flag)
		{
			am_strINICache (portstr,"IPDrwPort",NULL,cache) ;
			am_strINICache (portstr,"_IPDrwPort",NULL,cache) ;

			am_strINICache (sendstr,"IPDrwSendCmd",NULL,cache) ;
			am_strINICache (sendstr,"_IPDrwSendCmd",NULL,cache) ;

		}
		//am_freeINICache (cache) ;
	}
	if (flag)
	{
		if (strncmpi (portstr,"COM",3)==0)
		{
			IPDrw_com (portstr,sendstr) ;
			return (TRUE) ;
		}
	}
	return (FALSE) ;
}
}}}

{{{
void	IPPos_getINI (void)
{
	char	*cache ;	

	IPPos_PDet1[0]= IPPos_PDet2[0]= 0 ;
	IPPos_ChangesWait = 6 ;
#ifdef	_CUSBONUS_LINK_
	strcpy (IPPosBN_CusGrp,"*วฟฮจูก*") ;
	IPPosBN_Unit	= 5 ;
	IPPosBN_Step	= 100 ;
#endif
	//cache	= am_loadINICache (NULL) ;
	cache = am_cacheFSys () ;
	if (cache)
	{
		am_strINICache (IPPos_PDet1,"_IPPosPDet1",((char **)IPPos_XParm)[0],cache) ;
		am_strINICache (IPPos_PDet2,"_IPPosPDet2",((char **)IPPos_XParm)[0],cache) ;
		am_switchINICache (&IPPos_ChangesWait,"_IPPosChangesWait",((char **)IPPos_XParm)[0],cache) ;

#ifdef	_CUSBONUS_LINK_
		am_strINICache (IPPosBN_CusGrp,"_IPPosBNCusGrp",((char **)IPPos_XParm)[0],cache) ;
		am_valINICache (DOUBLE,&IPPosBN_Step,"_IPPosBNStep",((char **)IPPos_XParm)[0],cache) ;
		am_valINICache (DOUBLE,&IPPosBN_Unit,"_IPPosBNUnit",((char **)IPPos_XParm)[0],cache) ;
#endif
		//am_freeINICache (cache) ;
	}

}
}}}

{{{
void	PstInv_getINI (char *invtype)
{
	char	*s,*cache ;
#ifdef	_ESTCOSTCHK_
	int	i ;
#endif
	//cache	= am_loadINICache (NULL) ;
	cache = am_cacheFSys () ;

#ifndef	_NO_TAX_
	PstInvTaxForce = FALSE ;
	am_switchINICache (&PstInvTaxForce,"_PInvTaxForce",invtype,cache) ;

	PstInvTaxChkCus = FALSE ;
	am_switchINICache (&PstInvTaxChkCus,"_PInvTaxChkCus",invtype,cache) ;
#endif
	
#ifndef	_NO_CUS_
	PstInvAddNewCus	= OFF ;
	am_switchINICache (&PstInvAddNewCus,"_PInvAddNewCus",invtype,cache) ;
	PstInvChkCusLimit= 9 ;
	if ((s=am_switchINICache (&PstInvChkCusLimit,"_PInvChkCusLimit",invtype,cache))==NULL)
	{
		if (am_switchINICache (&PstInvChkCusLimit,"_InvChkCusLimit",invtype,cache))
		{
			if (PstInvChkCusLimit>0)
				PstInvChkCusLimit	= 0 ;
			else
				PstInvChkCusLimit	= - PstInvChkCusLimit ;
		}
	}
	else
	{
		if (strcmpi(s,"ON")==0)
			PstInvChkCusLimit	= 9 ;
	}
#endif

#ifndef	_NO_STK_
	PstInvAddNewStk	= OFF ;
	am_switchINICache (&PstInvAddNewStk,"_PInvAddNewStk",invtype,cache) ;

	PstInvMaxMat	= 20 ;
	if (am_switchINICache (&PstInvMaxMat,"_PInvMaxMat",invtype,cache))
	{
		if (PstInvMaxMat<8)
			PstInvMaxMat	= 8 ;
	}

	am_switchINICache (&PstRI_CrossChk,"_PStkCrossChk",NULL,cache) ;

#endif

#if	!defined(_NO_STK_) || !defined(_NO_CUS_)
	PstInvSkipTrn= OFF ;
	am_switchINICache (&PstInvSkipTrn,"_PInvSkipTrn",invtype,cache) ;
#endif
	
#ifdef	_ESTCOSTCHK_
	for (i=0; i<MAXECOSTLIST; i++)
	{
		PstInvECostTypeList[i]	= -1 ;
		if (i<2)
			PstInvECostTypeList[i]	= 4+i ;
		//if (i==2)
		//	PstInvECostTypeList[i]	= 4 ;
		PstInvECostLevList[i]	= 0 ;
		if (i==2)
		{
			PstInvECostTypeList[i]	= 5 ;
			PstInvECostLevList[i]	= 12003 ;
		}
		if (i==3)
		{
			PstInvECostTypeList[i]	= 5 ;
			PstInvECostLevList[i]	= 3703 ;
		}
	}

	s = am_switchINICache (NULL,"_PInvECostType",invtype,cache) ;
	if (s!=NULL)
	{
		char	buf[40] ;
		i	= 0 ;
		do
		{
			s	= _GetStrSeg (s,buf) ;
			PstInvECostTypeList[i]	= atoi(buf);
			i++ ;
			if (i>=MAXECOSTLIST)
				break ;
		} while (s);
		for (;i<MAXECOSTLIST; i++)
			PstInvECostTypeList[i]	= -1 ;
	}
	s = am_switchINICache (NULL,"_PInvECostLev",invtype,cache) ;
	if (s!=NULL)
	{
		char	buf[40] ;
		i	= 0 ;
		do
		{
			s	= _GetStrSeg (s,buf) ;
			PstInvECostLevList[i]	= atoi(buf);
			i++ ;
			if (i>=MAXECOSTLIST)
				break ;
		} while (s);
	}
#endif

#ifdef	_USE_BOSTKDET_
	PstInvChkStkBo = 0 ;
	am_switchINICache (&PstInvChkStkBo,"_PInvChkStkBo",NULL,cache) ;
#endif
	//am_freeINICache (cache) ;

}

}}}
{{{
เป็นระบบที่ modify ให้กับร้าน บ้านหนังสือ ชุมพร
ทำไว้ตั้งแต่สมัย 5-6 ปีก่อน
เรียกได้ว่า 
เป็นผู้บุกเบิกใช้ Billing + Barcode ทำเป็น POS รุ่นแรก

โจทย์ก็มีมีอยู่ว่า
ลูกค้าที่ซื้อหนังสือเขา ยอดทุก 100 บาท
จะให้ Bonus สะสม 5 คะแนน 
เท่ากับส่วนลด 5 บาท สำหรับการซื้อครั้งต่อไป
เท่านี้ เอง
แต่ทั้งหมด ใช้โปรแกรมควบคุมทั้งหมด
ตั้งแต่เก็บยอด หักยอดเมื่อเอามาใช้ลด

กลไกก็คือ 
ใช้หมวดลูกค้า 
เป็นตัวกำหนดว่า เป็น Member หรือไม่
ถ้าเป็น Member เมื่อรับชำระเงิน
ก็จะคำนวณสะสมยอด Bonus ให้อัตโนมัติ

ในขั้นตอนรับชำระเงิน 
สามารถดึงยอด Bonus สะสมมาใช้แทนเงินได้

ความยีดหยุ่นของโปรแกรมที่ทำไว้คือ

_IPPosBNCusGrp
ชื่อหมวดลูกค้า ที่เป็น Member 
สามารถกำหนดเป็น wildcard เองได้

ตัวเลขคำนวณเพื่อปันผล Bonus สามารถกำหนดเองได้
ประกับดัวย _IPPosBNStep และ _IPPosBNUnit
ตามโจทย์ ทุก 100 ให้ 5 จะต้องตั้งเป็น
_IPPosBNStep=100 และ _IPPosBNUnit=5
}}}
{{{
U_HongA 
โอนข้อมูล หงษไทย ISoft
8/10/2548

1. Set เมนูให้เรียกใช้โปรแกรมเพิ่ม
ไฟล์ MN_BL3.INI ใส่คำสั่งต่อไปนี้

U_HongA,100=โอนข้อมูล Hong - ISoft

เพื่อให้ผู้ใช้สามารถเรียกใช้ จากเมนู เบ็ดเตล็ด -> [ส่วนขยายเพิ่มระบบ]
ภายในโปรแกรม BILLING

2. __FSYS.INI
กำหนด Folder ที่ ISoft ส่งไฟล์ข้อมูลมาให้

ISOFTDATAPATH=C:\TEMPDATA	

3. ไฟล์ ISOFT1.ALI
ใช้สำหรับ Map รหัส ISoft เป็นชื่อลูกค้า/ชื่อสินค้า
เหมือนกับไฟล์ ALI ใน Genesis เช่น

พ-050 = ไพโรจน์การไฟฟ้า บจก.

*สำหรับรหัสลูกค้า ถ้าไ่ม่ได้่ Map ผ่าน ISOFT.ALI
โปรแกรมจะตรวจหารหัสจาก ข้อมูลลูกค้าเพิ่มเติม ใส่ส่วนของชื่อแทน
เช่น {C#พ-050}
เหมือนกับโปรแกรมเดิม ที่ส่งรหัสลูกค้าให้ ISOFT

note: format รหัสสินค้า ISOFT 
1-xx -yy = กล่อง xx ลอน yy
2-SB-yy = แผ่น ลอน yy
3-SR-xx = ลูกฟูกม้วน หน้า xx


=========================
note: อ้างอิงจาก ISOFT
=========================
                                    การโอน INVOICE เข้า SOFTCRAFT

1. รูปแบบ EXPORT FILE
    1.1 FILE NAME
          1.1.1 เป็นแบบ RUN ชื่อไปเรื่อยๆ
             1.1.2 รูปแบบ   YYRRRRRR  เช่น  05000001,05000002,05000003...
2. โครงสร้าง
     2.1 HEADER RECORD
          ลำดับ         ข้อมูล                          รายละเอียด
                1          H                        หมายถึง HEADER
            2          เลขที่                          
                3         สถานะ                        N=ปกติ   C=รายการยกเลิก
                4         วันที่เอกสาร                YYYY/MM/DD
            5         รหัสลูกค้า
                6         เครดิตเทอม                 
            7         ครบกำหนด                YYYY/MM/DD
            8         VAT RATE
            9         ยอดรวมสินค้า
              10        คำพูดส่วนลด              เช่น  3%
           11        ส่วนลด                       เป็นส่วนลดจำนวนเงิน
              12        ยอดเงินรวม                ยอดรวมสินค้า - ส่วนลด
              13        มูลค่า VAT
    2.2  DETAIL RECORD
         ลำดับ         ข้อมูล                          รายละเอียด
               1            D                      หมายถึง DETAIL
            2          ลำดับ                         1,2,3...
            3         รหัสสินค้า                  3.1  กล่อง    
                                                        1-XX-ลอน     
                                                                          XX = MODEL หรือสูตรกล่อง
                                                                 3.2  แผ่น
                                                                          2-SB-ลอน
                                                                 3.3  ลูกฟูกม้วน
                                                                           3-SR-XX
                                                          XX = หน้ากระดาษ  เช่น 48

  
    



3. ตัวอย่างข้อมูล
H,48090083,N,2005/09/02,พ-050,30,2005/10/02,7,23170.60,,0.00,23170.60,1621.94
D,1,1-08-B,552
D,2,1-08-B,559
D,3,1-01-BC,1093
H,48090084,N,2005/09/02,ค-028,30,2005/10/02,7,8211.00,3%,246.33,7964.67,557.53
D,1,2-SB-E,1050
H,48090085,N,2005/09/02,ค-028,30,2005/10/02,7,42210.00,3%,1266.30,40943.70,2866.06
D,1,2-SB-E,1280
D,2,2-SB-E,900
D,3,2-SB-E,2700
H,48090083,C,2005/09/02,พ-050,30,2005/10/02,7,23170.60,,0.00,23170.60,1621.94
D,1,1-08-B,552
D,2,1-08-B,559
D,3,1-01-BC,1093
H,48090085,N,2005/09/02,ค-028,30,2005/10/02,7,40824.00,3%,1224.72,39599.28,2771.95
D,1,2-SB-E,1280
D,2,2-SB-E,900
D,3,2-SB-E,2500

}}}
{{{
31/12/50
ลาทีปีเก่า สวัสดีปีใหม่

โจทย์:
บริษัทฯ ขายเข็มจักร อุตสาหกรรม
เสนอราคาลูกค้าเป็นเงินบาท แต่ใช้ฐานราคาเป็นเงินต่างประเทศ คูณด้วยอัตราแลกเปลี่ยน
ปัจจุบันใช้หลายสกุลเงิน ขึ้นอยู่กับว่าสินค้านั้นสั่งมาจากประเทศไหน เช่น USD JPY
เนื่องจากปัจจุบัน อัตราแลกเปลี่ยนมันลอยตัว เปลี่ยนแปลงขึ้นลงทุกวัน
ทำให้บริษัทฯ จำเป็นต้องคำนวณราคาใหม่ตามอัตราแลกเปลี่ยนทุกวัน เพื่อเสนอราคาลูกค้า
ซึ่งยุ่งยาก และมีโอกาสผิดพลาดสูง

วิธีแก้ปัญหา:
ข้อเสนอจาก adm...
แก้โปรแกรม ให้บริษัทฯ สามารถเปลี่ยนค่าอัตราแลกเปลียนใน __FSYS.INI
แล้วปรับโปรแกรมให้สามารถคำนวณราคาขาย โดยคูณกับอัตราแลกเปลี่ยนที่กำหนด

โปรแกรมเมอร์ก่ายหน้าผาก...
คิดไปคิดมา 2 คืน 3 วัน
แล้วก็ทำดังนี้

1. ปรับโปรแกรมสต็อก เพิ่มคำสั่ง _StockAltPrice=100 ใน __FSYS.INI
มีผลคือ กรณีที่ข้อมูลสต็อก ไม่ได้กำหนดราคาต่อหน่วย (ราคา=0)
โปรแกรมจะไปหาราคาต่อหน่วย จากตารางราคา ในข้อมูลสต็อกเพิ่มเติม (AltName)
ที่เป็นของระดับ 0 มาแสดง ตัวอย่างเช่น
0=12,-5%, -10%
แปลว่า ลูกค้าทั่วไปใช้ราคา 12 บาท
ลูกค้าระดับ 1 ลด 5% (จากระดับ 0)
ลูกค้าระดับ 2 ลด 10% (จากระดับ 0)

ปกติ ในบิลขาย สำหรับลูกค้าที่ไม่มีการกำหนดระดับให้ จะใช้ระดับ 0 อัตโนมัติ
ซึ่งถ้ามีการกำหนด ราคาของระดับ 0 ไว้ในตารางราคา 
ก็จะใช้ราคาในตารางราคาแทน ราคาในข้อมูลสินค้าอยู่แล้ว

ดังนั้นคำสั่งนี้ มีประโยชน์คือ ช่วยให้สามารถแสดงราคาขายจากตารางราคา ขึ้นมาบนจอ 
เมื่อเรากด F1 เพื่อ browse เลือกรายการสินค้า (กรณีที่สินค้านั้นไม่มีการตั้งราคาใน master stock)

หมายเหตุ: การแสดงราคาแทน จำกัดเฉพาะกรณี inline ตารางราคาในข้อมูลสต็อกเพิ่มเติมเท่านั้น
ไม่สามารถใช้ตารางราคา ที่ชี้ไปหา ชื่อตาราง ในแฟ้มข้อมูลตารางราคา

2. สร้างระบบ ตัวแปร ในตารางราคา
เพิ่มความสามารถให้กำหนดตัวแปร ไว้ในตารางราคาได้
โดยตัวแปรจะต้องตั้งดังนี้ {$ชื่อตัวแปร}($ชื่อตัวแปร) เช่น {$USD} ($USD)
แล้วใน __FSYS.INI ก็กำหนดค่าให้ตัวแปรนั้นเป็น 
$USD=33.2345

ก่อนที่จะตีความตารางราคา โปรแกรมจะถอดค่าตัวแปรก่อน
ตามตัวอย่างข้างต้น
ถ้าตารางราคาเป็น
0=2.1X{$USD} 0=2.1X($USD)
ก็จะได้เป็น 0=2.1X33.2345

ปิ๊งงงงงงงง บรรเจิดดดดด
เดี๋ยว adm ต้องขอตัวแปรไป ทำอย่างอื่น อีกแน่เลย... เฮ้อ
}}}

{{{
วันนี้เพิ่งเสร็จ

อัตรากำไร ก็ตั้งเป็นตัวแปรได้ ไม่ใช่ว่าจะตั้งแค่ อัตราแลกเปลี่ยน

model การตั้งค่าเท่าที่นึกได้เป็นดังนี้

ที่ข้อมูลสินค้า ในช่องราคาขาย ใส่ฐาน ที่เป็นเงิน ตปท. เช่น 2.00 (ดอลล่าร์)

ในข้อมูลเพิ่มเติมใส่ตางราคาเป็น 0={$USD}{$MARKUP1} ($USD)($MARKUP1)

ใน __FSYS.INI เราก็กำหนดไว้เป็น
$USD=X32.1223
$MARKUP1=+10%

เมื่อโปรแกรมแปลมาก็จะได้ 0=X32.1223+10%

ค่า MOD 500 USD ครับ

เรื่องรายงานตรวจสอบ ใช้รายงานสต็อก -> รายงานยอดคงเหลือ -> รายงานแสดงราคา ไงครับ มี field ตารางราคาให้ดูด้วย

และที่สำคัญ เราสามารถใช้ _InvPrcTabByDsc=7
ให้แสดงการคำนวณในช่องส่วนลด แทนการปรับราคาต่อหน่วย โดยไร้ร่องรอย

ดังนั้นเมื่อเปิดบิล ก็จะเห็นทันทีว่า สินค้ารายการนี้ ใช้ตัวคูณ ตัวบวกเท่าไหร่

อ้อ... เพื่อตอบโจทย์ล่วงหน้าว่า แล้วเวลาพิมพ์บิล ไม่อยากให้ช่องส่วนลดออกมาด้วย
แต่ราคาต่อหน่วย มันไม่สอดคล้องกับ จำนวนเงินรวม

ล่าสุด กำลังจะแก้ส่วนพิมพ์บิล เพิ่ม field QPriceN มาให้
เป็นช่องราคาต่อหน่วย ที่ไม่ใช่ ราคาต่อหน่วยที่ เก็บในข้อมูลบิล
แต่เป็นราคาต่อหน่วย ที่ได้จากการเอา จำนวนเงิน มาหารกับ จำนวนหน่วย โดยตรง
ดังนั้นจึงเป็นราคาต่อหน่วย ที่เกิดจากการหารเฉลี่ยจริงๆ ไม่ต้องแสดงส่วนลดอีก

จุดที่หลุดตอนนี้คือ 
รายงานสต็อก แสดงราคา นั้น ยังใช้ราคาจาก master สต็อก 
ไม่ได้ตรวจสอบราคา level 0 จาก ตารางราคา

2/1/50

update เพิ่มเติมแล้ว download ที่นี่

ปรับปรุง _StockAltPrice ให้สามารถคำนวน แบบอ้างถึง ชื่อตารางราคาแล้ว
(ตาม spec ของส่วนขยายตารางราคา ที่ใช้กับบิลขายนั่นเอง)

ในช่อง ตารางราคา ของข้อมูลเพิ่มเติม สามารถอ้างอิงเป็นชื่อตาราง
แล้วไปกำหนด ตารางราคา ในแฟ้มข้อมูล ตารางราคาอีกทีหนึ่ง

นอกจากนี้ กรณีที่ ไม่มีข้อมูลสต็อกเพิ่มเติม หรือ ไม่มีข้อมูล ในช่องตารางราคา ของข้อมูลเพิ่มเติม
โปรแกรมจะไปใช้ ตารางราคา ที่ตั้งชื่อตรงกับ ชื่อหมวดสินค้า (แต่เพิ่ม ดอกจัน นำหน้าชื่อหมวด)
เช่น สินค้าเป็นหมวด SOFTWARE โปรแกรมก็จะไม่หาตารางราคาที่ชื่อ *SOFTWARE
หรือแม้กระทั่ง สินค้าที่ไม่มีชื่อหมวด ก็จะไปหาตารางราคาที่ชื่อ "*" 
}}}

{{{
6/2/2551

ปรับ concept เพิ่มเติม

1. เปลี่ยน notation ของตัวแปรส่วนลด จาก {$ชื่อตัวแปร} เป็น ($ชื่อตัวแปร)

2. เพิ่มความหมายพิเศษสำหรับ ($$ชื่อตัวแปร) เป็นตัวแปรที่ไม่แสดงค่าจริง ให้ผู้ใช้เห็น เช่นกรณีเราใส่ค่า Margin ซึ่งไม่ต้องการให้พนักงานอื่นทราบว่ามีค่าเท่าไหร่

3. ส่วนลดแบบค้างตัวแปร $$ ถือเป็นส่วนลดแบบ ไม่สามารถใช้คำนวณในช่องส่วนลดต่อรายการได้ เช่นเดียวกับ กรณีส่วนลดที่กำหนดใน ตารางราคาแล้ว มีความยาวเกิน 10 ตัวอักษร ซึ่งทำให้มีผลดังนี้

- ในการแสดงราคาขายในบิล ไม่สามารถใช้แบบ ส่วนลดต่อรายการ คู่กับ ราคาเต็มที่ยังไม่ลด จะต้องใช้ราคาที่คำนวณส่วนลดแล้ว ถึงแม้ว่าจะกำหนด _InvPrcTabByDsc=7 ก็ไม่มีผล

4. สามารถกำหนด _InvPrcTabByDsc=0x10 หรือ 0x20 เพื่อให้แสดงข้อความส่วนลด ในบรรทัดถัดไป (โดยราคาขายในบรรทัดรายการสินค้า จะเป็นราคาที่คำนวณส่วนลดแล้ว)  
สำหรับ 0x20 แตกต่างคือ ข้อความส่วนลดที่แสดงจะมีเครื่องหมาย {....} จึงสามารถเห็นบนจอ แต่ไม่พิมพ์ออกมา
หรือ ถ้ากำหนดแบบผสมผสานคือ _InvPrcTabByDsc=0x17 หรือ 0x27 ถ้าส่วนลดเป็นแบบปกติ ก็แสดงแบบในบรรทัด แต่ถ้าเป็นส่วนลดแบบไม่สามารถแสดงในบรรทัด ก็จะแสดงที่บรรทัดถัดไป

ข้อจำกัด 
ส่วนลดในตารางราคา มีข้อจำกัด ยาวได้ไม่เกิน 30 ตัวอักษร
}}}
{{{
3/2/52

เพิ่มความสามารถของโปรแกรม
ในระหว่างป้อนข้อมูล สำหร้บ field ที่มีข้อความเป็น [xxxx]
เช่น
**ส่วนลด [3%]
ค่าไฟฟ้า มิเตอร [12332] - [10202]

โดยเราสามารถกด Ctrl-X
โปรแกรมจะแปลงส่วนที่อยู่ภายใน [...] ให้กลายเป็น sub field อัตโนมัติ

นอกจากนี้ เมื่ออยู่ในหน้าจอ sub field ยังมี key พิเศษ
เพื่อสลับค่าระหว่าง sub field ดังนี้
F2 Shift Left เอาค่าใน sub field ทางขวา ไปใส่ใน sub field ทางซ้าย
F3 Shift Right เอาค่าใน sub field ทางซ้าย ไปใส่ใน sub field ทางขวา
F4 Rotate Left ทำงานเหมือน Shift Left แต่แทนที่จะเว้นว่าง sub field ทางขวาท้ายสุด จะเอาค่าใน sub field ซ้ายแรกสุด มาใส่แทน
F5 Rotate Right ทำงานเหมือน Shift Right แต่แทนที่จะเว้นว่าง sub field ทางซ้ายแรกสุด จะเอาค่าใน sub field ขวาท้ายสุด มาใส่แทน

ดูตัวอย่าง
รูปที่ 1 ข้อความเดิม ใน field
รูปที่ 2 เมื่อกด Ctrl-X จะขึ้นเป็นจอ sub field
รูปที่ 3 ภายในจอ sub field เมื่อกด F3 เลื่อนเอาค่าใน field แรก ไปใส่ใน field ที่สอง
}}}

[img[Bracket2SubFld01.jpg]]
[img[Bracket2SubFld02.jpg]]
[img[Bracket2SubFld03.jpg]]